mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-10 05:54:04 +00:00
1b83dc2476
Mostly it's about Go 1.22+ syntax with ranging over integers, but it also prefers ranging over slices where possible (it makes code a little better to read). Notice that we have a number of dangerous loops where slices are mutated during loop execution, many of these can't be converted since we need proper length evalutation at every iteration. Signed-off-by: Roman Khimov <roman@nspcc.ru>
497 lines
16 KiB
Go
497 lines
16 KiB
Go
package compiler_test
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
|
"github.com/nspcc-dev/neo-go/pkg/core"
|
|
"github.com/nspcc-dev/neo-go/pkg/core/interop"
|
|
"github.com/nspcc-dev/neo-go/pkg/core/interop/interopnames"
|
|
istorage "github.com/nspcc-dev/neo-go/pkg/core/interop/storage"
|
|
"github.com/nspcc-dev/neo-go/pkg/interop/contract"
|
|
"github.com/nspcc-dev/neo-go/pkg/interop/storage"
|
|
"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"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// Checks that changes in `smartcontract` are reflected in compiler interop package.
|
|
func TestCallFlags(t *testing.T) {
|
|
require.EqualValues(t, contract.ReadStates, callflag.ReadStates)
|
|
require.EqualValues(t, contract.WriteStates, callflag.WriteStates)
|
|
require.EqualValues(t, contract.AllowCall, callflag.AllowCall)
|
|
require.EqualValues(t, contract.AllowNotify, callflag.AllowNotify)
|
|
require.EqualValues(t, contract.States, callflag.States)
|
|
require.EqualValues(t, contract.ReadOnly, callflag.ReadOnly)
|
|
require.EqualValues(t, contract.All, callflag.All)
|
|
require.EqualValues(t, contract.NoneFlag, callflag.NoneFlag)
|
|
}
|
|
|
|
func TestFindFlags(t *testing.T) {
|
|
require.EqualValues(t, storage.None, istorage.FindDefault)
|
|
require.EqualValues(t, storage.KeysOnly, istorage.FindKeysOnly)
|
|
require.EqualValues(t, storage.RemovePrefix, istorage.FindRemovePrefix)
|
|
require.EqualValues(t, storage.ValuesOnly, istorage.FindValuesOnly)
|
|
require.EqualValues(t, storage.DeserializeValues, istorage.FindDeserialize)
|
|
require.EqualValues(t, storage.PickField0, istorage.FindPick0)
|
|
require.EqualValues(t, storage.PickField1, istorage.FindPick1)
|
|
require.EqualValues(t, storage.Backwards, istorage.FindBackwards)
|
|
}
|
|
|
|
type syscallTestCase struct {
|
|
method string
|
|
params []string
|
|
isVoid bool
|
|
}
|
|
|
|
// This test ensures that our wrappers have the necessary number of parameters
|
|
// and execute the appropriate syscall. Because of lack of typing (compared to native contracts),
|
|
// parameter types can't be checked.
|
|
func TestSyscallExecution(t *testing.T) {
|
|
b := `[]byte{1}`
|
|
u160 := `interop.Hash160("aaaaaaaaaaaaaaaaaaaa")`
|
|
pub := `interop.PublicKey("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa")`
|
|
pubs := "[]interop.PublicKey{ " + pub + "}"
|
|
sig := `interop.Signature("aaaaaa")`
|
|
sigs := "[]interop.Signature{" + sig + "}"
|
|
sctx := "storage.Context{}"
|
|
interops := map[string]syscallTestCase{
|
|
"contract.Call": {interopnames.SystemContractCall, []string{u160, `"m"`, "1", "3"}, false},
|
|
"contract.CreateMultisigAccount": {interopnames.SystemContractCreateMultisigAccount, []string{"1", pubs}, false},
|
|
"contract.CreateStandardAccount": {interopnames.SystemContractCreateStandardAccount, []string{pub}, false},
|
|
"contract.GetCallFlags": {interopnames.SystemContractGetCallFlags, nil, false},
|
|
"iterator.Next": {interopnames.SystemIteratorNext, []string{"iterator.Iterator{}"}, false},
|
|
"iterator.Value": {interopnames.SystemIteratorValue, []string{"iterator.Iterator{}"}, false},
|
|
"runtime.BurnGas": {interopnames.SystemRuntimeBurnGas, []string{"1"}, true},
|
|
"runtime.CheckWitness": {interopnames.SystemRuntimeCheckWitness, []string{b}, false},
|
|
"runtime.CurrentSigners": {interopnames.SystemRuntimeCurrentSigners, nil, false},
|
|
"runtime.GasLeft": {interopnames.SystemRuntimeGasLeft, nil, false},
|
|
"runtime.GetAddressVersion": {interopnames.SystemRuntimeGetAddressVersion, nil, false},
|
|
"runtime.GetCallingScriptHash": {interopnames.SystemRuntimeGetCallingScriptHash, nil, false},
|
|
"runtime.GetEntryScriptHash": {interopnames.SystemRuntimeGetEntryScriptHash, nil, false},
|
|
"runtime.GetExecutingScriptHash": {interopnames.SystemRuntimeGetExecutingScriptHash, nil, false},
|
|
"runtime.GetInvocationCounter": {interopnames.SystemRuntimeGetInvocationCounter, nil, false},
|
|
"runtime.GetNetwork": {interopnames.SystemRuntimeGetNetwork, nil, false},
|
|
"runtime.GetNotifications": {interopnames.SystemRuntimeGetNotifications, []string{u160}, false},
|
|
"runtime.GetRandom": {interopnames.SystemRuntimeGetRandom, nil, false},
|
|
"runtime.GetScriptContainer": {interopnames.SystemRuntimeGetScriptContainer, nil, false},
|
|
"runtime.GetTime": {interopnames.SystemRuntimeGetTime, nil, false},
|
|
"runtime.GetTrigger": {interopnames.SystemRuntimeGetTrigger, nil, false},
|
|
"runtime.LoadScript": {interopnames.SystemRuntimeLoadScript, []string{b, "0", b}, false},
|
|
"runtime.Log": {interopnames.SystemRuntimeLog, []string{`"msg"`}, true},
|
|
"runtime.Notify": {interopnames.SystemRuntimeNotify, []string{`"ev"`, "1"}, true},
|
|
"runtime.Platform": {interopnames.SystemRuntimePlatform, nil, false},
|
|
"storage.Delete": {interopnames.SystemStorageDelete, []string{sctx, b}, true},
|
|
"storage.Find": {interopnames.SystemStorageFind, []string{sctx, b, "storage.None"}, false},
|
|
"storage.Get": {interopnames.SystemStorageGet, []string{sctx, b}, false},
|
|
"storage.GetContext": {interopnames.SystemStorageGetContext, nil, false},
|
|
"storage.GetReadOnlyContext": {interopnames.SystemStorageGetReadOnlyContext, nil, false},
|
|
"storage.Put": {interopnames.SystemStoragePut, []string{sctx, b, b}, true},
|
|
"storage.ConvertContextToReadOnly": {interopnames.SystemStorageAsReadOnly, []string{sctx}, false},
|
|
"crypto.CheckMultisig": {interopnames.SystemCryptoCheckMultisig, []string{pubs, sigs}, false},
|
|
"crypto.CheckSig": {interopnames.SystemCryptoCheckSig, []string{pub, sig}, false},
|
|
}
|
|
ic := &interop.Context{}
|
|
core.SpawnVM(ic) // set Functions field
|
|
for _, fs := range ic.Functions {
|
|
// It will be set in test and we want to fail if calling invalid syscall.
|
|
fs.Func = nil
|
|
}
|
|
|
|
srcBuilder := bytes.NewBuffer(nil)
|
|
srcBuilder.WriteString(`package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/contract"
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/storage"
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/iterator"
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/crypto"
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
func unused() { var _ interop.Hash160 }
|
|
`)
|
|
for goName, tc := range interops {
|
|
realName := strings.ToTitle(strings.Replace(goName, ".", "", 1))
|
|
var tmpl string
|
|
if tc.isVoid {
|
|
tmpl = "func %s() { %s(%s) }\n"
|
|
} else {
|
|
tmpl = "func %s() any { return %s(%s) }\n"
|
|
}
|
|
srcBuilder.WriteString(fmt.Sprintf(tmpl, realName, goName, strings.Join(tc.params, ", ")))
|
|
}
|
|
|
|
nf, di, err := compiler.CompileWithOptions("foo.go", srcBuilder, nil)
|
|
require.NoError(t, err)
|
|
|
|
for goName, tc := range interops {
|
|
t.Run(goName, func(t *testing.T) {
|
|
name := strings.ToTitle(strings.Replace(goName, ".", "", 1))
|
|
runSyscallTestCase(t, ic, name, nf.Script, di, tc)
|
|
})
|
|
}
|
|
}
|
|
|
|
func runSyscallTestCase(t *testing.T, ic *interop.Context, realName string,
|
|
script []byte, debugInfo *compiler.DebugInfo, tc syscallTestCase) {
|
|
syscallID := interopnames.ToID([]byte(tc.method))
|
|
f := ic.GetFunction(syscallID)
|
|
require.NotNil(t, f)
|
|
require.Equal(t, f.ParamCount, len(tc.params))
|
|
called := false
|
|
f.Func = func(ic *interop.Context) error {
|
|
called = true
|
|
if ic.VM.Estack().Len() < f.ParamCount {
|
|
return errors.New("not enough parameters")
|
|
}
|
|
for range f.ParamCount {
|
|
ic.VM.Estack().Pop()
|
|
}
|
|
if !tc.isVoid {
|
|
ic.VM.Estack().PushVal(42)
|
|
}
|
|
return nil
|
|
}
|
|
defer func() { f.Func = nil }()
|
|
|
|
invokeMethod(t, realName, script, ic.VM, debugInfo)
|
|
require.NoError(t, ic.VM.Run())
|
|
require.True(t, called)
|
|
if tc.isVoid {
|
|
require.Equal(t, 0, ic.VM.Estack().Len())
|
|
} else {
|
|
require.Equal(t, 1, ic.VM.Estack().Len())
|
|
require.Equal(t, big.NewInt(42), ic.VM.Estack().Pop().Value())
|
|
}
|
|
}
|
|
|
|
func TestStoragePutGet(t *testing.T) {
|
|
src := `
|
|
package foo
|
|
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/storage"
|
|
|
|
func Main() string {
|
|
ctx := storage.GetContext()
|
|
key := []byte("token")
|
|
storage.Put(ctx, key, []byte("foo"))
|
|
x := storage.Get(ctx, key)
|
|
return x.(string)
|
|
}
|
|
`
|
|
eval(t, src, []byte("foo"))
|
|
}
|
|
|
|
func TestNotify(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
|
func Main(arg int) {
|
|
runtime.Notify("Event1", arg, "sum", arg+1)
|
|
runtime.Notify("single")
|
|
}`
|
|
|
|
v, s, _ := vmAndCompileInterop(t, src)
|
|
v.Estack().PushVal(11)
|
|
|
|
require.NoError(t, v.Run())
|
|
require.Equal(t, 2, len(s.events))
|
|
|
|
exp0 := []stackitem.Item{stackitem.NewBigInteger(big.NewInt(11)), stackitem.NewByteArray([]byte("sum")), stackitem.NewBigInteger(big.NewInt(12))}
|
|
assert.Equal(t, "Event1", s.events[0].Name)
|
|
assert.Equal(t, exp0, s.events[0].Item.Value())
|
|
assert.Equal(t, "single", s.events[1].Name)
|
|
assert.Equal(t, []stackitem.Item{}, s.events[1].Item.Value())
|
|
|
|
t.Run("long event name", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
|
func Main(arg int) {
|
|
runtime.Notify("long event12345678901234567890123")
|
|
}`
|
|
|
|
_, _, err := compiler.CompileWithOptions("foo.go", strings.NewReader(src), nil)
|
|
require.Error(t, err)
|
|
})
|
|
}
|
|
|
|
func TestSyscallInGlobalInit(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/runtime"
|
|
var a = runtime.CheckWitness([]byte("5T"))
|
|
func Main() bool {
|
|
return a
|
|
}`
|
|
v, s, _ := vmAndCompileInterop(t, src)
|
|
s.interops[interopnames.ToID([]byte(interopnames.SystemRuntimeCheckWitness))] = func(v *vm.VM) error {
|
|
s := v.Estack().Pop().Value().([]byte)
|
|
require.Equal(t, "5T", string(s))
|
|
v.Estack().PushVal(true)
|
|
return nil
|
|
}
|
|
require.NoError(t, v.Run())
|
|
require.Equal(t, true, v.Estack().Pop().Value())
|
|
}
|
|
|
|
func TestOpcode(t *testing.T) {
|
|
t.Run("1 argument", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/neogointernal"
|
|
func abs(a int) int {
|
|
return neogointernal.Opcode1("ABS", a).(int)
|
|
}
|
|
func Main() int {
|
|
return abs(-42)
|
|
}`
|
|
eval(t, src, big.NewInt(42))
|
|
})
|
|
t.Run("2 arguments", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/neogointernal"
|
|
func add3(a, b, c int) int {
|
|
return neogointernal.Opcode2("SUB", a,
|
|
neogointernal.Opcode2("SUB", b, c).(int)).(int)
|
|
}
|
|
func Main() int {
|
|
return add3(53, 12, 1)
|
|
}`
|
|
eval(t, src, big.NewInt(42))
|
|
})
|
|
t.Run("POW", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() int {
|
|
return math.Pow(2, math.Pow(3, 2))
|
|
}`
|
|
eval(t, src, big.NewInt(512))
|
|
})
|
|
t.Run("SRQT", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() int {
|
|
return math.Sqrt(math.Sqrt(101)) // == sqrt(10) == 3
|
|
}`
|
|
eval(t, src, big.NewInt(3))
|
|
})
|
|
t.Run("SIGN", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() []int {
|
|
signs := make([]int, 3)
|
|
signs[0] = math.Sign(-123)
|
|
signs[1] = math.Sign(0)
|
|
signs[2] = math.Sign(42)
|
|
return signs
|
|
}`
|
|
eval(t, src, []stackitem.Item{
|
|
stackitem.Make(-1),
|
|
stackitem.Make(0),
|
|
stackitem.Make(1),
|
|
})
|
|
})
|
|
t.Run("ABS", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() int {
|
|
return math.Abs(-3)
|
|
}`
|
|
eval(t, src, big.NewInt(3))
|
|
})
|
|
t.Run("MAX", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() int {
|
|
return math.Max(1, 2) + math.Max(8, 3)
|
|
}`
|
|
eval(t, src, big.NewInt(10))
|
|
})
|
|
t.Run("MIN", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() int {
|
|
return math.Min(1, 2) + math.Min(8, 3)
|
|
}`
|
|
eval(t, src, big.NewInt(4))
|
|
})
|
|
t.Run("WITHIN", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() []bool {
|
|
r := make([]bool, 5)
|
|
r[0] = math.Within(2, 3, 5)
|
|
r[1] = math.Within(3, 3, 5)
|
|
r[2] = math.Within(4, 3, 5)
|
|
r[3] = math.Within(5, 3, 5)
|
|
r[4] = math.Within(6, 3, 5)
|
|
return r
|
|
}`
|
|
eval(t, src, []stackitem.Item{
|
|
stackitem.Make(false),
|
|
stackitem.Make(true),
|
|
stackitem.Make(true),
|
|
stackitem.Make(false),
|
|
stackitem.Make(false),
|
|
})
|
|
})
|
|
t.Run("MODMUL", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() []int {
|
|
r := make([]int, 5)
|
|
r[0] = math.ModMul(3, 4, 5)
|
|
r[1] = math.ModMul(-3, 4, 5)
|
|
r[2] = math.ModMul(3, 4, -5)
|
|
r[3] = math.ModMul(-3, 4, -5)
|
|
r[4] = math.ModMul(0, 4, 5)
|
|
return r
|
|
}`
|
|
eval(t, src, []stackitem.Item{
|
|
stackitem.Make(2),
|
|
stackitem.Make(3),
|
|
stackitem.Make(2),
|
|
stackitem.Make(3),
|
|
stackitem.Make(0),
|
|
})
|
|
})
|
|
t.Run("MODPOW", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop/math"
|
|
func Main() []int {
|
|
r := make([]int, 5)
|
|
r[0] = math.ModPow(3, 4, 5)
|
|
r[1] = math.ModPow(-3, 5, 5)
|
|
r[2] = math.ModPow(3, 4, -5)
|
|
r[3] = math.ModPow(-3, 5, -5)
|
|
r[4] = math.ModPow(19, -1, 141)
|
|
return r
|
|
}`
|
|
eval(t, src, []stackitem.Item{
|
|
stackitem.Make(1),
|
|
stackitem.Make(2),
|
|
stackitem.Make(1),
|
|
stackitem.Make(2),
|
|
stackitem.Make(52),
|
|
})
|
|
})
|
|
}
|
|
|
|
func TestInteropTypesComparison(t *testing.T) {
|
|
typeCheck := func(t *testing.T, typeName string, typeLen int) {
|
|
t.Run(typeName, func(t *testing.T) {
|
|
var ha, hb string
|
|
for i := range typeLen {
|
|
if i == typeLen-1 {
|
|
ha += "2"
|
|
hb += "3"
|
|
} else {
|
|
ha += "1, "
|
|
hb += "1, "
|
|
}
|
|
}
|
|
check := func(t *testing.T, a, b string, expected bool) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + a + `}
|
|
b := interop.` + typeName + `{` + b + `}
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, expected)
|
|
}
|
|
t.Run("same type", func(t *testing.T) {
|
|
check(t, ha, ha, true)
|
|
check(t, ha, hb, false)
|
|
})
|
|
t.Run("a is nil", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
var a interop.` + typeName + `
|
|
b := interop.` + typeName + `{` + hb + `}
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, false)
|
|
})
|
|
t.Run("b is nil", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + ha + `}
|
|
var b interop.` + typeName + `
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, false)
|
|
})
|
|
t.Run("both nil", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
var a interop.` + typeName + `
|
|
var b interop.` + typeName + `
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, true)
|
|
})
|
|
t.Run("different types", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + ha + `}
|
|
b := 123
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, false)
|
|
})
|
|
t.Run("b is Buffer", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + ha + `}
|
|
b := []byte{` + ha + `}
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, true)
|
|
})
|
|
t.Run("b is ByteString", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + ha + `}
|
|
b := string([]byte{` + ha + `})
|
|
return a.Equals(b)
|
|
}`
|
|
eval(t, src, true)
|
|
})
|
|
t.Run("b is compound type", func(t *testing.T) {
|
|
src := `package foo
|
|
import "github.com/nspcc-dev/neo-go/pkg/interop"
|
|
|
|
func Main() bool {
|
|
a := interop.` + typeName + `{` + ha + `}
|
|
b := struct{}{}
|
|
return a.Equals(b)
|
|
}`
|
|
vm, _, _ := vmAndCompileInterop(t, src)
|
|
err := vm.Run()
|
|
require.Error(t, err)
|
|
require.True(t, strings.Contains(err.Error(), "invalid conversion: Struct/ByteString"), err)
|
|
})
|
|
})
|
|
}
|
|
typeCheck(t, "Hash160", util.Uint160Size)
|
|
typeCheck(t, "Hash256", util.Uint256Size)
|
|
typeCheck(t, "Signature", smartcontract.SignatureLen)
|
|
typeCheck(t, "PublicKey", smartcontract.PublicKeyLen)
|
|
}
|