vm: add 'events' command to VM CLI
And dump events automatically after HALTed or FAULTed end of execution.
This commit is contained in:
parent
f45d8fc08d
commit
f1ecdb82cc
3 changed files with 116 additions and 7 deletions
|
@ -225,6 +225,12 @@ example:
|
||||||
Description: "Dump opcodes of the current loaded program",
|
Description: "Dump opcodes of the current loaded program",
|
||||||
Action: handleOps,
|
Action: handleOps,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
Name: "events",
|
||||||
|
Usage: "Dump events emitted by the current loaded program",
|
||||||
|
Description: "Dump events emitted by the current loaded program",
|
||||||
|
Action: handleEvents,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
var completer *readline.PrefixCompleter
|
var completer *readline.PrefixCompleter
|
||||||
|
@ -626,12 +632,17 @@ func runVMWithHandling(c *cli.Context) {
|
||||||
writeErr(c.App.ErrWriter, err)
|
writeErr(c.App.ErrWriter, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
var message string
|
var (
|
||||||
|
message string
|
||||||
|
dumpNtf bool
|
||||||
|
)
|
||||||
switch {
|
switch {
|
||||||
case v.HasFailed():
|
case v.HasFailed():
|
||||||
message = "" // the error will be printed on return
|
message = "" // the error will be printed on return
|
||||||
|
dumpNtf = true
|
||||||
case v.HasHalted():
|
case v.HasHalted():
|
||||||
message = v.DumpEStack()
|
message = v.DumpEStack()
|
||||||
|
dumpNtf = true
|
||||||
case v.AtBreakpoint():
|
case v.AtBreakpoint():
|
||||||
ctx := v.Context()
|
ctx := v.Context()
|
||||||
if ctx.NextIP() < ctx.LenInstr() {
|
if ctx.NextIP() < ctx.LenInstr() {
|
||||||
|
@ -641,6 +652,16 @@ func runVMWithHandling(c *cli.Context) {
|
||||||
message = "execution has finished"
|
message = "execution has finished"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if dumpNtf {
|
||||||
|
var e string
|
||||||
|
e, err = dumpEvents(c.App)
|
||||||
|
if err == nil && len(e) != 0 {
|
||||||
|
if message != "" {
|
||||||
|
message += "\n"
|
||||||
|
}
|
||||||
|
message += "Events:\n" + e
|
||||||
|
}
|
||||||
|
}
|
||||||
if message != "" {
|
if message != "" {
|
||||||
fmt.Fprintln(c.App.Writer, message)
|
fmt.Fprintln(c.App.Writer, message)
|
||||||
}
|
}
|
||||||
|
@ -733,6 +754,28 @@ func changePrompt(app *cli.App) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func handleEvents(c *cli.Context) error {
|
||||||
|
e, err := dumpEvents(c.App)
|
||||||
|
if err != nil {
|
||||||
|
writeErr(c.App.ErrWriter, err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
fmt.Fprintln(c.App.Writer, e)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func dumpEvents(app *cli.App) (string, error) {
|
||||||
|
ic := getInteropContextFromContext(app)
|
||||||
|
if len(ic.Notifications) == 0 {
|
||||||
|
return "", nil
|
||||||
|
}
|
||||||
|
b, err := json.MarshalIndent(ic.Notifications, "", "\t")
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("failed to marshal notifications: %w", err)
|
||||||
|
}
|
||||||
|
return string(b), nil
|
||||||
|
}
|
||||||
|
|
||||||
// Run waits for user input from Stdin and executes the passed command.
|
// Run waits for user input from Stdin and executes the passed command.
|
||||||
func (c *VMCLI) Run() error {
|
func (c *VMCLI) Run() error {
|
||||||
if getPrintLogoFromContext(c.shell) {
|
if getPrintLogoFromContext(c.shell) {
|
||||||
|
|
|
@ -20,6 +20,7 @@ import (
|
||||||
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/config"
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/interop/interopnames"
|
"github.com/nspcc-dev/neo-go/pkg/core/interop/interopnames"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/storage"
|
"github.com/nspcc-dev/neo-go/pkg/core/storage"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/core/storage/dbconfig"
|
"github.com/nspcc-dev/neo-go/pkg/core/storage/dbconfig"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
||||||
|
@ -158,6 +159,12 @@ func (e *executor) checkNextLine(t *testing.T, expected string) {
|
||||||
require.Regexp(t, expected, line)
|
require.Regexp(t, expected, line)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *executor) checkNextLineExact(t *testing.T, expected string) {
|
||||||
|
line, err := e.out.ReadString('\n')
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, expected, line)
|
||||||
|
}
|
||||||
|
|
||||||
func (e *executor) checkError(t *testing.T, expectedErr error) {
|
func (e *executor) checkError(t *testing.T, expectedErr error) {
|
||||||
line, err := e.out.ReadString('\n')
|
line, err := e.out.ReadString('\n')
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
@ -190,6 +197,30 @@ func (e *executor) checkStack(t *testing.T, items ...interface{}) {
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *executor) checkEvents(t *testing.T, isKeywordExpected bool, events ...state.NotificationEvent) {
|
||||||
|
if isKeywordExpected {
|
||||||
|
e.checkNextLine(t, "Events:")
|
||||||
|
}
|
||||||
|
d := json.NewDecoder(e.out)
|
||||||
|
var actual interface{}
|
||||||
|
require.NoError(t, d.Decode(&actual))
|
||||||
|
rawActual, err := json.Marshal(actual)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
rawExpected, err := json.Marshal(events)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.JSONEq(t, string(rawExpected), string(rawActual))
|
||||||
|
|
||||||
|
// Decoder has it's own buffer, we need to return unread part to the output.
|
||||||
|
outRemain := e.out.String()
|
||||||
|
e.out.Reset()
|
||||||
|
_, err = gio.Copy(e.out, d.Buffered())
|
||||||
|
require.NoError(t, err)
|
||||||
|
e.out.WriteString(outRemain)
|
||||||
|
_, err = e.out.ReadString('\n')
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
func (e *executor) checkSlot(t *testing.T, items ...interface{}) {
|
func (e *executor) checkSlot(t *testing.T, items ...interface{}) {
|
||||||
d := json.NewDecoder(e.out)
|
d := json.NewDecoder(e.out)
|
||||||
var actual interface{}
|
var actual interface{}
|
||||||
|
@ -728,3 +759,30 @@ func TestRunWithState(t *testing.T) {
|
||||||
e.checkNextLine(t, "READY: loaded 37 instructions")
|
e.checkNextLine(t, "READY: loaded 37 instructions")
|
||||||
e.checkStack(t, 3)
|
e.checkStack(t, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestEvents(t *testing.T) {
|
||||||
|
e := newTestVMClIWithState(t)
|
||||||
|
|
||||||
|
script := io.NewBufBinWriter()
|
||||||
|
h, err := e.cli.chain.GetContractScriptHash(2) // examples/runtime/runtime.go
|
||||||
|
require.NoError(t, err)
|
||||||
|
emit.AppCall(script.BinWriter, h, "notify", callflag.All, []interface{}{true, 5})
|
||||||
|
e.runProg(t,
|
||||||
|
"loadhex "+hex.EncodeToString(script.Bytes()),
|
||||||
|
"run",
|
||||||
|
"events")
|
||||||
|
expectedEvent := state.NotificationEvent{
|
||||||
|
ScriptHash: h,
|
||||||
|
Name: "Event",
|
||||||
|
Item: stackitem.NewArray([]stackitem.Item{
|
||||||
|
stackitem.NewArray([]stackitem.Item{
|
||||||
|
stackitem.Make(true),
|
||||||
|
stackitem.Make(5),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
e.checkNextLine(t, "READY: loaded 44 instructions")
|
||||||
|
e.checkStack(t, stackitem.Null{})
|
||||||
|
e.checkEvents(t, true, expectedEvent) // automatically printed after `run` command
|
||||||
|
e.checkEvents(t, false, expectedEvent) // printed after `events` command
|
||||||
|
}
|
||||||
|
|
|
@ -23,7 +23,7 @@ import (
|
||||||
|
|
||||||
const neoAmount = 99999000
|
const neoAmount = 99999000
|
||||||
|
|
||||||
// InitSimple initializes chain with single contract storing several storage values.
|
// InitSimple initializes chain with simple contracts from 'examples' folder.
|
||||||
// It's not as complicated as chain got after Init and may be used for tests where
|
// It's not as complicated as chain got after Init and may be used for tests where
|
||||||
// chain with a small amount of data is needed and for historical functionality testing.
|
// chain with a small amount of data is needed and for historical functionality testing.
|
||||||
// Needs a path to the root directory.
|
// Needs a path to the root directory.
|
||||||
|
@ -31,14 +31,19 @@ func InitSimple(t *testing.T, rootpath string, e *neotest.Executor) {
|
||||||
// examplesPrefix is a prefix of the example smart-contracts.
|
// examplesPrefix is a prefix of the example smart-contracts.
|
||||||
var examplesPrefix = filepath.Join(rootpath, "examples")
|
var examplesPrefix = filepath.Join(rootpath, "examples")
|
||||||
|
|
||||||
|
deployExample := func(t *testing.T, name string) util.Uint160 {
|
||||||
|
_, h := newDeployTx(t, e, e.Validator,
|
||||||
|
filepath.Join(examplesPrefix, name, name+".go"),
|
||||||
|
filepath.Join(examplesPrefix, name, name+".yml"),
|
||||||
|
true)
|
||||||
|
return h
|
||||||
|
}
|
||||||
|
|
||||||
// Block #1: deploy storage contract (examples/storage/storage.go).
|
// Block #1: deploy storage contract (examples/storage/storage.go).
|
||||||
_, storageHash := newDeployTx(t, e, e.Validator,
|
storageHash := deployExample(t, "storage")
|
||||||
filepath.Join(examplesPrefix, "storage", "storage.go"),
|
storageValidatorInvoker := e.ValidatorInvoker(storageHash)
|
||||||
filepath.Join(examplesPrefix, "storage", "storage.yml"),
|
|
||||||
true)
|
|
||||||
|
|
||||||
// Block #2: put (1, 1) kv pair.
|
// Block #2: put (1, 1) kv pair.
|
||||||
storageValidatorInvoker := e.ValidatorInvoker(storageHash)
|
|
||||||
storageValidatorInvoker.Invoke(t, 1, "put", 1, 1)
|
storageValidatorInvoker.Invoke(t, 1, "put", 1, 1)
|
||||||
|
|
||||||
// Block #3: put (2, 2) kv pair.
|
// Block #3: put (2, 2) kv pair.
|
||||||
|
@ -46,6 +51,9 @@ func InitSimple(t *testing.T, rootpath string, e *neotest.Executor) {
|
||||||
|
|
||||||
// Block #4: update (1, 1) -> (1, 2).
|
// Block #4: update (1, 1) -> (1, 2).
|
||||||
storageValidatorInvoker.Invoke(t, 1, "put", 1, 2)
|
storageValidatorInvoker.Invoke(t, 1, "put", 1, 2)
|
||||||
|
|
||||||
|
// Block #5: deploy runtime contract (examples/runtime/runtime.go).
|
||||||
|
_ = deployExample(t, "runtime")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Init pushes some predefined set of transactions into the given chain, it needs a path to
|
// Init pushes some predefined set of transactions into the given chain, it needs a path to
|
||||||
|
|
Loading…
Reference in a new issue