diff --git a/pkg/vm/stackitem/serialization.go b/pkg/vm/stackitem/serialization.go index 960cb4252..93b625095 100644 --- a/pkg/vm/stackitem/serialization.go +++ b/pkg/vm/stackitem/serialization.go @@ -9,10 +9,22 @@ import ( "github.com/nspcc-dev/neo-go/pkg/io" ) +// serContext is an internal serialization context. +type serContext struct { + *io.BinWriter + allowInvalid bool + seen map[Item]bool +} + // SerializeItem encodes given Item into the byte slice. func SerializeItem(item Item) ([]byte, error) { w := io.NewBufBinWriter() - EncodeBinaryStackItem(item, w.BinWriter) + sc := serContext{ + BinWriter: w.BinWriter, + allowInvalid: false, + seen: make(map[Item]bool), + } + sc.serialize(item) if w.Err != nil { return nil, w.Err } @@ -23,14 +35,24 @@ func SerializeItem(item Item) ([]byte, error) { // similar to io.Serializable's EncodeBinary, but works with Item // interface. func EncodeBinaryStackItem(item Item, w *io.BinWriter) { - serializeItemTo(item, w, false, make(map[Item]bool)) + sc := serContext{ + BinWriter: w, + allowInvalid: false, + seen: make(map[Item]bool), + } + sc.serialize(item) } // EncodeBinaryStackItemAppExec encodes given Item into the given BinWriter. It's // similar to EncodeBinaryStackItem but allows to encode interop (only type, value is lost). func EncodeBinaryStackItemAppExec(item Item, w *io.BinWriter) { bw := io.NewBufBinWriter() - serializeItemTo(item, bw.BinWriter, true, make(map[Item]bool)) + sc := serContext{ + BinWriter: bw.BinWriter, + allowInvalid: true, + seen: make(map[Item]bool), + } + sc.serialize(item) if bw.Err != nil { w.WriteBytes([]byte{byte(InvalidT)}) return @@ -38,15 +60,15 @@ func EncodeBinaryStackItemAppExec(item Item, w *io.BinWriter) { w.WriteBytes(bw.Bytes()) } -func serializeItemTo(item Item, w *io.BinWriter, allowInvalid bool, seen map[Item]bool) { +func (w *serContext) serialize(item Item) { if w.Err != nil { return } - if seen[item] { + if w.seen[item] { w.Err = errors.New("recursive structures can't be serialized") return } - if item == nil && allowInvalid { + if item == nil && w.allowInvalid { w.WriteBytes([]byte{byte(InvalidT)}) return } @@ -65,13 +87,13 @@ func serializeItemTo(item Item, w *io.BinWriter, allowInvalid bool, seen map[Ite w.WriteBytes([]byte{byte(IntegerT)}) w.WriteVarBytes(bigint.ToBytes(t.Value().(*big.Int))) case *Interop: - if allowInvalid { + if w.allowInvalid { w.WriteBytes([]byte{byte(InteropT)}) - return + } else { + w.Err = errors.New("interop item can't be serialized") } - w.Err = errors.New("interop item can't be serialized") case *Array, *Struct: - seen[item] = true + w.seen[item] = true _, isArray := t.(*Array) if isArray { @@ -83,19 +105,19 @@ func serializeItemTo(item Item, w *io.BinWriter, allowInvalid bool, seen map[Ite arr := t.Value().([]Item) w.WriteVarUint(uint64(len(arr))) for i := range arr { - serializeItemTo(arr[i], w, allowInvalid, seen) + w.serialize(arr[i]) } - delete(seen, item) + delete(w.seen, item) case *Map: - seen[item] = true + w.seen[item] = true w.WriteBytes([]byte{byte(MapT)}) w.WriteVarUint(uint64(len(t.Value().([]MapElement)))) for i := range t.Value().([]MapElement) { - serializeItemTo(t.Value().([]MapElement)[i].Key, w, allowInvalid, seen) - serializeItemTo(t.Value().([]MapElement)[i].Value, w, allowInvalid, seen) + w.serialize(t.Value().([]MapElement)[i].Key) + w.serialize(t.Value().([]MapElement)[i].Value) } - delete(seen, item) + delete(w.seen, item) case Null: w.WriteB(byte(AnyT)) }