forked from TrueCloudLab/neoneo-go
220 lines
6.3 KiB
Go
220 lines
6.3 KiB
Go
package result
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/interop"
|
|
"github.com/nspcc-dev/neo-go/pkg/core/interop/iterator"
|
|
"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/transaction"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
)
|
|
|
|
// Invoke represents a code invocation result and is used by several RPC calls
|
|
// that invoke functions, scripts and generic bytecode.
|
|
type Invoke struct {
|
|
State string
|
|
GasConsumed int64
|
|
Script []byte
|
|
Stack []stackitem.Item
|
|
FaultException string
|
|
Notifications []state.NotificationEvent
|
|
Transaction *transaction.Transaction
|
|
Diagnostics *InvokeDiag
|
|
maxIteratorResultItems int
|
|
finalize func()
|
|
}
|
|
|
|
// InvokeDiag is an additional diagnostic data for invocation.
|
|
type InvokeDiag struct {
|
|
Changes []storage.Operation `json:"storagechanges"`
|
|
Invocations []*vm.InvocationTree `json:"invokedcontracts"`
|
|
}
|
|
|
|
// NewInvoke returns a new Invoke structure with the given fields set.
|
|
func NewInvoke(ic *interop.Context, script []byte, faultException string, maxIteratorResultItems int) *Invoke {
|
|
var diag *InvokeDiag
|
|
tree := ic.VM.GetInvocationTree()
|
|
if tree != nil {
|
|
diag = &InvokeDiag{
|
|
Invocations: tree.Calls,
|
|
Changes: storage.BatchToOperations(ic.DAO.GetBatch()),
|
|
}
|
|
}
|
|
notifications := ic.Notifications
|
|
if notifications == nil {
|
|
notifications = make([]state.NotificationEvent, 0)
|
|
}
|
|
return &Invoke{
|
|
State: ic.VM.State().String(),
|
|
GasConsumed: ic.VM.GasConsumed(),
|
|
Script: script,
|
|
Stack: ic.VM.Estack().ToArray(),
|
|
FaultException: faultException,
|
|
Notifications: notifications,
|
|
Diagnostics: diag,
|
|
maxIteratorResultItems: maxIteratorResultItems,
|
|
finalize: ic.Finalize,
|
|
}
|
|
}
|
|
|
|
type invokeAux struct {
|
|
State string `json:"state"`
|
|
GasConsumed int64 `json:"gasconsumed,string"`
|
|
Script []byte `json:"script"`
|
|
Stack json.RawMessage `json:"stack"`
|
|
FaultException *string `json:"exception"`
|
|
Notifications []state.NotificationEvent `json:"notifications"`
|
|
Transaction []byte `json:"tx,omitempty"`
|
|
Diagnostics *InvokeDiag `json:"diagnostics,omitempty"`
|
|
}
|
|
|
|
type iteratorAux struct {
|
|
Type string `json:"type"`
|
|
Value []json.RawMessage `json:"iterator"`
|
|
Truncated bool `json:"truncated"`
|
|
}
|
|
|
|
// Iterator represents deserialized VM iterator values with a truncated flag.
|
|
type Iterator struct {
|
|
Values []stackitem.Item
|
|
Truncated bool
|
|
}
|
|
|
|
// Finalize releases resources occupied by Iterators created at the script invocation.
|
|
// This method will be called automatically on Invoke marshalling.
|
|
func (r *Invoke) Finalize() {
|
|
if r.finalize != nil {
|
|
r.finalize()
|
|
}
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler.
|
|
func (r Invoke) MarshalJSON() ([]byte, error) {
|
|
defer r.Finalize()
|
|
var (
|
|
st json.RawMessage
|
|
err error
|
|
faultSep string
|
|
arr = make([]json.RawMessage, len(r.Stack))
|
|
)
|
|
if len(r.FaultException) != 0 {
|
|
faultSep = " / "
|
|
}
|
|
arrloop:
|
|
for i := range arr {
|
|
var data []byte
|
|
if (r.Stack[i].Type() == stackitem.InteropT) && iterator.IsIterator(r.Stack[i]) {
|
|
iteratorValues, truncated := iterator.Values(r.Stack[i], r.maxIteratorResultItems)
|
|
value := make([]json.RawMessage, len(iteratorValues))
|
|
for j := range iteratorValues {
|
|
value[j], err = stackitem.ToJSONWithTypes(iteratorValues[j])
|
|
if err != nil {
|
|
r.FaultException += fmt.Sprintf("%sjson error: %v", faultSep, err)
|
|
break arrloop
|
|
}
|
|
}
|
|
data, err = json.Marshal(iteratorAux{
|
|
Type: stackitem.InteropT.String(),
|
|
Value: value,
|
|
Truncated: truncated,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to marshal iterator: %w", err)
|
|
}
|
|
} else {
|
|
data, err = stackitem.ToJSONWithTypes(r.Stack[i])
|
|
if err != nil {
|
|
r.FaultException += fmt.Sprintf("%sjson error: %v", faultSep, err)
|
|
break
|
|
}
|
|
}
|
|
arr[i] = data
|
|
}
|
|
|
|
if err == nil {
|
|
st, err = json.Marshal(arr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
var txbytes []byte
|
|
if r.Transaction != nil {
|
|
txbytes = r.Transaction.Bytes()
|
|
}
|
|
aux := &invokeAux{
|
|
GasConsumed: r.GasConsumed,
|
|
Script: r.Script,
|
|
State: r.State,
|
|
Stack: st,
|
|
Notifications: r.Notifications,
|
|
Transaction: txbytes,
|
|
Diagnostics: r.Diagnostics,
|
|
}
|
|
if len(r.FaultException) != 0 {
|
|
aux.FaultException = &r.FaultException
|
|
}
|
|
return json.Marshal(aux)
|
|
}
|
|
|
|
// UnmarshalJSON implements the json.Unmarshaler.
|
|
func (r *Invoke) UnmarshalJSON(data []byte) error {
|
|
var err error
|
|
aux := new(invokeAux)
|
|
if err = json.Unmarshal(data, aux); err != nil {
|
|
return err
|
|
}
|
|
var arr []json.RawMessage
|
|
if err = json.Unmarshal(aux.Stack, &arr); err == nil {
|
|
st := make([]stackitem.Item, len(arr))
|
|
for i := range arr {
|
|
st[i], err = stackitem.FromJSONWithTypes(arr[i])
|
|
if err != nil {
|
|
break
|
|
}
|
|
if st[i].Type() == stackitem.InteropT {
|
|
iteratorAux := new(iteratorAux)
|
|
if json.Unmarshal(arr[i], iteratorAux) == nil {
|
|
iteratorValues := make([]stackitem.Item, len(iteratorAux.Value))
|
|
for j := range iteratorValues {
|
|
iteratorValues[j], err = stackitem.FromJSONWithTypes(iteratorAux.Value[j])
|
|
if err != nil {
|
|
err = fmt.Errorf("failed to unmarshal iterator values: %w", err)
|
|
break
|
|
}
|
|
}
|
|
|
|
// it's impossible to restore initial iterator type; also iterator is almost
|
|
// useless outside of the VM, thus let's replace it with a special structure.
|
|
st[i] = stackitem.NewInterop(Iterator{
|
|
Values: iteratorValues,
|
|
Truncated: iteratorAux.Truncated,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
if err == nil {
|
|
r.Stack = st
|
|
}
|
|
}
|
|
var tx *transaction.Transaction
|
|
if len(aux.Transaction) != 0 {
|
|
tx, err = transaction.NewTransactionFromBytes(aux.Transaction)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
r.GasConsumed = aux.GasConsumed
|
|
r.Script = aux.Script
|
|
r.State = aux.State
|
|
if aux.FaultException != nil {
|
|
r.FaultException = *aux.FaultException
|
|
}
|
|
r.Notifications = aux.Notifications
|
|
r.Transaction = tx
|
|
r.Diagnostics = aux.Diagnostics
|
|
return nil
|
|
}
|