mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-04 03:51:35 +00:00
176 lines
5.5 KiB
Go
176 lines
5.5 KiB
Go
|
package runtime
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"math/big"
|
||
|
"math/rand"
|
||
|
"testing"
|
||
|
|
||
|
"github.com/nspcc-dev/neo-go/internal/random"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/core/block"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/core/interop"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/crypto/hash"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
||
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/trigger"
|
||
|
"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/require"
|
||
|
"go.uber.org/zap"
|
||
|
"go.uber.org/zap/zapcore"
|
||
|
"go.uber.org/zap/zaptest"
|
||
|
)
|
||
|
|
||
|
func checkStack(t *testing.T, v *vm.VM, args ...interface{}) {
|
||
|
require.Equal(t, len(args), v.Estack().Len())
|
||
|
for i := range args {
|
||
|
require.Equal(t, stackitem.Make(args[i]), v.Estack().Pop().Item(), "%d", i)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestGetTrigger(t *testing.T) {
|
||
|
triggers := []trigger.Type{trigger.Application, trigger.Verification}
|
||
|
for _, tr := range triggers {
|
||
|
ic := &interop.Context{Trigger: tr, VM: vm.New()}
|
||
|
require.NoError(t, GetTrigger(ic))
|
||
|
checkStack(t, ic.VM, int64(tr))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestPlatform(t *testing.T) {
|
||
|
ic := &interop.Context{VM: vm.New()}
|
||
|
require.NoError(t, Platform(ic))
|
||
|
checkStack(t, ic.VM, "NEO")
|
||
|
}
|
||
|
|
||
|
func TestGetTime(t *testing.T) {
|
||
|
b := block.New(netmode.UnitTestNet, false)
|
||
|
b.Timestamp = rand.Uint64()
|
||
|
ic := &interop.Context{VM: vm.New(), Block: b}
|
||
|
require.NoError(t, GetTime(ic))
|
||
|
checkStack(t, ic.VM, new(big.Int).SetUint64(b.Timestamp))
|
||
|
}
|
||
|
|
||
|
func TestGetScriptHash(t *testing.T) {
|
||
|
scripts := []struct {
|
||
|
s []byte
|
||
|
h util.Uint160
|
||
|
}{
|
||
|
{[]byte{1, 2, 3, 4}, hash.Hash160([]byte{1, 2, 3, 4})},
|
||
|
{[]byte{1, 2, 3}, util.Uint160{4, 8, 15, 16}},
|
||
|
{[]byte{1, 2}, hash.Hash160([]byte{1, 2})},
|
||
|
{[]byte{1}, hash.Hash160([]byte{1})},
|
||
|
}
|
||
|
|
||
|
ic := &interop.Context{VM: vm.New()}
|
||
|
ic.VM.LoadScriptWithFlags(scripts[0].s, smartcontract.All)
|
||
|
require.NoError(t, GetEntryScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
require.NoError(t, GetCallingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, util.Uint160{}.BytesBE())
|
||
|
require.NoError(t, GetExecutingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
|
||
|
ic.VM.LoadScriptWithHash(scripts[1].s, scripts[1].h, smartcontract.All)
|
||
|
require.NoError(t, GetEntryScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
require.NoError(t, GetCallingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
require.NoError(t, GetExecutingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[1].h.BytesBE())
|
||
|
|
||
|
ic.VM.LoadScript(scripts[2].s)
|
||
|
require.NoError(t, GetEntryScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
require.NoError(t, GetCallingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[1].h.BytesBE())
|
||
|
require.NoError(t, GetExecutingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[2].h.BytesBE())
|
||
|
|
||
|
ic.VM.LoadScript(scripts[3].s)
|
||
|
require.NoError(t, GetEntryScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[0].h.BytesBE())
|
||
|
require.NoError(t, GetCallingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[2].h.BytesBE())
|
||
|
require.NoError(t, GetExecutingScriptHash(ic))
|
||
|
checkStack(t, ic.VM, scripts[3].h.BytesBE())
|
||
|
}
|
||
|
|
||
|
func TestLog(t *testing.T) {
|
||
|
newL := func(l zapcore.Level) (*zap.Logger, *zaptest.Buffer) {
|
||
|
enc := zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig())
|
||
|
w := &zaptest.Buffer{}
|
||
|
zc := zapcore.NewCore(enc, w, l)
|
||
|
return zap.New(zc, zap.ErrorOutput(w)), w
|
||
|
}
|
||
|
h := random.Uint160()
|
||
|
|
||
|
t.Run("big message", func(t *testing.T) {
|
||
|
ic := &interop.Context{Log: zap.NewNop(), VM: vm.New()}
|
||
|
ic.VM.LoadScriptWithHash([]byte{1}, h, smartcontract.All)
|
||
|
ic.VM.Estack().PushVal(string(make([]byte, MaxNotificationSize+1)))
|
||
|
require.Error(t, Log(ic))
|
||
|
})
|
||
|
|
||
|
t.Run("good", func(t *testing.T) {
|
||
|
log, buf := newL(zapcore.InfoLevel)
|
||
|
ic := &interop.Context{Log: log, VM: vm.New()}
|
||
|
ic.VM.LoadScriptWithHash([]byte{1}, h, smartcontract.All)
|
||
|
ic.VM.Estack().PushVal("hello")
|
||
|
require.NoError(t, Log(ic))
|
||
|
|
||
|
ls := buf.Lines()
|
||
|
require.Equal(t, 1, len(ls))
|
||
|
|
||
|
var logMsg map[string]interface{}
|
||
|
require.NoError(t, json.Unmarshal([]byte(ls[0]), &logMsg))
|
||
|
require.Equal(t, "info", logMsg["level"])
|
||
|
require.Equal(t, "runtime log", logMsg["msg"])
|
||
|
require.Equal(t, h.StringBE(), logMsg["script"])
|
||
|
require.Equal(t, `"hello"`, logMsg["logs"])
|
||
|
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func TestNotify(t *testing.T) {
|
||
|
h := random.Uint160()
|
||
|
newIC := func(name string, args interface{}) *interop.Context {
|
||
|
ic := &interop.Context{VM: vm.New()}
|
||
|
ic.VM.LoadScriptWithHash([]byte{1}, h, smartcontract.NoneFlag)
|
||
|
ic.VM.Estack().PushVal(args)
|
||
|
ic.VM.Estack().PushVal(name)
|
||
|
return ic
|
||
|
}
|
||
|
t.Run("big name", func(t *testing.T) {
|
||
|
ic := newIC(string(make([]byte, MaxEventNameLen+1)), []byte{42})
|
||
|
require.Error(t, Notify(ic))
|
||
|
})
|
||
|
t.Run("recursive struct", func(t *testing.T) {
|
||
|
arr := stackitem.NewArray([]stackitem.Item{stackitem.Null{}})
|
||
|
arr.Append(arr)
|
||
|
ic := newIC("event", arr)
|
||
|
require.Error(t, Notify(ic))
|
||
|
})
|
||
|
t.Run("big notification", func(t *testing.T) {
|
||
|
bs := stackitem.NewByteArray(make([]byte, MaxNotificationSize+1))
|
||
|
arr := stackitem.NewArray([]stackitem.Item{bs})
|
||
|
ic := newIC("event", arr)
|
||
|
require.Error(t, Notify(ic))
|
||
|
})
|
||
|
t.Run("good", func(t *testing.T) {
|
||
|
arr := stackitem.NewArray([]stackitem.Item{stackitem.Make(42)})
|
||
|
ic := newIC("good event", arr)
|
||
|
require.NoError(t, Notify(ic))
|
||
|
require.Equal(t, 1, len(ic.Notifications))
|
||
|
|
||
|
ev := ic.Notifications[0]
|
||
|
require.Equal(t, "good event", ev.Name)
|
||
|
require.Equal(t, h, ev.ScriptHash)
|
||
|
require.Equal(t, arr, ev.Item)
|
||
|
// Check deep copy.
|
||
|
arr.Value().([]stackitem.Item)[0] = stackitem.Null{}
|
||
|
require.NotEqual(t, arr, ev.Item)
|
||
|
})
|
||
|
}
|