2022-07-07 14:41:01 +00:00
|
|
|
package params
|
2018-12-21 09:32:18 +00:00
|
|
|
|
|
|
|
import (
|
2019-11-26 10:13:17 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2019-11-20 13:07:43 +00:00
|
|
|
|
2020-03-03 14:21:42 +00:00
|
|
|
"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"
|
2020-12-29 10:44:07 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/callflag"
|
2020-03-03 14:21:42 +00:00
|
|
|
"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"
|
2018-12-21 09:32:18 +00:00
|
|
|
)
|
|
|
|
|
2024-02-27 16:56:59 +00:00
|
|
|
// ExpandFuncParameterIntoScript pushes provided FuncParam parameter
|
2024-08-26 18:27:09 +00:00
|
|
|
// into the given buffer. Returns [errors.ErrUnsupported] for types it can't
|
|
|
|
// process.
|
2024-02-27 16:56:59 +00:00
|
|
|
func ExpandFuncParameterIntoScript(script *io.BinWriter, fp FuncParam) error {
|
|
|
|
switch fp.Type {
|
|
|
|
case smartcontract.ByteArrayType:
|
|
|
|
str, err := fp.Value.GetBytesBase64()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Bytes(script, str)
|
|
|
|
case smartcontract.SignatureType:
|
|
|
|
str, err := fp.Value.GetBytesBase64()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Bytes(script, str)
|
|
|
|
case smartcontract.StringType:
|
|
|
|
str, err := fp.Value.GetString()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.String(script, str)
|
|
|
|
case smartcontract.Hash160Type:
|
|
|
|
hash, err := fp.Value.GetUint160FromHex()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Bytes(script, hash.BytesBE())
|
|
|
|
case smartcontract.Hash256Type:
|
|
|
|
hash, err := fp.Value.GetUint256()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Bytes(script, hash.BytesBE())
|
|
|
|
case smartcontract.PublicKeyType:
|
|
|
|
str, err := fp.Value.GetString()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
key, err := keys.NewPublicKeyFromString(string(str))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Bytes(script, key.Bytes())
|
|
|
|
case smartcontract.IntegerType:
|
|
|
|
bi, err := fp.Value.GetBigInt()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.BigInt(script, bi)
|
|
|
|
case smartcontract.BoolType:
|
|
|
|
val, err := fp.Value.GetBoolean() // not GetBooleanStrict(), because that's the way C# code works
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("not a bool")
|
|
|
|
}
|
|
|
|
emit.Bool(script, val)
|
|
|
|
case smartcontract.ArrayType:
|
|
|
|
val, err := fp.Value.GetArray()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = ExpandArrayIntoScriptAndPack(script, val)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case smartcontract.MapType:
|
|
|
|
val, err := fp.Value.GetArray()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = ExpandMapIntoScriptAndPack(script, val)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case smartcontract.AnyType:
|
|
|
|
if fp.Value.IsNull() || len(fp.Value.RawMessage) == 0 {
|
|
|
|
emit.Opcodes(script, opcode.PUSHNULL)
|
|
|
|
}
|
|
|
|
default:
|
2024-08-26 18:27:09 +00:00
|
|
|
return fmt.Errorf("%w: parameter type %v", errors.ErrUnsupported, fp.Type)
|
2024-02-27 16:56:59 +00:00
|
|
|
}
|
|
|
|
return script.Err
|
|
|
|
}
|
|
|
|
|
2021-03-10 14:43:52 +00:00
|
|
|
// ExpandArrayIntoScript pushes all FuncParam parameters from the given array
|
2022-04-20 18:30:09 +00:00
|
|
|
// into the given buffer in the reverse order.
|
2021-03-10 14:43:52 +00:00
|
|
|
func ExpandArrayIntoScript(script *io.BinWriter, slice []Param) error {
|
2019-11-28 16:08:31 +00:00
|
|
|
for j := len(slice) - 1; j >= 0; j-- {
|
|
|
|
fp, err := slice[j].GetFuncParam()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-27 16:56:59 +00:00
|
|
|
err = ExpandFuncParameterIntoScript(script, fp)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("param %d: %w", j, err)
|
2019-11-28 16:08:31 +00:00
|
|
|
}
|
|
|
|
}
|
2022-02-09 12:38:22 +00:00
|
|
|
return script.Err
|
2019-11-28 16:08:31 +00:00
|
|
|
}
|
|
|
|
|
rpc: adjust script creation with empty parameters list
Always use NEWARRAY0 where possible, see
https://github.com/neo-project/neo/blob/26d04a642ac5a1dd1827dabf5602767e0acba25c/src/neo/VM/Helper.cs#L41.
Compatibility is tested:
```
anna@kiwi:~/Documents/GitProjects/nspcc-dev/neo-go$ curl -d '{ "jsonrpc": "2.0", "id": 1, "method": "invokefunction", "params": ["50befd26fdf6e4d957c11e078b24ebce6291456f", "a", [] ]}' seed1.neo.org:10332 | json_pp
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 370 0 248 100 122 127 62 0:00:01 0:00:01 --:--:-- 190
{
"result" : {
"notifications" : [],
"stack" : [],
"script" : "wh8MAWEMFG9FkWLO6ySLBx7BV9nk9v0m/b5QQWJ9W1I=",
"gasconsumed" : "98403",
"state" : "FAULT",
"exception" : "Called Contract Does Not Exist: 0x50befd26fdf6e4d957c11e078b24ebce6291456f"
},
"jsonrpc" : "2.0",
"id" : 1
}
```
2022-06-22 10:23:55 +00:00
|
|
|
// ExpandArrayIntoScriptAndPack expands provided array into script and packs the
|
|
|
|
// resulting items in the array.
|
|
|
|
func ExpandArrayIntoScriptAndPack(script *io.BinWriter, slice []Param) error {
|
|
|
|
if len(slice) == 0 {
|
|
|
|
emit.Opcodes(script, opcode.NEWARRAY0)
|
|
|
|
return script.Err
|
|
|
|
}
|
|
|
|
err := ExpandArrayIntoScript(script, slice)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
emit.Int(script, int64(len(slice)))
|
|
|
|
emit.Opcodes(script, opcode.PACK)
|
|
|
|
return script.Err
|
|
|
|
}
|
|
|
|
|
2024-02-27 16:56:59 +00:00
|
|
|
// ExpandMapIntoScriptAndPack expands provided array of key-value items into script
|
|
|
|
// and packs the resulting pairs in the [stackitem.Map].
|
|
|
|
func ExpandMapIntoScriptAndPack(script *io.BinWriter, slice []Param) error {
|
|
|
|
if len(slice) == 0 {
|
|
|
|
emit.Opcodes(script, opcode.NEWMAP)
|
|
|
|
return script.Err
|
|
|
|
}
|
|
|
|
for i := len(slice) - 1; i >= 0; i-- {
|
|
|
|
pair, err := slice[i].GetFuncParamPair()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = ExpandFuncParameterIntoScript(script, pair.Value)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("map value %d: %w", i, err)
|
|
|
|
}
|
|
|
|
err = ExpandFuncParameterIntoScript(script, pair.Key)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("map key %d: %w", i, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
emit.Int(script, int64(len(slice)))
|
|
|
|
emit.Opcodes(script, opcode.PACKMAP)
|
|
|
|
return script.Err
|
|
|
|
}
|
|
|
|
|
2022-04-20 18:30:09 +00:00
|
|
|
// CreateFunctionInvocationScript creates a script to invoke the given contract with
|
|
|
|
// the given parameters.
|
2021-11-20 16:25:42 +00:00
|
|
|
func CreateFunctionInvocationScript(contract util.Uint160, method string, param *Param) ([]byte, error) {
|
2020-02-03 14:46:51 +00:00
|
|
|
script := io.NewBufBinWriter()
|
2021-11-20 16:25:42 +00:00
|
|
|
if param == nil {
|
2021-10-16 18:09:29 +00:00
|
|
|
emit.Opcodes(script.BinWriter, opcode.NEWARRAY0)
|
2021-11-20 16:25:42 +00:00
|
|
|
} else if slice, err := param.GetArray(); err == nil {
|
rpc: adjust script creation with empty parameters list
Always use NEWARRAY0 where possible, see
https://github.com/neo-project/neo/blob/26d04a642ac5a1dd1827dabf5602767e0acba25c/src/neo/VM/Helper.cs#L41.
Compatibility is tested:
```
anna@kiwi:~/Documents/GitProjects/nspcc-dev/neo-go$ curl -d '{ "jsonrpc": "2.0", "id": 1, "method": "invokefunction", "params": ["50befd26fdf6e4d957c11e078b24ebce6291456f", "a", [] ]}' seed1.neo.org:10332 | json_pp
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 370 0 248 100 122 127 62 0:00:01 0:00:01 --:--:-- 190
{
"result" : {
"notifications" : [],
"stack" : [],
"script" : "wh8MAWEMFG9FkWLO6ySLBx7BV9nk9v0m/b5QQWJ9W1I=",
"gasconsumed" : "98403",
"state" : "FAULT",
"exception" : "Called Contract Does Not Exist: 0x50befd26fdf6e4d957c11e078b24ebce6291456f"
},
"jsonrpc" : "2.0",
"id" : 1
}
```
2022-06-22 10:23:55 +00:00
|
|
|
err = ExpandArrayIntoScriptAndPack(script.BinWriter, slice)
|
2021-11-20 16:25:42 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil, fmt.Errorf("failed to convert %s to script parameter", param)
|
2021-10-16 18:09:29 +00:00
|
|
|
}
|
2019-11-26 10:13:17 +00:00
|
|
|
|
2020-12-29 10:44:07 +00:00
|
|
|
emit.AppCallNoArgs(script.BinWriter, contract, method, callflag.All)
|
2019-11-26 10:13:17 +00:00
|
|
|
return script.Bytes(), nil
|
|
|
|
}
|