2016-12-01 16:11:07 +08:00
|
|
|
package jsoniter
|
|
|
|
|
|
|
|
import (
|
2017-06-06 23:27:00 +08:00
|
|
|
"bytes"
|
2017-06-19 23:43:53 +08:00
|
|
|
"fmt"
|
2017-06-06 23:27:00 +08:00
|
|
|
"testing"
|
2017-07-05 11:40:20 +08:00
|
|
|
|
2017-07-07 09:13:25 +08:00
|
|
|
"github.com/stretchr/testify/require"
|
2016-12-01 16:11:07 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
func Test_empty_object(t *testing.T) {
|
2017-01-20 12:56:49 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 10:21:37 +08:00
|
|
|
iter := ParseString(ConfigDefault, `{}`)
|
2016-12-01 16:11:07 +08:00
|
|
|
field := iter.ReadObject()
|
2017-01-20 12:56:49 +08:00
|
|
|
should.Equal("", field)
|
2017-06-17 10:21:37 +08:00
|
|
|
iter = ParseString(ConfigDefault, `{}`)
|
2017-01-20 12:56:49 +08:00
|
|
|
iter.ReadObjectCB(func(iter *Iterator, field string) bool {
|
|
|
|
should.FailNow("should not call")
|
|
|
|
return true
|
|
|
|
})
|
2016-12-01 16:11:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func Test_one_field(t *testing.T) {
|
2017-01-20 12:56:49 +08:00
|
|
|
should := require.New(t)
|
2017-07-09 14:48:34 +08:00
|
|
|
iter := ParseString(ConfigDefault, `{"a": "stream"}`)
|
2016-12-01 16:11:07 +08:00
|
|
|
field := iter.ReadObject()
|
2017-01-20 12:56:49 +08:00
|
|
|
should.Equal("a", field)
|
2016-12-01 16:11:07 +08:00
|
|
|
value := iter.ReadString()
|
2017-07-09 14:48:34 +08:00
|
|
|
should.Equal("stream", value)
|
2016-12-01 16:11:07 +08:00
|
|
|
field = iter.ReadObject()
|
2017-01-20 12:56:49 +08:00
|
|
|
should.Equal("", field)
|
2017-07-09 14:48:34 +08:00
|
|
|
iter = ParseString(ConfigDefault, `{"a": "stream"}`)
|
2017-01-20 12:56:49 +08:00
|
|
|
should.True(iter.ReadObjectCB(func(iter *Iterator, field string) bool {
|
|
|
|
should.Equal("a", field)
|
2017-07-18 09:45:25 +08:00
|
|
|
iter.Skip()
|
2017-01-20 12:56:49 +08:00
|
|
|
return true
|
|
|
|
}))
|
2017-07-06 11:44:39 +08:00
|
|
|
|
2016-12-01 16:11:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func Test_two_field(t *testing.T) {
|
2017-01-24 00:23:07 +08:00
|
|
|
should := require.New(t)
|
2017-07-09 14:48:34 +08:00
|
|
|
iter := ParseString(ConfigDefault, `{ "a": "stream" , "c": "d" }`)
|
2016-12-01 16:11:07 +08:00
|
|
|
field := iter.ReadObject()
|
2017-01-24 00:23:07 +08:00
|
|
|
should.Equal("a", field)
|
2016-12-01 16:11:07 +08:00
|
|
|
value := iter.ReadString()
|
2017-07-09 14:48:34 +08:00
|
|
|
should.Equal("stream", value)
|
2016-12-01 16:11:07 +08:00
|
|
|
field = iter.ReadObject()
|
2017-01-24 00:23:07 +08:00
|
|
|
should.Equal("c", field)
|
2016-12-01 16:11:07 +08:00
|
|
|
value = iter.ReadString()
|
2017-01-24 00:23:07 +08:00
|
|
|
should.Equal("d", value)
|
2016-12-01 16:11:07 +08:00
|
|
|
field = iter.ReadObject()
|
2017-01-24 00:23:07 +08:00
|
|
|
should.Equal("", field)
|
2017-06-17 10:21:37 +08:00
|
|
|
iter = ParseString(ConfigDefault, `{"field1": "1", "field2": 2}`)
|
2016-12-01 16:11:07 +08:00
|
|
|
for field := iter.ReadObject(); field != ""; field = iter.ReadObject() {
|
|
|
|
switch field {
|
|
|
|
case "field1":
|
|
|
|
iter.ReadString()
|
|
|
|
case "field2":
|
|
|
|
iter.ReadInt64()
|
|
|
|
default:
|
2017-06-20 15:11:01 +08:00
|
|
|
iter.ReportError("bind object", "unexpected field")
|
2016-12-01 16:11:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_object_wrapper_any_get_all(t *testing.T) {
|
2017-01-24 00:23:07 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
type TestObject struct {
|
|
|
|
Field1 []int
|
|
|
|
Field2 []int
|
|
|
|
}
|
|
|
|
any := Wrap(TestObject{[]int{1, 2}, []int{3, 4}})
|
|
|
|
should.Contains(any.Get('*', 0).ToString(), `"Field2":3`)
|
2017-07-06 11:44:39 +08:00
|
|
|
should.Contains(any.Keys(), "Field1")
|
|
|
|
should.Contains(any.Keys(), "Field2")
|
|
|
|
should.NotContains(any.Keys(), "Field3")
|
|
|
|
|
|
|
|
//should.Contains(any.GetObject()["Field1"].GetArray()[0], 1)
|
2017-01-24 00:23:07 +08:00
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_write_object(t *testing.T) {
|
2017-01-24 00:23:07 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
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)
|
2017-06-30 14:01:50 +08:00
|
|
|
should.Equal("{\n \"hello\": 1,\n \"world\": 2\n}", buf.String())
|
2017-06-17 21:32:48 +08:00
|
|
|
}
|
2017-01-24 00:23:07 +08:00
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_write_val_zero_field_struct(t *testing.T) {
|
2017-01-24 23:13:58 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
type TestObject struct {
|
|
|
|
}
|
|
|
|
obj := TestObject{}
|
|
|
|
str, err := MarshalToString(obj)
|
2017-01-24 23:13:58 +08:00
|
|
|
should.Nil(err)
|
2017-06-17 21:32:48 +08:00
|
|
|
should.Equal(`{}`, str)
|
2017-01-24 23:13:58 +08:00
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_write_val_one_field_struct(t *testing.T) {
|
2017-01-31 20:26:35 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
type TestObject struct {
|
|
|
|
Field1 string `json:"field-1"`
|
|
|
|
}
|
|
|
|
obj := TestObject{"hello"}
|
|
|
|
str, err := MarshalToString(obj)
|
2017-01-31 20:26:35 +08:00
|
|
|
should.Nil(err)
|
2017-06-17 21:32:48 +08:00
|
|
|
should.Equal(`{"field-1":"hello"}`, str)
|
2017-01-31 20:26:35 +08:00
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_mixed(t *testing.T) {
|
2017-01-26 16:24:01 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
type AA struct {
|
|
|
|
ID int `json:"id"`
|
|
|
|
Payload map[string]interface{} `json:"payload"`
|
2017-07-09 11:10:44 +08:00
|
|
|
buf *bytes.Buffer
|
2017-06-17 21:32:48 +08:00
|
|
|
}
|
|
|
|
aa := AA{}
|
|
|
|
err := UnmarshalFromString(` {"id":1, "payload":{"account":"123","password":"456"}}`, &aa)
|
2017-01-26 16:24:01 +08:00
|
|
|
should.Nil(err)
|
2017-06-17 21:32:48 +08:00
|
|
|
should.Equal(1, aa.ID)
|
|
|
|
should.Equal("123", aa.Payload["account"])
|
2017-01-26 16:24:01 +08:00
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_omit_empty(t *testing.T) {
|
2017-01-25 22:43:57 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
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)
|
2017-01-26 00:25:17 +08:00
|
|
|
should.Nil(err)
|
2017-06-17 21:32:48 +08:00
|
|
|
should.Equal(`{"field-2":"hello"}`, str)
|
2017-01-25 22:43:57 +08:00
|
|
|
}
|
|
|
|
|
2017-07-08 15:38:27 +02:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2018-01-07 13:57:46 +08:00
|
|
|
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{
|
2018-02-05 22:45:04 +08:00
|
|
|
//F1: "abc",
|
2018-01-07 13:57:46 +08:00
|
|
|
}
|
|
|
|
s2 := &S2{
|
|
|
|
S1: s1,
|
|
|
|
F2: "123",
|
|
|
|
}
|
|
|
|
str, err := MarshalToString(s2)
|
|
|
|
should.Nil(err)
|
|
|
|
should.Equal(`{"F2":"123"}`, str)
|
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_recursive_struct(t *testing.T) {
|
2017-01-28 22:45:03 +08:00
|
|
|
should := require.New(t)
|
|
|
|
type TestObject struct {
|
|
|
|
Field1 string
|
2017-06-17 21:32:48 +08:00
|
|
|
Me *TestObject
|
2017-01-28 22:45:03 +08:00
|
|
|
}
|
2017-06-17 21:32:48 +08:00
|
|
|
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)
|
2017-01-28 22:45:03 +08:00
|
|
|
}
|
|
|
|
|
2017-06-19 23:02:57 +08:00
|
|
|
func Test_encode_anonymous_struct(t *testing.T) {
|
2017-01-07 22:08:45 +08:00
|
|
|
should := require.New(t)
|
2017-06-17 21:32:48 +08:00
|
|
|
type TestObject struct {
|
|
|
|
Field string
|
|
|
|
}
|
|
|
|
str, err := MarshalToString(struct {
|
|
|
|
TestObject
|
|
|
|
Field int
|
|
|
|
}{
|
|
|
|
Field: 100,
|
|
|
|
})
|
|
|
|
should.Nil(err)
|
|
|
|
should.Equal(`{"Field":100}`, str)
|
2017-01-07 22:08:45 +08:00
|
|
|
}
|
|
|
|
|
2017-06-19 23:02:57 +08:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2017-06-20 13:33:40 +08:00
|
|
|
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)
|
2017-06-21 00:26:18 +08:00
|
|
|
obj := Level3{Level2{Level1{"1"}, "2"}, "3"}
|
|
|
|
output, err := MarshalToString(obj)
|
2017-06-20 13:33:40 +08:00
|
|
|
should.Nil(err)
|
2017-06-23 08:21:02 +08:00
|
|
|
should.Equal(`{"Field1":"1","Field2":"2","Field3":"3"}`, output)
|
2017-06-20 13:33:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
2017-06-21 00:26:18 +08:00
|
|
|
obj := Level3{&Level2{&Level1{"1", "", "4"}, "2", ""}, "3"}
|
|
|
|
output, err := MarshalToString(obj)
|
2017-06-20 13:33:40 +08:00
|
|
|
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
|
2017-06-21 00:26:18 +08:00
|
|
|
MaxAge int `json:"max_age"`
|
2017-06-20 13:33:40 +08:00
|
|
|
}{
|
|
|
|
CacheItem: &CacheItem{
|
|
|
|
Key: "value",
|
|
|
|
MaxAge: 100,
|
|
|
|
},
|
|
|
|
MaxAge: 20,
|
|
|
|
})
|
|
|
|
should.Nil(err)
|
|
|
|
should.Contains(output, `"key":"value"`)
|
|
|
|
should.Contains(output, `"max_age":20`)
|
|
|
|
}
|
|
|
|
|
2017-07-05 11:40:20 +08:00
|
|
|
func Test_embedded_order(t *testing.T) {
|
2017-06-30 14:01:50 +08:00
|
|
|
type A struct {
|
|
|
|
Field2 string
|
|
|
|
}
|
|
|
|
|
|
|
|
type C struct {
|
|
|
|
Field5 string
|
|
|
|
}
|
|
|
|
|
|
|
|
type B struct {
|
|
|
|
Field4 string
|
|
|
|
C
|
|
|
|
Field6 string
|
2017-06-23 07:45:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-30 14:01:50 +08:00
|
|
|
type TestObject struct {
|
|
|
|
Field1 string
|
|
|
|
A
|
|
|
|
Field3 string
|
|
|
|
B
|
|
|
|
Field7 string
|
2017-06-23 07:45:18 +08:00
|
|
|
}
|
|
|
|
should := require.New(t)
|
2017-06-30 14:01:50 +08:00
|
|
|
s := TestObject{}
|
2017-06-23 07:45:18 +08:00
|
|
|
output, err := MarshalToString(s)
|
|
|
|
should.Nil(err)
|
2017-06-30 14:01:50 +08:00
|
|
|
should.Equal(`{"Field1":"","Field2":"","Field3":"","Field4":"","Field5":"","Field6":"","Field7":""}`, output)
|
2017-06-23 07:45:18 +08:00
|
|
|
}
|
|
|
|
|
2017-06-17 21:32:48 +08:00
|
|
|
func Test_decode_nested(t *testing.T) {
|
|
|
|
type StructOfString struct {
|
2017-02-12 22:49:45 +08:00
|
|
|
Field1 string
|
2017-06-17 21:32:48 +08:00
|
|
|
Field2 string
|
2017-02-12 22:49:45 +08:00
|
|
|
}
|
2017-06-17 21:32:48 +08:00
|
|
|
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))
|
2016-12-04 11:06:38 +08:00
|
|
|
}
|
2017-06-17 21:32:48 +08:00
|
|
|
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])
|
2017-02-12 22:49:45 +08:00
|
|
|
}
|
2017-06-17 21:32:48 +08:00
|
|
|
if slice[2].Field2 != "world" {
|
|
|
|
fmt.Println(iter.Error)
|
|
|
|
t.Fatal(slice[2])
|
2016-12-01 16:11:07 +08:00
|
|
|
}
|
|
|
|
}
|
2017-10-31 22:38:41 +08:00
|
|
|
|
|
|
|
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)
|
|
|
|
}
|