| 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_nested_field_omit_empty(t *testing.T) { |
| should := require.New(t) |
| type S1 struct { |
| F1 string `json:",omitempty"` |
| } |
| |
| type S2 struct { |
| *S1 |
| F2 string `json:",omitempty"` |
| } |
| s1 := &S1{ |
| //F1: "abc", |
| } |
| s2 := &S2{ |
| S1: s1, |
| F2: "123", |
| } |
| str, err := MarshalToString(s2) |
| should.Nil(err) |
| should.Equal(`{"F2":"123"}`, 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) |
| } |