294 lines
9.6 KiB
Go
294 lines
9.6 KiB
Go
package result
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/google/uuid"
|
|
"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
|
|
Session uuid.UUID
|
|
finalize func()
|
|
registerIterator RegisterIterator
|
|
}
|
|
|
|
// RegisterIterator is a callback used to register new iterator on the server side.
|
|
type RegisterIterator func(sessionID string, item stackitem.Item, id int, finalize func()) (uuid.UUID, error)
|
|
|
|
// 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, registerIterator RegisterIterator, 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,
|
|
finalize: ic.Finalize,
|
|
maxIteratorResultItems: maxIteratorResultItems,
|
|
registerIterator: registerIterator,
|
|
}
|
|
}
|
|
|
|
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"`
|
|
Session string `json:"session,omitempty"`
|
|
}
|
|
|
|
// iteratorInterfaceName is a string used to mark Iterator inside the InteropInterface.
|
|
const iteratorInterfaceName = "IIterator"
|
|
|
|
type iteratorAux struct {
|
|
Type string `json:"type"`
|
|
Interface string `json:"interface,omitempty"`
|
|
ID string `json:"id,omitempty"`
|
|
Value []json.RawMessage `json:"iterator,omitempty"`
|
|
Truncated bool `json:"truncated,omitempty"`
|
|
}
|
|
|
|
// Iterator represents VM iterator identifier. It either has ID set (for those JSON-RPC servers
|
|
// that support sessions) or non-nil Values and Truncated set (for those JSON-RPC servers that
|
|
// doesn't support sessions but perform in-place iterator traversing) or doesn't have ID, Values
|
|
// and Truncated set at all (for those JSON-RPC servers that doesn't support iterator sessions
|
|
// and doesn't perform in-place iterator traversing).
|
|
type Iterator struct {
|
|
// ID represents iterator ID. It is non-nil iff JSON-RPC server support session mechanism.
|
|
ID *uuid.UUID
|
|
|
|
// Values contains deserialized VM iterator values with a truncated flag. It is non-nil
|
|
// iff JSON-RPC server does not support sessions mechanism and able to traverse iterator.
|
|
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 or by the Server's
|
|
// sessions handler.
|
|
func (r *Invoke) Finalize() {
|
|
if r.finalize != nil {
|
|
r.finalize()
|
|
}
|
|
}
|
|
|
|
// MarshalJSON implements the json.Marshaler.
|
|
func (r Invoke) MarshalJSON() ([]byte, error) {
|
|
var (
|
|
st json.RawMessage
|
|
err error
|
|
faultSep string
|
|
arr = make([]json.RawMessage, len(r.Stack))
|
|
sessionsEnabled = r.registerIterator != nil
|
|
sessionID string
|
|
)
|
|
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]) {
|
|
if sessionsEnabled {
|
|
if sessionID == "" {
|
|
sessionID = uuid.NewString()
|
|
}
|
|
iteratorID, err := r.registerIterator(sessionID, r.Stack[i], i, r.finalize)
|
|
if err != nil {
|
|
// Call finalizer immediately, there can't be race between server and marshaller because session wasn't added to server's session pool.
|
|
r.Finalize()
|
|
return nil, fmt.Errorf("failed to register iterator session: %w", err)
|
|
}
|
|
data, err = json.Marshal(iteratorAux{
|
|
Type: stackitem.InteropT.String(),
|
|
Interface: iteratorInterfaceName,
|
|
ID: iteratorID.String(),
|
|
})
|
|
if err != nil {
|
|
r.FaultException += fmt.Sprintf("%sjson error: failed to marshal iterator: %v", faultSep, err)
|
|
break
|
|
}
|
|
} else {
|
|
iteratorValues, truncated := iterator.ValuesTruncated(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 {
|
|
r.FaultException += fmt.Sprintf("%sjson error: %v", faultSep, err)
|
|
break
|
|
}
|
|
}
|
|
} 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 !sessionsEnabled || sessionID == "" {
|
|
// Call finalizer manually if iterators are disabled or there's no unnested iterators on estack.
|
|
defer r.Finalize()
|
|
}
|
|
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,
|
|
Session: sessionID,
|
|
}
|
|
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
|
|
}
|
|
if len(aux.Session) != 0 {
|
|
r.Session, err = uuid.Parse(aux.Session)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to parse session ID: %w", 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 {
|
|
if len(iteratorAux.Interface) != 0 {
|
|
if iteratorAux.Interface != iteratorInterfaceName {
|
|
err = fmt.Errorf("unknown InteropInterface: %s", iteratorAux.Interface)
|
|
break
|
|
}
|
|
var iID uuid.UUID
|
|
iID, err = uuid.Parse(iteratorAux.ID) // iteratorAux.ID is always non-empty, see https://github.com/neo-project/neo-modules/pull/715#discussion_r897635424.
|
|
if err != nil {
|
|
err = fmt.Errorf("failed to unmarshal iterator ID: %w", err)
|
|
break
|
|
}
|
|
// It's impossible to restore initial iterator type; also iterator is almost
|
|
// useless outside the VM, thus let's replace it with a special structure.
|
|
st[i] = stackitem.NewInterop(Iterator{
|
|
ID: &iID,
|
|
})
|
|
} else {
|
|
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 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
|
|
}
|