mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-04 03:51:35 +00:00
ec6fc54bc6
Signed-off-by: Anna Shaleva <shaleva.ann@nspcc.ru>
254 lines
11 KiB
Go
254 lines
11 KiB
Go
/*
|
|
Package notary provides an RPC-based wrapper for the Notary subsystem.
|
|
|
|
It provides both regular ContractReader/Contract interfaces for the notary
|
|
contract and notary-specific Actor as well as some helper functions to simplify
|
|
creation of notary requests.
|
|
*/
|
|
package notary
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"math"
|
|
"math/big"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/native/nativehashes"
|
|
"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/unwrap"
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
)
|
|
|
|
const (
|
|
setMaxNVBDeltaMethod = "setMaxNotValidBeforeDelta"
|
|
setFeePKMethod = "setNotaryServiceFeePerKey"
|
|
)
|
|
|
|
// ContractInvoker is used by ContractReader to perform read-only calls.
|
|
type ContractInvoker interface {
|
|
Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
|
}
|
|
|
|
// ContractActor is used by Contract to create and send transactions.
|
|
type ContractActor interface {
|
|
ContractInvoker
|
|
|
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
|
}
|
|
|
|
// ContractReader represents safe (read-only) methods of Notary. It can be
|
|
// used to query various data, but `verify` method is not exposed there because
|
|
// it can't be successful in standalone invocation (missing transaction with the
|
|
// NotaryAssisted attribute and its signature).
|
|
type ContractReader struct {
|
|
invoker ContractInvoker
|
|
}
|
|
|
|
// Contract provides full Notary interface, both safe and state-changing methods.
|
|
// The only method omitted is onNEP17Payment which can only be called
|
|
// successfully from the GASToken native contract.
|
|
type Contract struct {
|
|
ContractReader
|
|
|
|
actor ContractActor
|
|
}
|
|
|
|
// OnNEP17PaymentData is the data set that is accepted by the notary contract
|
|
// onNEP17Payment handler. It's mandatory for GAS tranfers to this contract.
|
|
type OnNEP17PaymentData struct {
|
|
// Account can be nil, in this case transfer sender (from) account is used.
|
|
Account *util.Uint160
|
|
// Till specifies the deposit lock time (in blocks).
|
|
Till uint32
|
|
}
|
|
|
|
// OnNEP17PaymentData have to implement stackitem.Convertible interface to be
|
|
// compatible with emit package.
|
|
var _ = stackitem.Convertible(&OnNEP17PaymentData{})
|
|
|
|
// Hash stores the hash of the native Notary contract.
|
|
var Hash = nativehashes.Notary
|
|
|
|
// NewReader creates an instance of ContractReader to get data from the Notary
|
|
// contract.
|
|
func NewReader(invoker ContractInvoker) *ContractReader {
|
|
return &ContractReader{invoker}
|
|
}
|
|
|
|
// New creates an instance of Contract to perform state-changing actions in the
|
|
// Notary contract.
|
|
func New(actor ContractActor) *Contract {
|
|
return &Contract{*NewReader(actor), actor}
|
|
}
|
|
|
|
// BalanceOf returns the locked GAS balance for the given account.
|
|
func (c *ContractReader) BalanceOf(account util.Uint160) (*big.Int, error) {
|
|
return unwrap.BigInt(c.invoker.Call(Hash, "balanceOf", account))
|
|
}
|
|
|
|
// ExpirationOf returns the index of the block when the GAS deposit for the given
|
|
// account will expire.
|
|
func (c *ContractReader) ExpirationOf(account util.Uint160) (uint32, error) {
|
|
res, err := c.invoker.Call(Hash, "expirationOf", account)
|
|
ret, err := unwrap.LimitedInt64(res, err, 0, math.MaxUint32)
|
|
return uint32(ret), err
|
|
}
|
|
|
|
// GetMaxNotValidBeforeDelta returns the maximum NotValidBefore attribute delta
|
|
// that can be used in notary-assisted transactions.
|
|
func (c *ContractReader) GetMaxNotValidBeforeDelta() (uint32, error) {
|
|
res, err := c.invoker.Call(Hash, "getMaxNotValidBeforeDelta")
|
|
ret, err := unwrap.LimitedInt64(res, err, 0, math.MaxUint32)
|
|
return uint32(ret), err
|
|
}
|
|
|
|
// LockDepositUntil creates and sends a transaction that extends the deposit lock
|
|
// time for the given account. The return result from the "lockDepositUntil"
|
|
// method is checked to be true, so transaction fails (with FAULT state) if not
|
|
// successful. The returned values are transaction hash, its ValidUntilBlock
|
|
// value and an error if any.
|
|
func (c *Contract) LockDepositUntil(account util.Uint160, index uint32) (util.Uint256, uint32, error) {
|
|
return c.actor.SendRun(lockScript(account, index))
|
|
}
|
|
|
|
// LockDepositUntilTransaction creates a transaction that extends the deposit lock
|
|
// time for the given account. The return result from the "lockDepositUntil"
|
|
// method is checked to be true, so transaction fails (with FAULT state) if not
|
|
// successful. The returned values are transaction hash, its ValidUntilBlock
|
|
// value and an error if any. The transaction is signed, but not sent to the
|
|
// network, instead it's returned to the caller.
|
|
func (c *Contract) LockDepositUntilTransaction(account util.Uint160, index uint32) (*transaction.Transaction, error) {
|
|
return c.actor.MakeRun(lockScript(account, index))
|
|
}
|
|
|
|
// LockDepositUntilUnsigned creates a transaction that extends the deposit lock
|
|
// time for the given account. The return result from the "lockDepositUntil"
|
|
// method is checked to be true, so transaction fails (with FAULT state) if not
|
|
// successful. The returned values are transaction hash, its ValidUntilBlock
|
|
// value and an error if any. The transaction is not signed and just returned to
|
|
// the caller.
|
|
func (c *Contract) LockDepositUntilUnsigned(account util.Uint160, index uint32) (*transaction.Transaction, error) {
|
|
return c.actor.MakeUnsignedRun(lockScript(account, index), nil)
|
|
}
|
|
|
|
func lockScript(account util.Uint160, index uint32) []byte {
|
|
// We know parameters exactly (unlike with nep17.Transfer), so this can't fail.
|
|
script, _ := smartcontract.CreateCallWithAssertScript(Hash, "lockDepositUntil", account.BytesBE(), int64(index))
|
|
return script
|
|
}
|
|
|
|
// SetMaxNotValidBeforeDelta creates and sends a transaction that sets the new
|
|
// maximum NotValidBefore attribute value delta that can be used in
|
|
// notary-assisted transactions. The action is successful when transaction
|
|
// ends in HALT state. Notice that this setting can be changed only by the
|
|
// network's committee, so use an appropriate Actor. The returned values are
|
|
// transaction hash, its ValidUntilBlock value and an error if any.
|
|
func (c *Contract) SetMaxNotValidBeforeDelta(blocks uint32) (util.Uint256, uint32, error) {
|
|
return c.actor.SendCall(Hash, setMaxNVBDeltaMethod, blocks)
|
|
}
|
|
|
|
// SetMaxNotValidBeforeDeltaTransaction creates a transaction that sets the new
|
|
// maximum NotValidBefore attribute value delta that can be used in
|
|
// notary-assisted transactions. The action is successful when transaction
|
|
// ends in HALT state. Notice that this setting can be changed only by the
|
|
// network's committee, so use an appropriate Actor. The transaction is signed,
|
|
// but not sent to the network, instead it's returned to the caller.
|
|
func (c *Contract) SetMaxNotValidBeforeDeltaTransaction(blocks uint32) (*transaction.Transaction, error) {
|
|
return c.actor.MakeCall(Hash, setMaxNVBDeltaMethod, blocks)
|
|
}
|
|
|
|
// SetMaxNotValidBeforeDeltaUnsigned creates a transaction that sets the new
|
|
// maximum NotValidBefore attribute value delta that can be used in
|
|
// notary-assisted transactions. The action is successful when transaction
|
|
// ends in HALT state. Notice that this setting can be changed only by the
|
|
// network's committee, so use an appropriate Actor. The transaction is not
|
|
// signed and just returned to the caller.
|
|
func (c *Contract) SetMaxNotValidBeforeDeltaUnsigned(blocks uint32) (*transaction.Transaction, error) {
|
|
return c.actor.MakeUnsignedCall(Hash, setMaxNVBDeltaMethod, nil, blocks)
|
|
}
|
|
|
|
// Withdraw creates and sends a transaction that withdraws the deposit belonging
|
|
// to "from" account and sends it to "to" account. The return result from the
|
|
// "withdraw" method is checked to be true, so transaction fails (with FAULT
|
|
// state) if not successful. The returned values are transaction hash, its
|
|
// ValidUntilBlock value and an error if any.
|
|
func (c *Contract) Withdraw(from util.Uint160, to util.Uint160) (util.Uint256, uint32, error) {
|
|
return c.actor.SendRun(withdrawScript(from, to))
|
|
}
|
|
|
|
// WithdrawTransaction creates a transaction that withdraws the deposit belonging
|
|
// to "from" account and sends it to "to" account. The return result from the
|
|
// "withdraw" method is checked to be true, so transaction fails (with FAULT
|
|
// state) if not successful. The transaction is signed, but not sent to the
|
|
// network, instead it's returned to the caller.
|
|
func (c *Contract) WithdrawTransaction(from util.Uint160, to util.Uint160) (*transaction.Transaction, error) {
|
|
return c.actor.MakeRun(withdrawScript(from, to))
|
|
}
|
|
|
|
// WithdrawUnsigned creates a transaction that withdraws the deposit belonging
|
|
// to "from" account and sends it to "to" account. The return result from the
|
|
// "withdraw" method is checked to be true, so transaction fails (with FAULT
|
|
// state) if not successful. The transaction is not signed and just returned to
|
|
// the caller.
|
|
func (c *Contract) WithdrawUnsigned(from util.Uint160, to util.Uint160) (*transaction.Transaction, error) {
|
|
return c.actor.MakeUnsignedRun(withdrawScript(from, to), nil)
|
|
}
|
|
|
|
func withdrawScript(from util.Uint160, to util.Uint160) []byte {
|
|
// We know parameters exactly (unlike with nep17.Transfer), so this can't fail.
|
|
script, _ := smartcontract.CreateCallWithAssertScript(Hash, "withdraw", from.BytesBE(), to.BytesBE())
|
|
return script
|
|
}
|
|
|
|
// ToStackItem implements stackitem.Convertible interface.
|
|
func (d *OnNEP17PaymentData) ToStackItem() (stackitem.Item, error) {
|
|
return stackitem.NewArray([]stackitem.Item{
|
|
stackitem.Make(d.Account),
|
|
stackitem.Make(d.Till),
|
|
}), nil
|
|
}
|
|
|
|
// FromStackItem implements stackitem.Convertible interface.
|
|
func (d *OnNEP17PaymentData) FromStackItem(si stackitem.Item) error {
|
|
arr, ok := si.Value().([]stackitem.Item)
|
|
if !ok {
|
|
return fmt.Errorf("unexpected stackitem type: %s", si.Type())
|
|
}
|
|
if len(arr) != 2 {
|
|
return fmt.Errorf("unexpected number of fields: %d vs %d", len(arr), 2)
|
|
}
|
|
|
|
if arr[0] != stackitem.Item(stackitem.Null{}) {
|
|
accBytes, err := arr[0].TryBytes()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to retrieve account bytes: %w", err)
|
|
}
|
|
acc, err := util.Uint160DecodeBytesBE(accBytes)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to decode account bytes: %w", err)
|
|
}
|
|
d.Account = &acc
|
|
}
|
|
till, err := arr[1].TryInteger()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to retrieve till: %w", err)
|
|
}
|
|
if !till.IsInt64() {
|
|
return errors.New("till is not an int64")
|
|
}
|
|
val := till.Int64()
|
|
if val > math.MaxUint32 {
|
|
return fmt.Errorf("till is larger than max uint32 value: %d", val)
|
|
}
|
|
d.Till = uint32(val)
|
|
|
|
return nil
|
|
}
|