mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-25 15:14:48 +00:00
49438798b5
Signed-off-by: Roman Khimov <roman@nspcc.ru>
144 lines
3.4 KiB
Go
144 lines
3.4 KiB
Go
package manifest
|
|
|
|
import (
|
|
"crypto/elliptic"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"errors"
|
|
"slices"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/hash"
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
)
|
|
|
|
// Group represents a group of smartcontracts identified by a public key.
|
|
// Every SC in a group must provide signature of its hash to prove
|
|
// it belongs to the group.
|
|
type Group struct {
|
|
PublicKey *keys.PublicKey `json:"pubkey"`
|
|
Signature []byte `json:"signature"`
|
|
}
|
|
|
|
// Groups is just an array of Group.
|
|
type Groups []Group
|
|
|
|
type groupAux struct {
|
|
PublicKey string `json:"pubkey"`
|
|
Signature []byte `json:"signature"`
|
|
}
|
|
|
|
// IsValid checks whether the group's signature corresponds to the given hash.
|
|
func (g *Group) IsValid(h util.Uint160) error {
|
|
if !g.PublicKey.Verify(g.Signature, hash.Sha256(h.BytesBE()).BytesBE()) {
|
|
return errors.New("incorrect group signature")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// AreValid checks for groups correctness and uniqueness.
|
|
// If the contract hash is empty, then hash-related checks are omitted.
|
|
func (g Groups) AreValid(h util.Uint160) error {
|
|
if g == nil {
|
|
return errors.New("null groups")
|
|
}
|
|
if !h.Equals(util.Uint160{}) {
|
|
for i := range g {
|
|
err := g[i].IsValid(h)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
if len(g) < 2 {
|
|
return nil
|
|
}
|
|
pkeys := make(keys.PublicKeys, len(g))
|
|
for i := range g {
|
|
pkeys[i] = g[i].PublicKey
|
|
}
|
|
slices.SortFunc(pkeys, (*keys.PublicKey).Cmp)
|
|
for i := range pkeys {
|
|
if i == 0 {
|
|
continue
|
|
}
|
|
if pkeys[i].Cmp(pkeys[i-1]) == 0 {
|
|
return errors.New("duplicate group keys")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (g Groups) Contains(k *keys.PublicKey) bool {
|
|
return slices.ContainsFunc(g, func(gr Group) bool {
|
|
return k.Equal(gr.PublicKey)
|
|
})
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler interface.
|
|
func (g *Group) MarshalJSON() ([]byte, error) {
|
|
aux := &groupAux{
|
|
PublicKey: g.PublicKey.StringCompressed(),
|
|
Signature: g.Signature,
|
|
}
|
|
return json.Marshal(aux)
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface.
|
|
func (g *Group) UnmarshalJSON(data []byte) error {
|
|
aux := new(groupAux)
|
|
if err := json.Unmarshal(data, aux); err != nil {
|
|
return err
|
|
}
|
|
b, err := hex.DecodeString(aux.PublicKey)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pub := new(keys.PublicKey)
|
|
if err := pub.DecodeBytes(b); err != nil {
|
|
return err
|
|
}
|
|
g.PublicKey = pub
|
|
if len(aux.Signature) != keys.SignatureLen {
|
|
return errors.New("wrong signature length")
|
|
}
|
|
g.Signature = aux.Signature
|
|
return nil
|
|
}
|
|
|
|
// ToStackItem converts Group to stackitem.Item.
|
|
func (g *Group) ToStackItem() stackitem.Item {
|
|
return stackitem.NewStruct([]stackitem.Item{
|
|
stackitem.NewByteArray(g.PublicKey.Bytes()),
|
|
stackitem.NewByteArray(g.Signature),
|
|
})
|
|
}
|
|
|
|
// FromStackItem converts stackitem.Item to Group.
|
|
func (g *Group) FromStackItem(item stackitem.Item) error {
|
|
if item.Type() != stackitem.StructT {
|
|
return errors.New("invalid Group stackitem type")
|
|
}
|
|
group := item.Value().([]stackitem.Item)
|
|
if len(group) != 2 {
|
|
return errors.New("invalid Group stackitem length")
|
|
}
|
|
pKey, err := group[0].TryBytes()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
g.PublicKey, err = keys.NewPublicKeyFromBytes(pKey, elliptic.P256())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sig, err := group[1].TryBytes()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(sig) != keys.SignatureLen {
|
|
return errors.New("wrong signature length")
|
|
}
|
|
g.Signature = sig
|
|
return nil
|
|
}
|