From 18b331d76507874653c5e6853f0493b3535b0817 Mon Sep 17 00:00:00 2001 From: Evgenii Stratonikov Date: Fri, 11 Dec 2020 13:23:07 +0300 Subject: [PATCH] stackitem: fix JSON encoding Encode both `Buffer` and `ByteString` to UTF-8 bytes. Follow https://github.com/neo-project/neo/pull/1715 . --- pkg/vm/stackitem/json.go | 20 ++++++++++---------- pkg/vm/stackitem/json_test.go | 13 ++++--------- 2 files changed, 14 insertions(+), 19 deletions(-) diff --git a/pkg/vm/stackitem/json.go b/pkg/vm/stackitem/json.go index a1886ef30..610fe17c3 100644 --- a/pkg/vm/stackitem/json.go +++ b/pkg/vm/stackitem/json.go @@ -88,12 +88,16 @@ func toJSON(buf *io.BufBinWriter, item Item) { return } w.WriteBytes([]byte(it.value.String())) - case *ByteArray: + case *ByteArray, *Buffer: w.WriteB('"') - val := it.Value().([]byte) - b := make([]byte, base64.StdEncoding.EncodedLen(len(val))) - base64.StdEncoding.Encode(b, val) - w.WriteBytes(b) + s, err := ToString(it) + if err != nil { + if buf.Err == nil { + buf.Err = err + } + return + } + w.WriteBytes([]byte(s)) w.WriteB('"') case *Bool: if it.value { @@ -158,11 +162,7 @@ func (d *decoder) decode() (Item, error) { return nil, nil } case string: - b, err := base64.StdEncoding.DecodeString(t) - if err != nil { - return nil, err - } - return NewByteArray(b), nil + return NewByteArray([]byte(t)), nil case float64: if math.Floor(t) != t { return nil, fmt.Errorf("real value is not allowed: %v", t) diff --git a/pkg/vm/stackitem/json_test.go b/pkg/vm/stackitem/json_test.go index c8e15dbed..fb6fcc672 100644 --- a/pkg/vm/stackitem/json_test.go +++ b/pkg/vm/stackitem/json_test.go @@ -1,7 +1,6 @@ package stackitem import ( - "encoding/base64" "math/big" "testing" @@ -28,10 +27,9 @@ func getTestDecodeFunc(js string, expected ...interface{}) func(t *testing.T) { } func TestFromToJSON(t *testing.T) { - var testBase64 = base64.StdEncoding.EncodeToString([]byte("test")) t.Run("ByteString", func(t *testing.T) { t.Run("Empty", getTestDecodeFunc(`""`, []byte{})) - t.Run("Base64", getTestDecodeFunc(`"`+testBase64+`"`, "test")) + t.Run("Base64", getTestDecodeFunc(`"test"`, "test")) }) t.Run("BigInteger", func(t *testing.T) { t.Run("ZeroFloat", getTestDecodeFunc(`12.000`, 12, nil)) @@ -46,7 +44,7 @@ func TestFromToJSON(t *testing.T) { t.Run("Null", getTestDecodeFunc(`null`, Null{})) t.Run("Array", func(t *testing.T) { t.Run("Empty", getTestDecodeFunc(`[]`, NewArray([]Item{}))) - t.Run("Simple", getTestDecodeFunc((`[1,"`+testBase64+`",true,null]`), + t.Run("Simple", getTestDecodeFunc((`[1,"test",true,null]`), NewArray([]Item{NewBigInteger(big.NewInt(1)), NewByteArray([]byte("test")), NewBool(true), Null{}}))) t.Run("Nested", getTestDecodeFunc(`[[],[{},null]]`, NewArray([]Item{NewArray([]Item{}), NewArray([]Item{NewMap(), Null{}})}))) @@ -59,11 +57,10 @@ func TestFromToJSON(t *testing.T) { large.Add(NewByteArray([]byte("arr")), NewArray([]Item{NewByteArray([]byte("test"))})) t.Run("Empty", getTestDecodeFunc(`{}`, NewMap())) t.Run("Small", getTestDecodeFunc(`{"a":3}`, small)) - t.Run("Big", getTestDecodeFunc(`{"3":{"a":3},"arr":["`+testBase64+`"]}`, large)) + t.Run("Big", getTestDecodeFunc(`{"3":{"a":3},"arr":["test"]}`, large)) }) t.Run("Invalid", func(t *testing.T) { t.Run("Empty", getTestDecodeFunc(``, nil)) - t.Run("InvalidString", getTestDecodeFunc(`"not a base64"`, nil)) t.Run("InvalidArray", getTestDecodeFunc(`[}`, nil)) t.Run("InvalidMap", getTestDecodeFunc(`{]`, nil)) t.Run("InvalidMapValue", getTestDecodeFunc(`{"a":{]}`, nil)) @@ -79,9 +76,7 @@ func TestFromToJSON(t *testing.T) { require.Error(t, err) }) t.Run("BigByteArray", func(t *testing.T) { - l := base64.StdEncoding.DecodedLen(MaxSize + 8) - require.True(t, l < MaxSize) // check if test makes sense - item := NewByteArray(make([]byte, l)) + item := NewByteArray(make([]byte, MaxSize)) _, err := ToJSON(item) require.Error(t, err) })