mirror of
https://github.com/json-iterator/go.git
synced 2024-11-24 08:22:14 +02:00
343 lines
7.7 KiB
Go
343 lines
7.7 KiB
Go
package jsoniter
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func Test_empty_object(t *testing.T) {
|
|
should := require.New(t)
|
|
iter := ParseString(ConfigDefault, `{}`)
|
|
field := iter.ReadObject()
|
|
should.Equal("", field)
|
|
iter = ParseString(ConfigDefault, `{}`)
|
|
iter.ReadObjectCB(func(iter *Iterator, field string) bool {
|
|
should.FailNow("should not call")
|
|
return true
|
|
})
|
|
}
|
|
|
|
func Test_one_field(t *testing.T) {
|
|
should := require.New(t)
|
|
iter := ParseString(ConfigDefault, `{"a": "stream"}`)
|
|
field := iter.ReadObject()
|
|
should.Equal("a", field)
|
|
value := iter.ReadString()
|
|
should.Equal("stream", value)
|
|
field = iter.ReadObject()
|
|
should.Equal("", field)
|
|
iter = ParseString(ConfigDefault, `{"a": "stream"}`)
|
|
should.True(iter.ReadObjectCB(func(iter *Iterator, field string) bool {
|
|
should.Equal("a", field)
|
|
iter.Skip()
|
|
return true
|
|
}))
|
|
|
|
}
|
|
|
|
func Test_two_field(t *testing.T) {
|
|
should := require.New(t)
|
|
iter := ParseString(ConfigDefault, `{ "a": "stream" , "c": "d" }`)
|
|
field := iter.ReadObject()
|
|
should.Equal("a", field)
|
|
value := iter.ReadString()
|
|
should.Equal("stream", value)
|
|
field = iter.ReadObject()
|
|
should.Equal("c", field)
|
|
value = iter.ReadString()
|
|
should.Equal("d", value)
|
|
field = iter.ReadObject()
|
|
should.Equal("", field)
|
|
iter = ParseString(ConfigDefault, `{"field1": "1", "field2": 2}`)
|
|
for field := iter.ReadObject(); field != ""; field = iter.ReadObject() {
|
|
switch field {
|
|
case "field1":
|
|
iter.ReadString()
|
|
case "field2":
|
|
iter.ReadInt64()
|
|
default:
|
|
iter.ReportError("bind object", "unexpected field")
|
|
}
|
|
}
|
|
}
|
|
|
|
func Test_object_wrapper_any_get_all(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 []int
|
|
Field2 []int
|
|
}
|
|
any := Wrap(TestObject{[]int{1, 2}, []int{3, 4}})
|
|
should.Contains(any.Get('*', 0).ToString(), `"Field2":3`)
|
|
should.Contains(any.Keys(), "Field1")
|
|
should.Contains(any.Keys(), "Field2")
|
|
should.NotContains(any.Keys(), "Field3")
|
|
|
|
//should.Contains(any.GetObject()["Field1"].GetArray()[0], 1)
|
|
}
|
|
|
|
func Test_write_object(t *testing.T) {
|
|
should := require.New(t)
|
|
buf := &bytes.Buffer{}
|
|
stream := NewStream(Config{IndentionStep: 2}.Froze(), buf, 4096)
|
|
stream.WriteObjectStart()
|
|
stream.WriteObjectField("hello")
|
|
stream.WriteInt(1)
|
|
stream.WriteMore()
|
|
stream.WriteObjectField("world")
|
|
stream.WriteInt(2)
|
|
stream.WriteObjectEnd()
|
|
stream.Flush()
|
|
should.Nil(stream.Error)
|
|
should.Equal("{\n \"hello\": 1,\n \"world\": 2\n}", buf.String())
|
|
}
|
|
|
|
func Test_write_val_zero_field_struct(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
}
|
|
obj := TestObject{}
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{}`, str)
|
|
}
|
|
|
|
func Test_write_val_one_field_struct(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 string `json:"field-1"`
|
|
}
|
|
obj := TestObject{"hello"}
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"field-1":"hello"}`, str)
|
|
}
|
|
|
|
func Test_mixed(t *testing.T) {
|
|
should := require.New(t)
|
|
type AA struct {
|
|
ID int `json:"id"`
|
|
Payload map[string]interface{} `json:"payload"`
|
|
buf *bytes.Buffer
|
|
}
|
|
aa := AA{}
|
|
err := UnmarshalFromString(` {"id":1, "payload":{"account":"123","password":"456"}}`, &aa)
|
|
should.Nil(err)
|
|
should.Equal(1, aa.ID)
|
|
should.Equal("123", aa.Payload["account"])
|
|
}
|
|
|
|
func Test_omit_empty(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 string `json:"field-1,omitempty"`
|
|
Field2 string `json:"field-2,omitempty"`
|
|
Field3 string `json:"field-3,omitempty"`
|
|
}
|
|
obj := TestObject{}
|
|
obj.Field2 = "hello"
|
|
str, err := MarshalToString(&obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"field-2":"hello"}`, str)
|
|
}
|
|
|
|
func Test_ignore_field_on_not_valid_type(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 string `json:"field-1,omitempty"`
|
|
Field2 func() `json:"-"`
|
|
}
|
|
obj := TestObject{}
|
|
obj.Field1 = "hello world"
|
|
obj.Field2 = func() {}
|
|
str, err := MarshalToString(&obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"field-1":"hello world"}`, str)
|
|
}
|
|
|
|
func Test_recursive_struct(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 string
|
|
Me *TestObject
|
|
}
|
|
obj := TestObject{}
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Contains(str, `"Field1":""`)
|
|
should.Contains(str, `"Me":null`)
|
|
err = UnmarshalFromString(str, &obj)
|
|
should.Nil(err)
|
|
}
|
|
|
|
func Test_encode_anonymous_struct(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field string
|
|
}
|
|
str, err := MarshalToString(struct {
|
|
TestObject
|
|
Field int
|
|
}{
|
|
Field: 100,
|
|
})
|
|
should.Nil(err)
|
|
should.Equal(`{"Field":100}`, str)
|
|
}
|
|
|
|
func Test_decode_anonymous_struct(t *testing.T) {
|
|
should := require.New(t)
|
|
type Inner struct {
|
|
Key string `json:"key"`
|
|
}
|
|
|
|
type Outer struct {
|
|
Inner
|
|
}
|
|
var outer Outer
|
|
j := []byte("{\"key\":\"value\"}")
|
|
should.Nil(Unmarshal(j, &outer))
|
|
should.Equal("value", outer.Key)
|
|
}
|
|
|
|
func Test_multiple_level_anonymous_struct(t *testing.T) {
|
|
type Level1 struct {
|
|
Field1 string
|
|
}
|
|
type Level2 struct {
|
|
Level1
|
|
Field2 string
|
|
}
|
|
type Level3 struct {
|
|
Level2
|
|
Field3 string
|
|
}
|
|
should := require.New(t)
|
|
obj := Level3{Level2{Level1{"1"}, "2"}, "3"}
|
|
output, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field1":"1","Field2":"2","Field3":"3"}`, output)
|
|
}
|
|
|
|
func Test_multiple_level_anonymous_struct_with_ptr(t *testing.T) {
|
|
type Level1 struct {
|
|
Field1 string
|
|
Field2 string
|
|
Field4 string
|
|
}
|
|
type Level2 struct {
|
|
*Level1
|
|
Field2 string
|
|
Field3 string
|
|
}
|
|
type Level3 struct {
|
|
*Level2
|
|
Field3 string
|
|
}
|
|
should := require.New(t)
|
|
obj := Level3{&Level2{&Level1{"1", "", "4"}, "2", ""}, "3"}
|
|
output, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Contains(output, `"Field1":"1"`)
|
|
should.Contains(output, `"Field2":"2"`)
|
|
should.Contains(output, `"Field3":"3"`)
|
|
should.Contains(output, `"Field4":"4"`)
|
|
}
|
|
|
|
func Test_shadow_struct_field(t *testing.T) {
|
|
should := require.New(t)
|
|
type omit *struct{}
|
|
type CacheItem struct {
|
|
Key string `json:"key"`
|
|
MaxAge int `json:"cacheAge"`
|
|
}
|
|
output, err := MarshalToString(struct {
|
|
*CacheItem
|
|
|
|
// Omit bad keys
|
|
OmitMaxAge omit `json:"cacheAge,omitempty"`
|
|
|
|
// Add nice keys
|
|
MaxAge int `json:"max_age"`
|
|
}{
|
|
CacheItem: &CacheItem{
|
|
Key: "value",
|
|
MaxAge: 100,
|
|
},
|
|
MaxAge: 20,
|
|
})
|
|
should.Nil(err)
|
|
should.Contains(output, `"key":"value"`)
|
|
should.Contains(output, `"max_age":20`)
|
|
}
|
|
|
|
func Test_embedded_order(t *testing.T) {
|
|
type A struct {
|
|
Field2 string
|
|
}
|
|
|
|
type C struct {
|
|
Field5 string
|
|
}
|
|
|
|
type B struct {
|
|
Field4 string
|
|
C
|
|
Field6 string
|
|
}
|
|
|
|
type TestObject struct {
|
|
Field1 string
|
|
A
|
|
Field3 string
|
|
B
|
|
Field7 string
|
|
}
|
|
should := require.New(t)
|
|
s := TestObject{}
|
|
output, err := MarshalToString(s)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field1":"","Field2":"","Field3":"","Field4":"","Field5":"","Field6":"","Field7":""}`, output)
|
|
}
|
|
|
|
func Test_decode_nested(t *testing.T) {
|
|
type StructOfString struct {
|
|
Field1 string
|
|
Field2 string
|
|
}
|
|
iter := ParseString(ConfigDefault, `[{"field1": "hello"}, null, {"field2": "world"}]`)
|
|
slice := []*StructOfString{}
|
|
iter.ReadVal(&slice)
|
|
if len(slice) != 3 {
|
|
fmt.Println(iter.Error)
|
|
t.Fatal(len(slice))
|
|
}
|
|
if slice[0].Field1 != "hello" {
|
|
fmt.Println(iter.Error)
|
|
t.Fatal(slice[0])
|
|
}
|
|
if slice[1] != nil {
|
|
fmt.Println(iter.Error)
|
|
t.Fatal(slice[1])
|
|
}
|
|
if slice[2].Field2 != "world" {
|
|
fmt.Println(iter.Error)
|
|
t.Fatal(slice[2])
|
|
}
|
|
}
|
|
|
|
func Test_decode_field_with_escape(t *testing.T) {
|
|
should := require.New(t)
|
|
type TestObject struct {
|
|
Field1 string
|
|
}
|
|
var obj TestObject
|
|
should.Nil(ConfigCompatibleWithStandardLibrary.Unmarshal([]byte(`{"Field\"1":"hello"}`), &obj))
|
|
should.Equal("", obj.Field1)
|
|
should.Nil(ConfigCompatibleWithStandardLibrary.Unmarshal([]byte(`{"\u0046ield1":"hello"}`), &obj))
|
|
should.Equal("hello", obj.Field1)
|
|
}
|