2018-02-04 19:54:51 +00:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
2018-03-25 10:45:54 +00:00
|
|
|
"crypto/sha256"
|
2018-02-04 19:54:51 +00:00
|
|
|
"encoding/hex"
|
2018-03-23 20:36:59 +00:00
|
|
|
"encoding/json"
|
2018-03-02 15:24:09 +00:00
|
|
|
"fmt"
|
2018-03-25 10:45:54 +00:00
|
|
|
|
|
|
|
"golang.org/x/crypto/ripemd160"
|
2018-02-04 19:54:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const uint160Size = 20
|
|
|
|
|
|
|
|
// Uint160 is a 20 byte long unsigned integer.
|
|
|
|
type Uint160 [uint160Size]uint8
|
|
|
|
|
2018-03-02 15:24:09 +00:00
|
|
|
// Uint160DecodeString attempts to decode the given string into an Uint160.
|
|
|
|
func Uint160DecodeString(s string) (u Uint160, err error) {
|
|
|
|
if len(s) != uint160Size*2 {
|
|
|
|
return u, fmt.Errorf("expected string size of %d got %d", uint160Size*2, len(s))
|
|
|
|
}
|
|
|
|
b, err := hex.DecodeString(s)
|
|
|
|
if err != nil {
|
|
|
|
return u, err
|
|
|
|
}
|
|
|
|
return Uint160DecodeBytes(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Uint160DecodeBytes attempts to decode the given bytes into an Uint160.
|
|
|
|
func Uint160DecodeBytes(b []byte) (u Uint160, err error) {
|
|
|
|
if len(b) != uint160Size {
|
|
|
|
return u, fmt.Errorf("expected byte size of %d got %d", uint160Size, len(b))
|
|
|
|
}
|
|
|
|
for i := 0; i < uint160Size; i++ {
|
|
|
|
u[i] = b[i]
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-25 10:45:54 +00:00
|
|
|
// Uint160FromScript returns a Uint160 type from a raw script.
|
|
|
|
func Uint160FromScript(script []byte) (u Uint160, err error) {
|
|
|
|
sha := sha256.New()
|
|
|
|
sha.Write(script)
|
|
|
|
b := sha.Sum(nil)
|
|
|
|
ripemd := ripemd160.New()
|
|
|
|
ripemd.Write(b)
|
|
|
|
b = ripemd.Sum(nil)
|
|
|
|
return Uint160DecodeBytes(b)
|
|
|
|
}
|
|
|
|
|
2018-03-02 15:24:09 +00:00
|
|
|
// Bytes returns the byte slice representation of u.
|
|
|
|
func (u Uint160) Bytes() []byte {
|
2018-02-04 19:54:51 +00:00
|
|
|
b := make([]byte, uint160Size)
|
|
|
|
for i := 0; i < uint160Size; i++ {
|
|
|
|
b[i] = byte(u[i])
|
|
|
|
}
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2018-03-25 10:45:54 +00:00
|
|
|
// BytesReverse return a reversed byte representation of u.
|
|
|
|
func (u Uint160) BytesReverse() []byte {
|
|
|
|
return ArrayReverse(u.Bytes())
|
|
|
|
}
|
|
|
|
|
2018-02-04 19:54:51 +00:00
|
|
|
// String implements the stringer interface.
|
|
|
|
func (u Uint160) String() string {
|
2018-03-02 15:24:09 +00:00
|
|
|
return hex.EncodeToString(u.Bytes())
|
2018-02-04 19:54:51 +00:00
|
|
|
}
|
2018-02-07 14:16:50 +00:00
|
|
|
|
|
|
|
// Equals returns true if both Uint256 values are the same.
|
|
|
|
func (u Uint160) Equals(other Uint160) bool {
|
2018-03-02 15:24:09 +00:00
|
|
|
for i := 0; i < uint160Size; i++ {
|
|
|
|
if u[i] != other[i] {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
2018-02-07 14:16:50 +00:00
|
|
|
}
|
2018-03-23 20:36:59 +00:00
|
|
|
|
|
|
|
// MarshalJSON implements the json marshaller interface.
|
|
|
|
func (u Uint160) MarshalJSON() ([]byte, error) {
|
2018-03-30 06:15:03 +00:00
|
|
|
return json.Marshal(
|
|
|
|
fmt.Sprintf("0x%s", hex.EncodeToString(ArrayReverse(u.Bytes()))),
|
|
|
|
)
|
2018-03-23 20:36:59 +00:00
|
|
|
}
|