mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-10 05:54:04 +00:00
a1a7e3d708
Signed-off-by: Roman Khimov <roman@nspcc.ru>
252 lines
7 KiB
Go
252 lines
7 KiB
Go
package context
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"slices"
|
|
"strings"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto"
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
|
)
|
|
|
|
// TransactionType is the ParameterContext Type used for transactions.
|
|
const TransactionType = "Neo.Network.P2P.Payloads.Transaction"
|
|
|
|
// compatTransactionType is the old, 2.x type used for transactions.
|
|
const compatTransactionType = "Neo.Core.ContractTransaction"
|
|
|
|
// ParameterContext represents smartcontract parameter's context.
|
|
type ParameterContext struct {
|
|
// Type is a type of a verifiable item.
|
|
Type string
|
|
// Network is a network this context belongs to.
|
|
Network netmode.Magic
|
|
// Verifiable is an object which can be (de-)serialized.
|
|
Verifiable crypto.VerifiableDecodable
|
|
// Items is a map from script hashes to context items.
|
|
Items map[util.Uint160]*Item
|
|
}
|
|
|
|
type paramContext struct {
|
|
Type string `json:"type"`
|
|
Net uint32 `json:"network"`
|
|
Hash util.Uint256 `json:"hash,omitempty"`
|
|
Data []byte `json:"data"`
|
|
Items map[string]json.RawMessage `json:"items"`
|
|
}
|
|
|
|
type sigWithIndex struct {
|
|
index int
|
|
sig []byte
|
|
}
|
|
|
|
// NewParameterContext returns ParameterContext with the specified type and item to sign.
|
|
func NewParameterContext(typ string, network netmode.Magic, verif crypto.VerifiableDecodable) *ParameterContext {
|
|
return &ParameterContext{
|
|
Type: typ,
|
|
Network: network,
|
|
Verifiable: verif,
|
|
Items: make(map[util.Uint160]*Item),
|
|
}
|
|
}
|
|
|
|
// GetCompleteTransaction clears transaction witnesses (if any) and refills them with
|
|
// signatures from the parameter context.
|
|
func (c *ParameterContext) GetCompleteTransaction() (*transaction.Transaction, error) {
|
|
tx, ok := c.Verifiable.(*transaction.Transaction)
|
|
if !ok {
|
|
return nil, errors.New("verifiable item is not a transaction")
|
|
}
|
|
if len(tx.Scripts) > 0 {
|
|
tx.Scripts = tx.Scripts[:0]
|
|
}
|
|
for i := range tx.Signers {
|
|
w, err := c.GetWitness(tx.Signers[i].Account)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't create witness for signer #%d: %w", i, err)
|
|
}
|
|
tx.Scripts = append(tx.Scripts, *w)
|
|
}
|
|
return tx, nil
|
|
}
|
|
|
|
// GetWitness returns invocation and verification scripts for the specified contract.
|
|
func (c *ParameterContext) GetWitness(h util.Uint160) (*transaction.Witness, error) {
|
|
item, ok := c.Items[h]
|
|
if !ok {
|
|
return nil, errors.New("witness not found")
|
|
}
|
|
bw := io.NewBufBinWriter()
|
|
for i := range item.Parameters {
|
|
if item.Parameters[i].Type != smartcontract.SignatureType {
|
|
return nil, fmt.Errorf("unsupported %s parameter #%d", item.Parameters[i].Type.String(), i)
|
|
} else if item.Parameters[i].Value == nil {
|
|
return nil, fmt.Errorf("no value for parameter #%d (not signed yet?)", i)
|
|
}
|
|
emit.Bytes(bw.BinWriter, item.Parameters[i].Value.([]byte))
|
|
}
|
|
return &transaction.Witness{
|
|
InvocationScript: bw.Bytes(),
|
|
VerificationScript: item.Script,
|
|
}, nil
|
|
}
|
|
|
|
// AddSignature adds a signature for the specified contract and public key.
|
|
func (c *ParameterContext) AddSignature(h util.Uint160, ctr *wallet.Contract, pub *keys.PublicKey, sig []byte) error {
|
|
item := c.getItemForContract(h, ctr)
|
|
if _, pubs, ok := vm.ParseMultiSigContract(ctr.Script); ok {
|
|
if item.GetSignature(pub) != nil {
|
|
return errors.New("signature is already added")
|
|
}
|
|
pubBytes := pub.Bytes()
|
|
var contained = slices.ContainsFunc(pubs, func(p []byte) bool {
|
|
return bytes.Equal(pubBytes, p)
|
|
})
|
|
if !contained {
|
|
return errors.New("public key is not present in script")
|
|
}
|
|
item.AddSignature(pub, sig)
|
|
if len(item.Signatures) >= len(ctr.Parameters) {
|
|
indexMap := map[string]int{}
|
|
for i := range pubs {
|
|
indexMap[hex.EncodeToString(pubs[i])] = i
|
|
}
|
|
sigs := make([]sigWithIndex, len(item.Parameters))
|
|
var i int
|
|
for pub, sig := range item.Signatures {
|
|
sigs[i] = sigWithIndex{index: indexMap[pub], sig: sig}
|
|
i++
|
|
if i == len(sigs) {
|
|
break
|
|
}
|
|
}
|
|
slices.SortFunc(sigs, func(a, b sigWithIndex) int {
|
|
return a.index - b.index
|
|
})
|
|
for i := range sigs {
|
|
item.Parameters[i] = smartcontract.Parameter{
|
|
Type: smartcontract.SignatureType,
|
|
Value: sigs[i].sig,
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
index := -1
|
|
for i := range ctr.Parameters {
|
|
if ctr.Parameters[i].Type == smartcontract.SignatureType {
|
|
if index >= 0 {
|
|
return errors.New("multiple signature parameters in non-multisig contract")
|
|
}
|
|
index = i
|
|
}
|
|
}
|
|
if index != -1 {
|
|
item.Parameters[index].Value = sig
|
|
} else if !ctr.Deployed {
|
|
return errors.New("missing signature parameter")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *ParameterContext) getItemForContract(h util.Uint160, ctr *wallet.Contract) *Item {
|
|
item, ok := c.Items[ctr.ScriptHash()]
|
|
if ok {
|
|
return item
|
|
}
|
|
params := make([]smartcontract.Parameter, len(ctr.Parameters))
|
|
for i := range params {
|
|
params[i].Type = ctr.Parameters[i].Type
|
|
}
|
|
script := ctr.Script
|
|
if ctr.Deployed {
|
|
script = nil
|
|
}
|
|
item = &Item{
|
|
Script: script,
|
|
Parameters: params,
|
|
Signatures: make(map[string][]byte),
|
|
}
|
|
c.Items[h] = item
|
|
return item
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler interface.
|
|
func (c ParameterContext) MarshalJSON() ([]byte, error) {
|
|
verif, err := c.Verifiable.EncodeHashableFields()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to encode hashable fields")
|
|
}
|
|
items := make(map[string]json.RawMessage, len(c.Items))
|
|
for u := range c.Items {
|
|
data, err := json.Marshal(c.Items[u])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
items["0x"+u.StringLE()] = data
|
|
}
|
|
pc := ¶mContext{
|
|
Type: c.Type,
|
|
Net: uint32(c.Network),
|
|
Hash: c.Verifiable.Hash(),
|
|
Data: verif,
|
|
Items: items,
|
|
}
|
|
return json.Marshal(pc)
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler interface.
|
|
func (c *ParameterContext) UnmarshalJSON(data []byte) error {
|
|
pc := new(paramContext)
|
|
if err := json.Unmarshal(data, pc); err != nil {
|
|
return err
|
|
}
|
|
|
|
var verif crypto.VerifiableDecodable
|
|
switch pc.Type {
|
|
case compatTransactionType, TransactionType:
|
|
tx := new(transaction.Transaction)
|
|
verif = tx
|
|
default:
|
|
return fmt.Errorf("unsupported type: %s", c.Type)
|
|
}
|
|
err := verif.DecodeHashableFields(pc.Data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
items := make(map[util.Uint160]*Item, len(pc.Items))
|
|
for h := range pc.Items {
|
|
u, err := util.Uint160DecodeStringLE(strings.TrimPrefix(h, "0x"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
item := new(Item)
|
|
if err := json.Unmarshal(pc.Items[h], item); err != nil {
|
|
return err
|
|
}
|
|
items[u] = item
|
|
}
|
|
if !pc.Hash.Equals(util.Uint256{}) {
|
|
if !verif.Hash().Equals(pc.Hash) {
|
|
return fmt.Errorf("hash parameter doesn't match calculated verifiable hash: %s vs %s", pc.Hash.StringLE(), verif.Hash().StringLE())
|
|
}
|
|
}
|
|
c.Type = pc.Type
|
|
c.Network = netmode.Magic(pc.Net)
|
|
c.Verifiable = verif
|
|
c.Items = items
|
|
return nil
|
|
}
|