forked from TrueCloudLab/neoneo-go
vm: allow historic calls inside VM CLI
This commit is contained in:
parent
79e13f73d8
commit
ff03c33e6d
2 changed files with 97 additions and 22 deletions
|
@ -54,9 +54,16 @@ const (
|
||||||
|
|
||||||
// Various flag names.
|
// Various flag names.
|
||||||
const (
|
const (
|
||||||
verboseFlagFullName = "verbose"
|
verboseFlagFullName = "verbose"
|
||||||
|
historicFlagFullName = "historic"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var historicFlag = cli.IntFlag{
|
||||||
|
Name: historicFlagFullName,
|
||||||
|
Usage: "Height for historic script invocation (for MPT-enabled blockchain configuration with KeepOnlyLatestState setting disabled). " +
|
||||||
|
"Assuming that block N-th is specified as an argument, the historic invocation is based on the storage state of height N and fake currently-accepting block with index N+1.",
|
||||||
|
}
|
||||||
|
|
||||||
var commands = []cli.Command{
|
var commands = []cli.Command{
|
||||||
{
|
{
|
||||||
Name: "exit",
|
Name: "exit",
|
||||||
|
@ -113,7 +120,8 @@ var commands = []cli.Command{
|
||||||
Name: "loadnef",
|
Name: "loadnef",
|
||||||
Usage: "Load a NEF-consistent script into the VM",
|
Usage: "Load a NEF-consistent script into the VM",
|
||||||
UsageText: `loadnef <file> <manifest>`,
|
UsageText: `loadnef <file> <manifest>`,
|
||||||
Description: `loadnef <file> <manifest>
|
Flags: []cli.Flag{historicFlag},
|
||||||
|
Description: `loadnef [--historic <height>] <file> <manifest>
|
||||||
both parameters are mandatory, example:
|
both parameters are mandatory, example:
|
||||||
> loadnef /path/to/script.nef /path/to/manifest.json`,
|
> loadnef /path/to/script.nef /path/to/manifest.json`,
|
||||||
Action: handleLoadNEF,
|
Action: handleLoadNEF,
|
||||||
|
@ -121,8 +129,9 @@ both parameters are mandatory, example:
|
||||||
{
|
{
|
||||||
Name: "loadbase64",
|
Name: "loadbase64",
|
||||||
Usage: "Load a base64-encoded script string into the VM",
|
Usage: "Load a base64-encoded script string into the VM",
|
||||||
UsageText: `loadbase64 <string>`,
|
UsageText: `loadbase64 [--historic <height>] <string>`,
|
||||||
Description: `loadbase64 <string>
|
Flags: []cli.Flag{historicFlag},
|
||||||
|
Description: `loadbase64 [--historic <height>] <string>
|
||||||
|
|
||||||
<string> is mandatory parameter, example:
|
<string> is mandatory parameter, example:
|
||||||
> loadbase64 AwAQpdToAAAADBQV9ehtQR1OrVZVhtHtoUHRfoE+agwUzmFvf3Rhfg/EuAVYOvJgKiON9j8TwAwIdHJhbnNmZXIMFDt9NxHG8Mz5sdypA9G/odiW8SOMQWJ9W1I4`,
|
> loadbase64 AwAQpdToAAAADBQV9ehtQR1OrVZVhtHtoUHRfoE+agwUzmFvf3Rhfg/EuAVYOvJgKiON9j8TwAwIdHJhbnNmZXIMFDt9NxHG8Mz5sdypA9G/odiW8SOMQWJ9W1I4`,
|
||||||
|
@ -131,8 +140,9 @@ both parameters are mandatory, example:
|
||||||
{
|
{
|
||||||
Name: "loadhex",
|
Name: "loadhex",
|
||||||
Usage: "Load a hex-encoded script string into the VM",
|
Usage: "Load a hex-encoded script string into the VM",
|
||||||
UsageText: `loadhex <string>`,
|
UsageText: `loadhex [--historic <height>] <string>`,
|
||||||
Description: `loadhex <string>
|
Flags: []cli.Flag{historicFlag},
|
||||||
|
Description: `loadhex [--historic <height>] <string>
|
||||||
|
|
||||||
<string> is mandatory parameter, example:
|
<string> is mandatory parameter, example:
|
||||||
> loadhex 0c0c48656c6c6f20776f726c6421`,
|
> loadhex 0c0c48656c6c6f20776f726c6421`,
|
||||||
|
@ -141,8 +151,9 @@ both parameters are mandatory, example:
|
||||||
{
|
{
|
||||||
Name: "loadgo",
|
Name: "loadgo",
|
||||||
Usage: "Compile and load a Go file with the manifest into the VM",
|
Usage: "Compile and load a Go file with the manifest into the VM",
|
||||||
UsageText: `loadgo <file>`,
|
UsageText: `loadgo [--historic <height>] <file>`,
|
||||||
Description: `loadgo <file>
|
Flags: []cli.Flag{historicFlag},
|
||||||
|
Description: `loadgo [--historic <height>] <file>
|
||||||
|
|
||||||
<file> is mandatory parameter, example:
|
<file> is mandatory parameter, example:
|
||||||
> loadgo /path/to/file.go`,
|
> loadgo /path/to/file.go`,
|
||||||
|
@ -150,7 +161,8 @@ both parameters are mandatory, example:
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
Name: "reset",
|
Name: "reset",
|
||||||
Usage: "Unload compiled script from the VM",
|
Usage: "Unload compiled script from the VM and reset context to proper (possibly, historic) state",
|
||||||
|
Flags: []cli.Flag{historicFlag},
|
||||||
Action: handleReset,
|
Action: handleReset,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -485,8 +497,19 @@ func handleSlots(c *cli.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// prepareVM retrieves --historic flag from context (if set) and resets app state
|
||||||
|
// (to the specified historic height if given).
|
||||||
|
func prepareVM(c *cli.Context) error {
|
||||||
|
if c.IsSet(historicFlagFullName) {
|
||||||
|
height := c.Int(historicFlagFullName)
|
||||||
|
return resetState(c.App, uint32(height))
|
||||||
|
}
|
||||||
|
|
||||||
|
return resetState(c.App)
|
||||||
|
}
|
||||||
|
|
||||||
func handleLoadNEF(c *cli.Context) error {
|
func handleLoadNEF(c *cli.Context) error {
|
||||||
err := resetState(c.App)
|
err := prepareVM(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -509,7 +532,7 @@ func handleLoadNEF(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleLoadBase64(c *cli.Context) error {
|
func handleLoadBase64(c *cli.Context) error {
|
||||||
err := resetState(c.App)
|
err := prepareVM(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -529,7 +552,7 @@ func handleLoadBase64(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleLoadHex(c *cli.Context) error {
|
func handleLoadHex(c *cli.Context) error {
|
||||||
err := resetState(c.App)
|
err := prepareVM(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -549,7 +572,7 @@ func handleLoadHex(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleLoadGo(c *cli.Context) error {
|
func handleLoadGo(c *cli.Context) error {
|
||||||
err := resetState(c.App)
|
err := prepareVM(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -579,7 +602,7 @@ func handleLoadGo(c *cli.Context) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleReset(c *cli.Context) error {
|
func handleReset(c *cli.Context) error {
|
||||||
err := resetState(c.App)
|
err := prepareVM(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -595,13 +618,25 @@ func finalizeInteropContext(app *cli.App) {
|
||||||
|
|
||||||
// resetInteropContext calls finalizer for current interop context and replaces
|
// resetInteropContext calls finalizer for current interop context and replaces
|
||||||
// it with the newly created one.
|
// it with the newly created one.
|
||||||
func resetInteropContext(app *cli.App) error {
|
func resetInteropContext(app *cli.App, height ...uint32) error {
|
||||||
finalizeInteropContext(app)
|
finalizeInteropContext(app)
|
||||||
bc := getChainFromContext(app)
|
bc := getChainFromContext(app)
|
||||||
newIc, err := bc.GetTestVM(trigger.Application, nil, nil)
|
var (
|
||||||
if err != nil {
|
newIc *interop.Context
|
||||||
return fmt.Errorf("failed to create test VM: %w", err)
|
err error
|
||||||
|
)
|
||||||
|
if len(height) != 0 {
|
||||||
|
newIc, err = bc.GetTestHistoricVM(trigger.Application, nil, height[0]+1)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create historic VM for height %d: %w", height[0], err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
newIc, err = bc.GetTestVM(trigger.Application, nil, nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create VM: %w", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setInteropContextInContext(app, newIc)
|
setInteropContextInContext(app, newIc)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -613,10 +648,10 @@ func resetManifest(app *cli.App) {
|
||||||
|
|
||||||
// resetState resets state of the app (clear interop context and manifest) so that it's ready
|
// resetState resets state of the app (clear interop context and manifest) so that it's ready
|
||||||
// to load new program.
|
// to load new program.
|
||||||
func resetState(app *cli.App) error {
|
func resetState(app *cli.App, height ...uint32) error {
|
||||||
err := resetInteropContext(app)
|
err := resetInteropContext(app, height...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to reset interop context state: %w", err)
|
return err
|
||||||
}
|
}
|
||||||
resetManifest(app)
|
resetManifest(app)
|
||||||
return nil
|
return nil
|
||||||
|
@ -822,7 +857,9 @@ func handleEvents(c *cli.Context) error {
|
||||||
func handleEnv(c *cli.Context) error {
|
func handleEnv(c *cli.Context) error {
|
||||||
bc := getChainFromContext(c.App)
|
bc := getChainFromContext(c.App)
|
||||||
cfg := getChainConfigFromContext(c.App)
|
cfg := getChainConfigFromContext(c.App)
|
||||||
message := fmt.Sprintf("Chain height: %d\nNetwork magic: %d\nDB type: %s\n", bc.BlockHeight(), bc.GetConfig().Magic, cfg.ApplicationConfiguration.DBConfiguration.Type)
|
ic := getInteropContextFromContext(c.App)
|
||||||
|
message := fmt.Sprintf("Chain height: %d\nVM height (may differ from chain height in case of historic call): %d\nNetwork magic: %d\nDB type: %s\n",
|
||||||
|
bc.BlockHeight(), ic.BlockHeight(), bc.GetConfig().Magic, cfg.ApplicationConfiguration.DBConfiguration.Type)
|
||||||
if c.Bool(verboseFlagFullName) {
|
if c.Bool(verboseFlagFullName) {
|
||||||
cfgBytes, err := json.MarshalIndent(cfg, "", "\t")
|
cfgBytes, err := json.MarshalIndent(cfg, "", "\t")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -760,6 +760,31 @@ func TestRunWithState(t *testing.T) {
|
||||||
e.checkStack(t, 3)
|
e.checkStack(t, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestRunWithHistoricState(t *testing.T) {
|
||||||
|
e := newTestVMClIWithState(t)
|
||||||
|
|
||||||
|
script := io.NewBufBinWriter()
|
||||||
|
h, err := e.cli.chain.GetContractScriptHash(1) // examples/storage/storage.go
|
||||||
|
require.NoError(t, err)
|
||||||
|
emit.AppCall(script.BinWriter, h, "get", callflag.All, 1)
|
||||||
|
b := script.Bytes()
|
||||||
|
|
||||||
|
e.runProg(t,
|
||||||
|
"loadhex "+hex.EncodeToString(b), // normal invocation
|
||||||
|
"run",
|
||||||
|
"loadhex --historic 3 "+hex.EncodeToString(b), // historic invocation, old value should be retrieved
|
||||||
|
"run",
|
||||||
|
"loadhex --historic 0 "+hex.EncodeToString(b), // historic invocation, contract is not deployed yet
|
||||||
|
"run",
|
||||||
|
)
|
||||||
|
e.checkNextLine(t, "READY: loaded 36 instructions")
|
||||||
|
e.checkStack(t, []byte{2})
|
||||||
|
e.checkNextLine(t, "READY: loaded 36 instructions")
|
||||||
|
e.checkStack(t, []byte{1})
|
||||||
|
e.checkNextLine(t, "READY: loaded 36 instructions")
|
||||||
|
e.checkNextLineExact(t, "Error: at instruction 31 (SYSCALL): failed to invoke syscall 1381727586: called contract a00e3c2643a08a452d8b0bdd31849ae11a17c445 not found: key not found\n")
|
||||||
|
}
|
||||||
|
|
||||||
func TestEvents(t *testing.T) {
|
func TestEvents(t *testing.T) {
|
||||||
e := newTestVMClIWithState(t)
|
e := newTestVMClIWithState(t)
|
||||||
|
|
||||||
|
@ -792,6 +817,7 @@ func TestEnv(t *testing.T) {
|
||||||
e := newTestVMCLI(t)
|
e := newTestVMCLI(t)
|
||||||
e.runProg(t, "env")
|
e.runProg(t, "env")
|
||||||
e.checkNextLine(t, "Chain height: 0")
|
e.checkNextLine(t, "Chain height: 0")
|
||||||
|
e.checkNextLineExact(t, "VM height (may differ from chain height in case of historic call): 0\n")
|
||||||
e.checkNextLine(t, "Network magic: 42")
|
e.checkNextLine(t, "Network magic: 42")
|
||||||
e.checkNextLine(t, "DB type: inmemory")
|
e.checkNextLine(t, "DB type: inmemory")
|
||||||
})
|
})
|
||||||
|
@ -799,6 +825,17 @@ func TestEnv(t *testing.T) {
|
||||||
e := newTestVMClIWithState(t)
|
e := newTestVMClIWithState(t)
|
||||||
e.runProg(t, "env")
|
e.runProg(t, "env")
|
||||||
e.checkNextLine(t, "Chain height: 5")
|
e.checkNextLine(t, "Chain height: 5")
|
||||||
|
e.checkNextLineExact(t, "VM height (may differ from chain height in case of historic call): 5\n")
|
||||||
|
e.checkNextLine(t, "Network magic: 42")
|
||||||
|
e.checkNextLine(t, "DB type: leveldb")
|
||||||
|
})
|
||||||
|
t.Run("setup with historic state", func(t *testing.T) {
|
||||||
|
e := newTestVMClIWithState(t)
|
||||||
|
e.runProg(t, "loadbase64 --historic 3 "+base64.StdEncoding.EncodeToString([]byte{byte(opcode.PUSH1)}),
|
||||||
|
"env")
|
||||||
|
e.checkNextLine(t, "READY: loaded 1 instructions")
|
||||||
|
e.checkNextLine(t, "Chain height: 5")
|
||||||
|
e.checkNextLineExact(t, "VM height (may differ from chain height in case of historic call): 3\n")
|
||||||
e.checkNextLine(t, "Network magic: 42")
|
e.checkNextLine(t, "Network magic: 42")
|
||||||
e.checkNextLine(t, "DB type: leveldb")
|
e.checkNextLine(t, "DB type: leveldb")
|
||||||
})
|
})
|
||||||
|
@ -806,6 +843,7 @@ func TestEnv(t *testing.T) {
|
||||||
e := newTestVMClIWithState(t)
|
e := newTestVMClIWithState(t)
|
||||||
e.runProg(t, "env -v")
|
e.runProg(t, "env -v")
|
||||||
e.checkNextLine(t, "Chain height: 5")
|
e.checkNextLine(t, "Chain height: 5")
|
||||||
|
e.checkNextLineExact(t, "VM height (may differ from chain height in case of historic call): 5\n")
|
||||||
e.checkNextLine(t, "Network magic: 42")
|
e.checkNextLine(t, "Network magic: 42")
|
||||||
e.checkNextLine(t, "DB type: leveldb")
|
e.checkNextLine(t, "DB type: leveldb")
|
||||||
e.checkNextLine(t, "Node config:") // Do not check exact node config.
|
e.checkNextLine(t, "Node config:") // Do not check exact node config.
|
||||||
|
|
Loading…
Reference in a new issue