| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576 |
- package jsoniter
- import (
- "encoding/json"
- "fmt"
- "testing"
- "unsafe"
- "github.com/stretchr/testify/require"
- "reflect"
- )
- func Test_write_empty_interface_via_placeholder(t *testing.T) {
- fmt.Println(^uint(0) >> 1)
- should := require.New(t)
- m := map[uint32]interface{}{1: "hello"}
- inf := reflect.ValueOf(m).MapIndex(reflect.ValueOf(uint32(1))).Interface()
- encoder := &placeholderEncoder{
- cfg: ConfigFastest.(*frozenConfig),
- cacheKey: reflect.TypeOf(m).Elem(),
- }
- stream := ConfigFastest.BorrowStream(nil)
- encoderOfType(ConfigFastest.(*frozenConfig), "", reflect.TypeOf(m).Elem())
- encoder.EncodeInterface(inf, stream)
- should.Equal(`"hello"`, string(stream.Buffer()))
- }
- func Test_write_array_of_interface(t *testing.T) {
- should := require.New(t)
- array := []interface{}{"hello"}
- str, err := MarshalToString(array)
- should.Nil(err)
- should.Equal(`["hello"]`, str)
- }
- func Test_write_map_of_interface(t *testing.T) {
- should := require.New(t)
- val := map[string]interface{}{"hello": "world"}
- str, err := MarshalToString(val)
- should.Nil(err)
- should.Equal(`{"hello":"world"}`, str)
- }
- func Test_write_map_of_interface_in_struct(t *testing.T) {
- type TestObject struct {
- Field map[string]interface{}
- }
- should := require.New(t)
- val := TestObject{map[string]interface{}{"hello": "world"}}
- str, err := MarshalToString(val)
- should.Nil(err)
- should.Equal(`{"Field":{"hello":"world"}}`, str)
- }
- func Test_write_map_of_interface_in_struct_with_two_fields(t *testing.T) {
- type TestObject struct {
- Field map[string]interface{}
- Field2 string
- }
- should := require.New(t)
- val := TestObject{map[string]interface{}{"hello": "world"}, ""}
- str, err := MarshalToString(val)
- should.Nil(err)
- should.Contains(str, `"Field":{"hello":"world"}`)
- }
- type MyInterface interface {
- Hello() string
- }
- type MyString string
- func (ms MyString) Hello() string {
- return string(ms)
- }
- func Test_write_map_of_custom_interface(t *testing.T) {
- should := require.New(t)
- myStr := MyString("world")
- should.Equal("world", myStr.Hello())
- val := map[string]MyInterface{"hello": myStr}
- str, err := MarshalToString(val)
- should.Nil(err)
- should.Equal(`{"hello":"world"}`, str)
- }
- func Test_write_interface(t *testing.T) {
- should := require.New(t)
- var val interface{}
- val = "hello"
- str, err := MarshalToString(val)
- should.Nil(err)
- should.Equal(`"hello"`, str)
- }
- func Test_read_interface(t *testing.T) {
- should := require.New(t)
- var val interface{}
- err := UnmarshalFromString(`"hello"`, &val)
- should.Nil(err)
- should.Equal("hello", val)
- err = UnmarshalFromString(`1e1`, &val)
- should.Nil(err)
- should.Equal(float64(10), val)
- err = UnmarshalFromString(`1.0e1`, &val)
- should.Nil(err)
- should.Equal(float64(10), val)
- err = json.Unmarshal([]byte(`1.0e1`), &val)
- should.Nil(err)
- should.Equal(float64(10), val)
- }
- func Test_read_custom_interface(t *testing.T) {
- should := require.New(t)
- var val MyInterface
- RegisterTypeDecoderFunc("jsoniter.MyInterface", func(ptr unsafe.Pointer, iter *Iterator) {
- *((*MyInterface)(ptr)) = MyString(iter.ReadString())
- })
- err := UnmarshalFromString(`"hello"`, &val)
- should.Nil(err)
- should.Equal("hello", val.Hello())
- }
- func Test_decode_object_contain_empty_interface(t *testing.T) {
- type TestObject struct {
- Field interface{}
- }
- should := require.New(t)
- obj := TestObject{}
- obj.Field = 1024
- should.Nil(UnmarshalFromString(`{"Field": "hello"}`, &obj))
- should.Equal("hello", obj.Field)
- }
- func Test_decode_object_contain_non_empty_interface(t *testing.T) {
- type TestObject struct {
- Field MyInterface
- }
- should := require.New(t)
- obj := TestObject{}
- obj.Field = MyString("abc")
- should.Nil(UnmarshalFromString(`{"Field": "hello"}`, &obj))
- should.Equal(MyString("hello"), obj.Field)
- }
- func Test_encode_object_contain_empty_interface(t *testing.T) {
- type TestObject struct {
- Field interface{}
- }
- should := require.New(t)
- obj := TestObject{}
- obj.Field = 1024
- str, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{"Field":1024}`, str)
- }
- func Test_encode_object_contain_non_empty_interface(t *testing.T) {
- type TestObject struct {
- Field MyInterface
- }
- should := require.New(t)
- obj := TestObject{}
- obj.Field = MyString("hello")
- str, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{"Field":"hello"}`, str)
- }
- func Test_nil_non_empty_interface(t *testing.T) {
- ConfigDefault.(*frozenConfig).cleanEncoders()
- ConfigDefault.(*frozenConfig).cleanDecoders()
- type TestObject struct {
- Field []MyInterface
- }
- should := require.New(t)
- obj := TestObject{}
- b := []byte(`{"Field":["AAA"]}`)
- should.NotNil(json.Unmarshal(b, &obj))
- should.NotNil(Unmarshal(b, &obj))
- }
- func Test_read_large_number_as_interface(t *testing.T) {
- should := require.New(t)
- var val interface{}
- err := Config{UseNumber: true}.Froze().UnmarshalFromString(`123456789123456789123456789`, &val)
- should.Nil(err)
- output, err := MarshalToString(val)
- should.Nil(err)
- should.Equal(`123456789123456789123456789`, output)
- }
- func Test_nested_one_field_struct(t *testing.T) {
- should := require.New(t)
- type YetYetAnotherObject struct {
- Field string
- }
- type YetAnotherObject struct {
- Field *YetYetAnotherObject
- }
- type AnotherObject struct {
- Field *YetAnotherObject
- }
- type TestObject struct {
- Me *AnotherObject
- }
- obj := TestObject{&AnotherObject{&YetAnotherObject{&YetYetAnotherObject{"abc"}}}}
- str, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{"Me":{"Field":{"Field":{"Field":"abc"}}}}`, str)
- str, err = MarshalToString(&obj)
- should.Nil(err)
- should.Equal(`{"Me":{"Field":{"Field":{"Field":"abc"}}}}`, str)
- }
- func Test_struct_with_embedded_ptr_with_tag(t *testing.T) {
- type O1 struct {
- O1F string
- }
- type Option struct {
- O1 *O1
- }
- type T struct {
- Option `json:","`
- }
- var obj T
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{"O1":null}`, output)
- }
- func Test_struct_with_one_nil(t *testing.T) {
- type TestObject struct {
- F *float64
- }
- var obj TestObject
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{"F":null}`, output)
- }
- func Test_struct_with_one_nil_embedded(t *testing.T) {
- type Parent struct {
- Field1 string
- Field2 string
- }
- type TestObject struct {
- *Parent
- }
- obj := TestObject{}
- should := require.New(t)
- bytes, err := json.Marshal(obj)
- should.Nil(err)
- should.Equal("{}", string(bytes))
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`{}`, output)
- }
- func Test_struct_with_not_nil_embedded(t *testing.T) {
- type Parent struct {
- Field0 string
- Field1 []string
- Field2 map[string]interface{}
- }
- type TestObject struct {
- *Parent
- }
- should := require.New(t)
- var obj TestObject
- err := UnmarshalFromString(`{"Field0":"1","Field1":null,"Field2":{"K":"V"}}`, &obj)
- should.Nil(err)
- should.Nil(obj.Field1)
- should.Equal(map[string]interface{}{"K": "V"}, obj.Field2)
- should.Equal("1", obj.Field0)
- }
- func Test_array_with_one_nil_ptr(t *testing.T) {
- obj := [1]*float64{nil}
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`[null]`, output)
- }
- func Test_array_with_one_not_nil_ptr(t *testing.T) {
- two := float64(2)
- obj := [1]*float64{&two}
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`[2]`, output)
- }
- func Test_embedded_array_with_one_nil(t *testing.T) {
- type TestObject struct {
- Field1 int
- Field2 [1]*float64
- }
- var obj TestObject
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Contains(output, `"Field2":[null]`)
- }
- func Test_array_with_nothing(t *testing.T) {
- var obj [2]*float64
- should := require.New(t)
- output, err := MarshalToString(obj)
- should.Nil(err)
- should.Equal(`[null,null]`, output)
- }
- func Test_unmarshal_ptr_to_interface(t *testing.T) {
- type TestData struct {
- Name string `json:"name"`
- }
- should := require.New(t)
- var obj interface{} = &TestData{}
- err := json.Unmarshal([]byte(`{"name":"value"}`), &obj)
- should.Nil(err)
- should.Equal("&{value}", fmt.Sprintf("%v", obj))
- obj = interface{}(&TestData{})
- err = Unmarshal([]byte(`{"name":"value"}`), &obj)
- should.Nil(err)
- should.Equal("&{value}", fmt.Sprintf("%v", obj))
- }
- func Test_nil_out_null_interface(t *testing.T) {
- type TestData struct {
- Field interface{} `json:"field"`
- }
- should := require.New(t)
- var boolVar bool
- obj := TestData{
- Field: &boolVar,
- }
- data1 := []byte(`{"field": true}`)
- err := Unmarshal(data1, &obj)
- should.NoError(err)
- should.Equal(true, *(obj.Field.(*bool)))
- data2 := []byte(`{"field": null}`)
- err = Unmarshal(data2, &obj)
- should.NoError(err)
- should.Equal(nil, obj.Field)
- // Checking stdlib behavior matches.
- obj2 := TestData{
- Field: &boolVar,
- }
- err = json.Unmarshal(data1, &obj2)
- should.NoError(err)
- should.Equal(true, *(obj2.Field.(*bool)))
- err = json.Unmarshal(data2, &obj2)
- should.NoError(err)
- should.Equal(nil, obj2.Field)
- }
- func Test_omitempty_nil_interface(t *testing.T) {
- type TestData struct {
- Field interface{} `json:"field,omitempty"`
- }
- should := require.New(t)
- obj := TestData{
- Field: nil,
- }
- js, err := json.Marshal(obj)
- should.NoError(err)
- should.Equal("{}", string(js))
- str, err := MarshalToString(obj)
- should.NoError(err)
- should.Equal(string(js), str)
- }
- func Test_omitempty_nil_nonempty_interface(t *testing.T) {
- type TestData struct {
- Field MyInterface `json:"field,omitempty"`
- }
- should := require.New(t)
- obj := TestData{
- Field: nil,
- }
- js, err := json.Marshal(obj)
- should.NoError(err)
- should.Equal("{}", string(js))
- str, err := MarshalToString(obj)
- should.NoError(err)
- should.Equal(string(js), str)
- obj.Field = MyString("hello")
- err = UnmarshalFromString(`{"field":null}`, &obj)
- should.NoError(err)
- should.Nil(obj.Field)
- }
- func Test_marshal_nil_marshaler_interface(t *testing.T) {
- type TestData struct {
- Field json.Marshaler `json:"field"`
- }
- should := require.New(t)
- obj := TestData{
- Field: nil,
- }
- js, err := json.Marshal(obj)
- should.NoError(err)
- should.Equal(`{"field":null}`, string(js))
- str, err := MarshalToString(obj)
- should.NoError(err)
- should.Equal(string(js), str)
- }
- func Test_marshal_nil_nonempty_interface(t *testing.T) {
- type TestData struct {
- Field MyInterface `json:"field"`
- }
- should := require.New(t)
- obj := TestData{
- Field: nil,
- }
- js, err := json.Marshal(obj)
- should.NoError(err)
- should.Equal(`{"field":null}`, string(js))
- str, err := MarshalToString(obj)
- should.NoError(err)
- should.Equal(string(js), str)
- obj.Field = MyString("hello")
- err = Unmarshal(js, &obj)
- should.NoError(err)
- should.Equal(nil, obj.Field)
- }
- func Test_overwrite_interface_ptr_value_with_nil(t *testing.T) {
- type Wrapper struct {
- Payload interface{} `json:"payload,omitempty"`
- }
- type Payload struct {
- Value int `json:"val,omitempty"`
- }
- should := require.New(t)
- payload := &Payload{}
- wrapper := &Wrapper{
- Payload: &payload,
- }
- err := json.Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(&payload, wrapper.Payload)
- should.Equal(42, (*(wrapper.Payload.(**Payload))).Value)
- err = json.Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(&payload, wrapper.Payload)
- should.Equal((*Payload)(nil), payload)
- payload = &Payload{}
- wrapper = &Wrapper{
- Payload: &payload,
- }
- err = Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(&payload, wrapper.Payload)
- should.Equal(42, (*(wrapper.Payload.(**Payload))).Value)
- err = Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(&payload, wrapper.Payload)
- should.Equal((*Payload)(nil), payload)
- }
- func Test_overwrite_interface_value_with_nil(t *testing.T) {
- type Wrapper struct {
- Payload interface{} `json:"payload,omitempty"`
- }
- type Payload struct {
- Value int `json:"val,omitempty"`
- }
- should := require.New(t)
- payload := &Payload{}
- wrapper := &Wrapper{
- Payload: payload,
- }
- err := json.Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(42, (*(wrapper.Payload.(*Payload))).Value)
- err = json.Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(nil, wrapper.Payload)
- should.Equal(42, payload.Value)
- payload = &Payload{}
- wrapper = &Wrapper{
- Payload: payload,
- }
- err = Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(42, (*(wrapper.Payload.(*Payload))).Value)
- err = Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Equal(nil, err)
- should.Equal(nil, wrapper.Payload)
- should.Equal(42, payload.Value)
- }
- func Test_unmarshal_into_nil(t *testing.T) {
- type Payload struct {
- Value int `json:"val,omitempty"`
- }
- type Wrapper struct {
- Payload interface{} `json:"payload,omitempty"`
- }
- should := require.New(t)
- var payload *Payload
- wrapper := &Wrapper{
- Payload: payload,
- }
- err := json.Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Nil(err)
- should.NotNil(wrapper.Payload)
- should.Nil(payload)
- err = json.Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Nil(err)
- should.Nil(wrapper.Payload)
- should.Nil(payload)
- payload = nil
- wrapper = &Wrapper{
- Payload: payload,
- }
- err = Unmarshal([]byte(`{"payload": {"val": 42}}`), &wrapper)
- should.Nil(err)
- should.NotNil(wrapper.Payload)
- should.Nil(payload)
- err = Unmarshal([]byte(`{"payload": null}`), &wrapper)
- should.Nil(err)
- should.Nil(wrapper.Payload)
- should.Nil(payload)
- }
|