forked from TrueCloudLab/neoneo-go
166 lines
5.6 KiB
Go
166 lines
5.6 KiB
Go
|
package client
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
|
||
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/wallet"
|
||
|
)
|
||
|
|
||
|
// NEP11Decimals invokes `decimals` NEP11 method on a specified contract.
|
||
|
func (c *Client) NEP11Decimals(tokenHash util.Uint160) (int64, error) {
|
||
|
return c.nepDecimals(tokenHash)
|
||
|
}
|
||
|
|
||
|
// NEP11Symbol invokes `symbol` NEP11 method on a specified contract.
|
||
|
func (c *Client) NEP11Symbol(tokenHash util.Uint160) (string, error) {
|
||
|
return c.nepSymbol(tokenHash)
|
||
|
}
|
||
|
|
||
|
// NEP11TotalSupply invokes `totalSupply` NEP11 method on a specified contract.
|
||
|
func (c *Client) NEP11TotalSupply(tokenHash util.Uint160) (int64, error) {
|
||
|
return c.nepTotalSupply(tokenHash)
|
||
|
}
|
||
|
|
||
|
// NEP11BalanceOf invokes `balanceOf` NEP11 method on a specified contract.
|
||
|
func (c *Client) NEP11BalanceOf(tokenHash, owner util.Uint160) (int64, error) {
|
||
|
return c.nepBalanceOf(tokenHash, owner, nil)
|
||
|
}
|
||
|
|
||
|
// TransferNEP11 creates an invocation transaction that invokes 'transfer' method
|
||
|
// on a given token to move the whole NEP11 token with the specified token ID to
|
||
|
// given account and sends it to the network returning just a hash of it.
|
||
|
func (c *Client) TransferNEP11(acc *wallet.Account, to util.Uint160,
|
||
|
tokenHash util.Uint160, tokenID string, gas int64) (util.Uint256, error) {
|
||
|
if !c.initDone {
|
||
|
return util.Uint256{}, errNetworkNotInitialized
|
||
|
}
|
||
|
tx, err := c.createNEP11TransferTx(acc, tokenHash, gas, to, tokenID)
|
||
|
if err != nil {
|
||
|
return util.Uint256{}, err
|
||
|
}
|
||
|
|
||
|
if err := acc.SignTx(c.GetNetwork(), tx); err != nil {
|
||
|
return util.Uint256{}, fmt.Errorf("can't sign NEP11 transfer tx: %w", err)
|
||
|
}
|
||
|
|
||
|
return c.SendRawTransaction(tx)
|
||
|
}
|
||
|
|
||
|
// createNEP11TransferTx is an internal helper for TransferNEP11 and
|
||
|
// TransferNEP11D which creates an invocation transaction for the
|
||
|
// 'transfer' method of a given contract (token) to move the whole (or the
|
||
|
// specified amount of) NEP11 token with the specified token ID to given account
|
||
|
// and returns it. The returned transaction is not signed.
|
||
|
// `args` for TransferNEP11: to util.Uint160, tokenID string;
|
||
|
// `args` for TransferNEP11D: from, to util.Uint160, amount int64, tokenID string.
|
||
|
func (c *Client) createNEP11TransferTx(acc *wallet.Account, tokenHash util.Uint160,
|
||
|
gas int64, args ...interface{}) (*transaction.Transaction, error) {
|
||
|
w := io.NewBufBinWriter()
|
||
|
emit.AppCall(w.BinWriter, tokenHash, "transfer", callflag.All, args...)
|
||
|
emit.Opcodes(w.BinWriter, opcode.ASSERT)
|
||
|
if w.Err != nil {
|
||
|
return nil, fmt.Errorf("failed to create NEP11 transfer script: %w", w.Err)
|
||
|
}
|
||
|
from, err := address.StringToUint160(acc.Address)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("bad account address: %w", err)
|
||
|
}
|
||
|
return c.CreateTxFromScript(w.Bytes(), acc, -1, gas, []SignerAccount{{
|
||
|
Signer: transaction.Signer{
|
||
|
Account: from,
|
||
|
Scopes: transaction.CalledByEntry,
|
||
|
},
|
||
|
Account: acc,
|
||
|
}})
|
||
|
}
|
||
|
|
||
|
// Non-divisible NFT methods section start.
|
||
|
|
||
|
// NEP11NDOwnerOf invokes `ownerOf` non-devisible NEP11 method with the
|
||
|
// specified token ID on a specified contract.
|
||
|
func (c *Client) NEP11NDOwnerOf(tokenHash util.Uint160, tokenID string) (util.Uint160, error) {
|
||
|
result, err := c.InvokeFunction(tokenHash, "ownerOf", []smartcontract.Parameter{
|
||
|
{
|
||
|
Type: smartcontract.StringType,
|
||
|
Value: tokenID,
|
||
|
},
|
||
|
}, nil)
|
||
|
if err != nil {
|
||
|
return util.Uint160{}, err
|
||
|
}
|
||
|
err = getInvocationError(result)
|
||
|
if err != nil {
|
||
|
return util.Uint160{}, err
|
||
|
}
|
||
|
|
||
|
return topUint160FromStack(result.Stack)
|
||
|
}
|
||
|
|
||
|
// Non-divisible NFT methods section end.
|
||
|
|
||
|
// Divisible NFT methods section start.
|
||
|
|
||
|
// TransferNEP11D creates an invocation transaction that invokes 'transfer'
|
||
|
// method on a given token to move specified amount of divisible NEP11 assets
|
||
|
// (in FixedN format using contract's number of decimals) to given account and
|
||
|
// sends it to the network returning just a hash of it.
|
||
|
func (c *Client) TransferNEP11D(acc *wallet.Account, to util.Uint160,
|
||
|
tokenHash util.Uint160, amount int64, tokenID string, gas int64) (util.Uint256, error) {
|
||
|
if !c.initDone {
|
||
|
return util.Uint256{}, errNetworkNotInitialized
|
||
|
}
|
||
|
from, err := address.StringToUint160(acc.Address)
|
||
|
if err != nil {
|
||
|
return util.Uint256{}, fmt.Errorf("bad account address: %w", err)
|
||
|
}
|
||
|
tx, err := c.createNEP11TransferTx(acc, tokenHash, gas, acc.Address, from, to, amount, tokenID)
|
||
|
if err != nil {
|
||
|
return util.Uint256{}, err
|
||
|
}
|
||
|
|
||
|
if err := acc.SignTx(c.GetNetwork(), tx); err != nil {
|
||
|
return util.Uint256{}, fmt.Errorf("can't sign NEP11 divisible transfer tx: %w", err)
|
||
|
}
|
||
|
|
||
|
return c.SendRawTransaction(tx)
|
||
|
}
|
||
|
|
||
|
// NEP11DBalanceOf invokes `balanceOf` divisible NEP11 method on a
|
||
|
// specified contract.
|
||
|
func (c *Client) NEP11DBalanceOf(tokenHash, owner util.Uint160, tokenID string) (int64, error) {
|
||
|
return c.nepBalanceOf(tokenHash, owner, &tokenID)
|
||
|
}
|
||
|
|
||
|
// Divisible NFT methods section end.
|
||
|
|
||
|
// Optional NFT methods section start.
|
||
|
|
||
|
// NEP11Properties invokes `properties` optional NEP11 method on a
|
||
|
// specified contract.
|
||
|
func (c *Client) NEP11Properties(tokenHash util.Uint160, tokenID string) (*stackitem.Map, error) {
|
||
|
result, err := c.InvokeFunction(tokenHash, "properties", []smartcontract.Parameter{{
|
||
|
Type: smartcontract.StringType,
|
||
|
Value: tokenID,
|
||
|
}}, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
err = getInvocationError(result)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return topMapFromStack(result.Stack)
|
||
|
}
|
||
|
|
||
|
// Optional NFT methods section end.
|