/* Package actor provides a way to change chain state via RPC client. This layer builds on top of the basic RPC client and [invoker] package, it simplifies creating, signing and sending transactions to the network (since that's the only way chain state is changed). It's generic enough to be used for any contract that you may want to invoke and contract-specific functions can build on top of it. */ package actor import ( "errors" "fmt" "github.com/nspcc-dev/neo-go/pkg/config/netmode" "github.com/nspcc-dev/neo-go/pkg/core/state" "github.com/nspcc-dev/neo-go/pkg/core/transaction" "github.com/nspcc-dev/neo-go/pkg/neorpc/result" "github.com/nspcc-dev/neo-go/pkg/rpcclient/invoker" "github.com/nspcc-dev/neo-go/pkg/rpcclient/waiter" "github.com/nspcc-dev/neo-go/pkg/util" "github.com/nspcc-dev/neo-go/pkg/vm/vmstate" "github.com/nspcc-dev/neo-go/pkg/wallet" ) var ( // ErrExecFailed is returned from [Actor.WaitSuccess] when transaction // is accepted into a block, but its execution ended up in non-HALT VM // state. ErrExecFailed = errors.New("execution failed") ) // RPCActor is an interface required from the RPC client to successfully // create and send transactions. type RPCActor interface { invoker.RPCInvoke // CalculateNetworkFee calculates network fee for the given transaction. // // CalculateNetworkFee MUST NOT call state-changing methods (like Hash or Size) // of the transaction through the passed pointer: make a copy if necessary. CalculateNetworkFee(tx *transaction.Transaction) (int64, error) GetBlockCount() (uint32, error) GetVersion() (*result.Version, error) SendRawTransaction(tx *transaction.Transaction) (util.Uint256, error) } // SignerAccount represents combination of the transaction.Signer and the // corresponding wallet.Account. It's used to create and sign transactions, each // transaction has a set of signers that must witness the transaction with their // signatures. type SignerAccount struct { Signer transaction.Signer Account *wallet.Account } // Actor keeps a connection to the RPC endpoint and allows to perform // state-changing actions (via transactions that can also be created without // sending them to the network) on behalf of a set of signers. It also provides // an Invoker interface to perform test calls with the same set of signers. // // Actor-specific APIs follow the naming scheme set by Invoker in method // suffixes. *Call methods operate with function calls and require a contract // hash, a method and parameters if any. *Run methods operate with scripts and // require a NeoVM script that will be used directly. Prefixes denote the // action to be performed, "Make" prefix is used for methods that create // transactions in various ways, while "Send" prefix is used by methods that // directly transmit created transactions to the RPC server. // // Actor also provides a [waiter.Waiter] interface to wait until transaction will be // accepted to the chain. Depending on the underlying RPCActor functionality, // transaction awaiting can be performed via web-socket using RPC notifications // subsystem with [waiter.EventBased], via regular RPC requests using a poll-based // algorithm with [waiter.PollingBased] or can not be performed if RPCActor doesn't // implement none of [waiter.RPCEventBased] and [waiter.RPCPollingBased] interfaces with // [waiter.Null]. [waiter.ErrAwaitingNotSupported] will be returned on attempt to await the // transaction in the latter case. [waiter.Waiter] uses context of the underlying RPCActor // and interrupts transaction awaiting process if the context is done. // [waiter.ErrContextDone] wrapped with the context's error will be returned in this case. // Otherwise, transaction awaiting process is ended with ValidUntilBlock acceptance // and [waiter.ErrTxNotAccepted] is returned if transaction wasn't accepted by this moment. type Actor struct { invoker.Invoker waiter.Waiter client RPCActor opts Options signers []SignerAccount txSigners []transaction.Signer version *result.Version } // Options are used to create Actor with non-standard transaction checkers or // additional attributes to be applied for all transactions. type Options struct { // Attributes are set as is into every transaction created by Actor, // unless they're explicitly set in a method call that accepts // attributes (like MakeTuned* or MakeUnsigned*). Attributes []transaction.Attribute // CheckerModifier is used by any method that creates and signs a // transaction inside (some of them provide ways to override this // default, some don't). CheckerModifier TransactionCheckerModifier // Modifier is used only by MakeUncheckedRun to modify transaction // before it's signed (other methods that perform test invocations // use CheckerModifier). MakeUnsigned* methods do not run it. Modifier TransactionModifier // waiter.PollConfig is used by [waiter.Waiter] constructor to customize // [waiter.PollingBased] behaviour. This option may be kept empty for default // polling behaviour. waiter.PollConfig } // New creates an Actor instance using the specified RPC interface and the set of // signers with corresponding accounts. Every transaction created by this Actor // will have this set of signers and all communication will be performed via this // RPC. Upon Actor instance creation a GetVersion call is made and the result of // it is cached forever (and used for internal purposes). The actor will use // default Options (which can be overridden using NewTuned). func New(ra RPCActor, signers []SignerAccount) (*Actor, error) { if len(signers) < 1 { return nil, errors.New("at least one signer (sender) is required") } invSigners := make([]transaction.Signer, len(signers)) for i := range signers { if signers[i].Account.Contract == nil { return nil, fmt.Errorf("empty contract for account %s", signers[i].Account.Address) } if !signers[i].Account.Contract.Deployed && signers[i].Account.Contract.ScriptHash() != signers[i].Signer.Account { return nil, fmt.Errorf("signer account doesn't match script hash for signer %s", signers[i].Account.Address) } invSigners[i] = signers[i].Signer } inv := invoker.New(ra, invSigners) version, err := ra.GetVersion() if err != nil { return nil, err } return &Actor{ Invoker: *inv, Waiter: waiter.New(ra, version), client: ra, opts: NewDefaultOptions(), signers: signers, txSigners: invSigners, version: version, }, nil } // NewSimple makes it easier to create an Actor for the most widespread case // when transactions have only one signer that uses CalledByEntry scope. When // other scopes or multiple signers are needed use New. func NewSimple(ra RPCActor, acc *wallet.Account) (*Actor, error) { return New(ra, []SignerAccount{{ Signer: transaction.Signer{ Account: acc.Contract.ScriptHash(), Scopes: transaction.CalledByEntry, }, Account: acc, }}) } // NewDefaultOptions returns Options that have no attributes and use the default // TransactionCheckerModifier function (that checks for the invocation result to // be in HALT state) and TransactionModifier (that does nothing). func NewDefaultOptions() Options { return Options{ CheckerModifier: DefaultCheckerModifier, Modifier: DefaultModifier, } } // NewTuned creates an Actor that will use the specified Options as defaults when // creating new transactions. If checker/modifier callbacks are not provided // (nil), then default ones (from NewDefaultOptions) are used. func NewTuned(ra RPCActor, signers []SignerAccount, opts Options) (*Actor, error) { a, err := New(ra, signers) if err != nil { return nil, err } a.opts.Attributes = opts.Attributes if opts.CheckerModifier != nil { a.opts.CheckerModifier = opts.CheckerModifier } if opts.Modifier != nil { a.opts.Modifier = opts.Modifier } a.Waiter = waiter.NewCustom(ra, a.version, opts.PollConfig) return a, err } // CalculateNetworkFee wraps RPCActor's CalculateNetworkFee, making it available // to Actor users directly. It returns network fee value for the given // transaction. func (a *Actor) CalculateNetworkFee(tx *transaction.Transaction) (int64, error) { return a.client.CalculateNetworkFee(tx) } // GetBlockCount wraps RPCActor's GetBlockCount, making it available to // Actor users directly. It returns current number of blocks in the chain. func (a *Actor) GetBlockCount() (uint32, error) { return a.client.GetBlockCount() } // GetNetwork is a convenience method that returns the network's magic number. func (a *Actor) GetNetwork() netmode.Magic { return a.version.Protocol.Network } // GetVersion returns version data from the RPC endpoint. func (a *Actor) GetVersion() result.Version { return *a.version } // Send allows to send arbitrary prepared transaction to the network. It returns // transaction hash and ValidUntilBlock value. func (a *Actor) Send(tx *transaction.Transaction) (util.Uint256, uint32, error) { h, err := a.client.SendRawTransaction(tx) return h, tx.ValidUntilBlock, err } // Sign adds signatures to arbitrary transaction using Actor signers wallets. // Most of the time it shouldn't be used directly since it'll be successful only // if the transaction is made using the same set of accounts as the one used // for Actor creation. func (a *Actor) Sign(tx *transaction.Transaction) error { if len(tx.Signers) != len(a.signers) { return errors.New("incorrect number of signers in the transaction") } for i, signer := range a.signers { err := signer.Account.SignTx(a.GetNetwork(), tx) if err != nil { // then account is non-contract-based and locked, but let's provide more detailed error if paramNum := len(signer.Account.Contract.Parameters); paramNum != 0 && signer.Account.Contract.Deployed { return fmt.Errorf("failed to add contract-based witness for signer #%d (%s): "+ "%d parameters must be provided to construct invocation script", i, signer.Account.Address, paramNum) } return fmt.Errorf("failed to add witness for signer #%d (%s): account should be unlocked to add the signature. "+ "Store partially-signed transaction and then use 'wallet sign' command to cosign it", i, signer.Account.Address) } } return nil } // SignAndSend signs arbitrary transaction (see also Sign) and sends it to the // network. func (a *Actor) SignAndSend(tx *transaction.Transaction) (util.Uint256, uint32, error) { return a.sendWrapper(tx, a.Sign(tx)) } // sendWrapper simplifies wrapping methods that create transactions. func (a *Actor) sendWrapper(tx *transaction.Transaction, err error) (util.Uint256, uint32, error) { if err != nil { return util.Uint256{}, 0, err } return a.Send(tx) } // SendCall creates a transaction that calls the given method of the given // contract with the given parameters (see also MakeCall) and sends it to the // network. func (a *Actor) SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error) { return a.sendWrapper(a.MakeCall(contract, method, params...)) } // SendTunedCall creates a transaction that calls the given method of the given // contract with the given parameters (see also MakeTunedCall) and attributes, // allowing to check for execution results of this call and modify transaction // before it's signed; this transaction is then sent to the network. func (a *Actor) SendTunedCall(contract util.Uint160, method string, attrs []transaction.Attribute, txHook TransactionCheckerModifier, params ...any) (util.Uint256, uint32, error) { return a.sendWrapper(a.MakeTunedCall(contract, method, attrs, txHook, params...)) } // SendRun creates a transaction with the given executable script (see also // MakeRun) and sends it to the network. func (a *Actor) SendRun(script []byte) (util.Uint256, uint32, error) { return a.sendWrapper(a.MakeRun(script)) } // SendTunedRun creates a transaction with the given executable script and // attributes, allowing to check for execution results of this script and modify // transaction before it's signed (see also MakeTunedRun). This transaction is // then sent to the network. func (a *Actor) SendTunedRun(script []byte, attrs []transaction.Attribute, txHook TransactionCheckerModifier) (util.Uint256, uint32, error) { return a.sendWrapper(a.MakeTunedRun(script, attrs, txHook)) } // SendUncheckedRun creates a transaction with the given executable script and // attributes that can use up to sysfee GAS for its execution, allowing to modify // this transaction before it's signed (see also MakeUncheckedRun). This // transaction is then sent to the network. func (a *Actor) SendUncheckedRun(script []byte, sysfee int64, attrs []transaction.Attribute, txHook TransactionModifier) (util.Uint256, uint32, error) { return a.sendWrapper(a.MakeUncheckedRun(script, sysfee, attrs, txHook)) } // SignerAccounts returns the array of actor's signers/accounts. It's useful in // case you need it elsewhere like for notary-related processing. Returned slice // is a newly allocated one with signers deeply copied, accounts however are not // so changing received account internals is an error. func (a *Actor) SignerAccounts() []SignerAccount { var res = make([]SignerAccount, len(a.signers)) for i := range a.signers { res[i].Signer = *a.signers[i].Signer.Copy() res[i].Account = a.signers[i].Account } return res } // Sender return the sender address that will be used in transactions created // by Actor. func (a *Actor) Sender() util.Uint160 { return a.txSigners[0].Account } // WaitSuccess is similar to [waiter.Wait], but also checks for the VM state // to be HALT (successful execution). Execution result is still returned (if // HALTed normally) in case you need to examine events or stack. func (a *Actor) WaitSuccess(h util.Uint256, vub uint32, err error) (*state.AppExecResult, error) { aer, err := a.Wait(h, vub, err) if err != nil { return nil, err } if aer.VMState != vmstate.Halt { return nil, fmt.Errorf("%w: %s", ErrExecFailed, aer.FaultException) } return aer, nil }