2020-07-24 13:54:03 +00:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2021-11-09 14:52:48 +00:00
|
|
|
"fmt"
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2020-12-11 08:33:27 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/encoding/fixedn"
|
2020-07-24 13:54:03 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
2020-08-27 11:57:13 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
2020-07-24 13:54:03 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// StaticClient is a wrapper over Neo:Morph client
|
|
|
|
// that invokes single smart contract methods with fixed fee.
|
|
|
|
//
|
|
|
|
// Working static client must be created via constructor NewStatic.
|
|
|
|
// Using the StaticClient that has been created with new(StaticClient)
|
|
|
|
// expression (or just declaring a StaticClient variable) is unsafe
|
|
|
|
// and can lead to panic.
|
|
|
|
type StaticClient struct {
|
2021-05-25 12:07:57 +00:00
|
|
|
tryNotary bool
|
2021-08-25 11:05:07 +00:00
|
|
|
alpha bool // use client's key to sign notary request's main TX
|
2021-05-25 12:07:57 +00:00
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
client *Client // neo-go client instance
|
|
|
|
|
|
|
|
scScriptHash util.Uint160 // contract script-hash
|
|
|
|
|
2020-12-11 08:33:27 +00:00
|
|
|
fee fixedn.Fixed8 // invocation fee
|
2020-07-24 13:54:03 +00:00
|
|
|
}
|
|
|
|
|
2021-05-25 12:07:57 +00:00
|
|
|
type staticOpts struct {
|
|
|
|
tryNotary bool
|
2021-08-25 11:05:07 +00:00
|
|
|
alpha bool
|
2021-05-25 12:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// StaticClientOption allows to set an optional
|
|
|
|
// parameter of StaticClient.
|
|
|
|
type StaticClientOption func(*staticOpts)
|
|
|
|
|
|
|
|
func defaultStaticOpts() *staticOpts {
|
|
|
|
return new(staticOpts)
|
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
// ErrNilStaticClient is returned by functions that expect
|
|
|
|
// a non-nil StaticClient pointer, but received nil.
|
|
|
|
var ErrNilStaticClient = errors.New("static client is nil")
|
|
|
|
|
|
|
|
// NewStatic creates, initializes and returns the StaticClient instance.
|
|
|
|
//
|
|
|
|
// If provided Client instance is nil, ErrNilClient is returned.
|
2021-05-25 12:07:57 +00:00
|
|
|
func NewStatic(client *Client, scriptHash util.Uint160, fee fixedn.Fixed8, opts ...StaticClientOption) (*StaticClient, error) {
|
2020-07-24 13:54:03 +00:00
|
|
|
if client == nil {
|
|
|
|
return nil, ErrNilClient
|
|
|
|
}
|
|
|
|
|
2021-05-25 12:07:57 +00:00
|
|
|
o := defaultStaticOpts()
|
|
|
|
|
|
|
|
for i := range opts {
|
|
|
|
opts[i](o)
|
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
return &StaticClient{
|
2021-05-25 12:07:57 +00:00
|
|
|
tryNotary: o.tryNotary,
|
2021-08-25 11:05:07 +00:00
|
|
|
alpha: o.alpha,
|
2020-07-24 13:54:03 +00:00
|
|
|
client: client,
|
|
|
|
scScriptHash: scriptHash,
|
|
|
|
fee: fee,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-09-03 17:19:16 +00:00
|
|
|
// Morph return wrapped raw morph client.
|
|
|
|
func (s StaticClient) Morph() *Client {
|
|
|
|
return s.client
|
|
|
|
}
|
|
|
|
|
2021-11-09 14:52:48 +00:00
|
|
|
// InvokePrm groups parameters of the Invoke operation.
|
|
|
|
type InvokePrm struct {
|
|
|
|
// required parameters
|
|
|
|
method string
|
|
|
|
args []interface{}
|
|
|
|
|
|
|
|
// optional parameters
|
|
|
|
InvokePrmOptional
|
|
|
|
}
|
|
|
|
|
|
|
|
// InvokePrmOptional groups optional parameters of the Invoke operation.
|
|
|
|
type InvokePrmOptional struct {
|
|
|
|
// hash is an optional hash of the transaction
|
|
|
|
// that generated the notification that required
|
|
|
|
// to invoke notary request.
|
|
|
|
// It is used to generate same but unique nonce and
|
|
|
|
// `validUntilBlock` values by all notification
|
|
|
|
// receivers.
|
|
|
|
hash *util.Uint256
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetMethod sets method of the contract to call.
|
|
|
|
func (i *InvokePrm) SetMethod(method string) {
|
|
|
|
i.method = method
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetArgs sets arguments of the contact call.
|
|
|
|
func (i *InvokePrm) SetArgs(args ...interface{}) {
|
|
|
|
i.args = args
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetHash sets optional hash of the transaction.
|
|
|
|
// If hash is set and notary is enabled, StaticClient
|
|
|
|
// uses it for notary nonce and `validUntilBlock`
|
|
|
|
// calculation.
|
|
|
|
func (i *InvokePrmOptional) SetHash(hash util.Uint256) {
|
|
|
|
i.hash = &hash
|
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
// Invoke calls Invoke method of Client with static internal script hash and fee.
|
|
|
|
// Supported args types are the same as in Client.
|
2021-05-25 12:07:57 +00:00
|
|
|
//
|
2021-11-09 14:52:48 +00:00
|
|
|
// If TryNotary is provided:
|
|
|
|
// - if AsAlphabet is provided, calls NotaryInvoke;
|
|
|
|
// - otherwise, calls NotaryInvokeNotAlpha.
|
|
|
|
func (s StaticClient) Invoke(prm InvokePrm) error {
|
2021-05-25 12:07:57 +00:00
|
|
|
if s.tryNotary {
|
2021-08-25 11:05:07 +00:00
|
|
|
if s.alpha {
|
2021-11-09 14:52:48 +00:00
|
|
|
var (
|
|
|
|
nonce uint32 = 1
|
|
|
|
vubP *uint32
|
|
|
|
vub uint32
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
if prm.hash != nil {
|
|
|
|
nonce, vub, err = s.client.CalculateNonceAndVUB(*prm.hash)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("could not calculate nonce and VUB for notary alphabet invoke: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
vubP = &vub
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.client.NotaryInvoke(s.scScriptHash, s.fee, nonce, vubP, prm.method, prm.args...)
|
2021-08-25 11:05:07 +00:00
|
|
|
}
|
|
|
|
|
2021-11-09 14:52:48 +00:00
|
|
|
return s.client.NotaryInvokeNotAlpha(s.scScriptHash, s.fee, prm.method, prm.args...)
|
2021-05-25 12:07:57 +00:00
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
return s.client.Invoke(
|
|
|
|
s.scScriptHash,
|
|
|
|
s.fee,
|
2021-11-09 14:52:48 +00:00
|
|
|
prm.method,
|
|
|
|
prm.args...,
|
2020-07-24 13:54:03 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-11-09 14:52:48 +00:00
|
|
|
// TestInvokePrm groups parameters of the TestInvoke operation.
|
|
|
|
type TestInvokePrm struct {
|
|
|
|
method string
|
|
|
|
args []interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetMethod sets method of the contract to call.
|
|
|
|
func (ti *TestInvokePrm) SetMethod(method string) {
|
|
|
|
ti.method = method
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetArgs sets arguments of the contact call.
|
|
|
|
func (ti *TestInvokePrm) SetArgs(args ...interface{}) {
|
|
|
|
ti.args = args
|
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
// TestInvoke calls TestInvoke method of Client with static internal script hash.
|
2021-11-09 14:52:48 +00:00
|
|
|
func (s StaticClient) TestInvoke(prm TestInvokePrm) ([]stackitem.Item, error) {
|
2020-07-24 13:54:03 +00:00
|
|
|
return s.client.TestInvoke(
|
|
|
|
s.scScriptHash,
|
2021-11-09 14:52:48 +00:00
|
|
|
prm.method,
|
|
|
|
prm.args...,
|
2020-07-24 13:54:03 +00:00
|
|
|
)
|
|
|
|
}
|
2021-08-25 11:05:07 +00:00
|
|
|
|
2021-09-15 10:45:28 +00:00
|
|
|
// ContractAddress returns the address of the associated contract.
|
|
|
|
func (s StaticClient) ContractAddress() util.Uint160 {
|
|
|
|
return s.scScriptHash
|
|
|
|
}
|
|
|
|
|
2021-08-25 11:05:07 +00:00
|
|
|
// TryNotary returns option to enable
|
|
|
|
// notary invocation tries.
|
|
|
|
func TryNotary() StaticClientOption {
|
|
|
|
return func(o *staticOpts) {
|
|
|
|
o.tryNotary = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AsAlphabet returns option to sign main TX
|
|
|
|
// of notary requests with client's private
|
|
|
|
// key.
|
|
|
|
//
|
|
|
|
// Considered to be used by IR nodes only.
|
|
|
|
func AsAlphabet() StaticClientOption {
|
|
|
|
return func(o *staticOpts) {
|
|
|
|
o.alpha = true
|
|
|
|
}
|
|
|
|
}
|