mirror of
https://github.com/json-iterator/go.git
synced 2025-01-23 18:54:21 +02:00
1f58120d43
Skip creating encoders for unexported fields. They are not participating in JSON marshaling anyway. This allows using unexported fields of non-marshalable types in structs. As a side-effect of this change it's no longer possible to marshal unexported JSON fields by adding a custom type extenstion. It seems this is desired behavior since it matches standard library and jsoniter already disallows `json:"-"` fields from participating in custom extensions. Fixes #174.
342 lines
8.4 KiB
Go
342 lines
8.4 KiB
Go
package jsoniter
|
|
|
|
import (
|
|
"encoding/json"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
"unsafe"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func Test_customize_type_decoder(t *testing.T) {
|
|
RegisterTypeDecoderFunc("time.Time", func(ptr unsafe.Pointer, iter *Iterator) {
|
|
t, err := time.ParseInLocation("2006-01-02 15:04:05", iter.ReadString(), time.UTC)
|
|
if err != nil {
|
|
iter.Error = err
|
|
return
|
|
}
|
|
*((*time.Time)(ptr)) = t
|
|
})
|
|
defer ConfigDefault.(*frozenConfig).cleanDecoders()
|
|
val := time.Time{}
|
|
err := Unmarshal([]byte(`"2016-12-05 08:43:28"`), &val)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
year, month, day := val.Date()
|
|
if year != 2016 || month != 12 || day != 5 {
|
|
t.Fatal(val)
|
|
}
|
|
}
|
|
|
|
func Test_customize_type_encoder(t *testing.T) {
|
|
should := require.New(t)
|
|
RegisterTypeEncoderFunc("time.Time", func(ptr unsafe.Pointer, stream *Stream) {
|
|
t := *((*time.Time)(ptr))
|
|
stream.WriteString(t.UTC().Format("2006-01-02 15:04:05"))
|
|
}, nil)
|
|
defer ConfigDefault.(*frozenConfig).cleanEncoders()
|
|
val := time.Unix(0, 0)
|
|
str, err := MarshalToString(val)
|
|
should.Nil(err)
|
|
should.Equal(`"1970-01-01 00:00:00"`, str)
|
|
}
|
|
|
|
func Test_customize_byte_array_encoder(t *testing.T) {
|
|
ConfigDefault.(*frozenConfig).cleanEncoders()
|
|
should := require.New(t)
|
|
RegisterTypeEncoderFunc("[]uint8", func(ptr unsafe.Pointer, stream *Stream) {
|
|
t := *((*[]byte)(ptr))
|
|
stream.WriteString(string(t))
|
|
}, nil)
|
|
defer ConfigDefault.(*frozenConfig).cleanEncoders()
|
|
val := []byte("abc")
|
|
str, err := MarshalToString(val)
|
|
should.Nil(err)
|
|
should.Equal(`"abc"`, str)
|
|
}
|
|
|
|
func Test_customize_float_marshal(t *testing.T) {
|
|
should := require.New(t)
|
|
json := Config{MarshalFloatWith6Digits: true}.Froze()
|
|
str, err := json.MarshalToString(float32(1.23456789))
|
|
should.Nil(err)
|
|
should.Equal("1.234568", str)
|
|
}
|
|
|
|
type Tom struct {
|
|
field1 string
|
|
}
|
|
|
|
func Test_customize_field_decoder(t *testing.T) {
|
|
RegisterFieldDecoderFunc("jsoniter.Tom", "field1", func(ptr unsafe.Pointer, iter *Iterator) {
|
|
*((*string)(ptr)) = strconv.Itoa(iter.ReadInt())
|
|
})
|
|
defer ConfigDefault.(*frozenConfig).cleanDecoders()
|
|
tom := Tom{}
|
|
err := Unmarshal([]byte(`{"field1": 100}`), &tom)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
type TestObject1 struct {
|
|
Field1 string
|
|
}
|
|
|
|
type testExtension struct {
|
|
DummyExtension
|
|
}
|
|
|
|
func (extension *testExtension) UpdateStructDescriptor(structDescriptor *StructDescriptor) {
|
|
if structDescriptor.Type.String() != "jsoniter.TestObject1" {
|
|
return
|
|
}
|
|
binding := structDescriptor.GetField("Field1")
|
|
binding.Encoder = &funcEncoder{fun: func(ptr unsafe.Pointer, stream *Stream) {
|
|
str := *((*string)(ptr))
|
|
val, _ := strconv.Atoi(str)
|
|
stream.WriteInt(val)
|
|
}}
|
|
binding.Decoder = &funcDecoder{func(ptr unsafe.Pointer, iter *Iterator) {
|
|
*((*string)(ptr)) = strconv.Itoa(iter.ReadInt())
|
|
}}
|
|
binding.ToNames = []string{"field-1"}
|
|
binding.FromNames = []string{"field-1"}
|
|
}
|
|
|
|
func Test_customize_field_by_extension(t *testing.T) {
|
|
should := require.New(t)
|
|
RegisterExtension(&testExtension{})
|
|
obj := TestObject1{}
|
|
err := UnmarshalFromString(`{"field-1": 100}`, &obj)
|
|
should.Nil(err)
|
|
should.Equal("100", obj.Field1)
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"field-1":100}`, str)
|
|
}
|
|
|
|
type timeImplementedMarshaler time.Time
|
|
|
|
func (obj timeImplementedMarshaler) MarshalJSON() ([]byte, error) {
|
|
seconds := time.Time(obj).Unix()
|
|
return []byte(strconv.FormatInt(seconds, 10)), nil
|
|
}
|
|
|
|
func Test_marshaler(t *testing.T) {
|
|
type TestObject struct {
|
|
Field timeImplementedMarshaler
|
|
}
|
|
should := require.New(t)
|
|
val := timeImplementedMarshaler(time.Unix(123, 0))
|
|
obj := TestObject{val}
|
|
bytes, err := json.Marshal(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field":123}`, string(bytes))
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field":123}`, str)
|
|
}
|
|
|
|
func Test_marshaler_and_encoder(t *testing.T) {
|
|
type TestObject struct {
|
|
Field *timeImplementedMarshaler
|
|
}
|
|
ConfigDefault.(*frozenConfig).cleanEncoders()
|
|
should := require.New(t)
|
|
RegisterTypeEncoderFunc("jsoniter.timeImplementedMarshaler", func(ptr unsafe.Pointer, stream *Stream) {
|
|
stream.WriteString("hello from encoder")
|
|
}, nil)
|
|
val := timeImplementedMarshaler(time.Unix(123, 0))
|
|
obj := TestObject{&val}
|
|
bytes, err := json.Marshal(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field":123}`, string(bytes))
|
|
str, err := MarshalToString(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"Field":"hello from encoder"}`, str)
|
|
}
|
|
|
|
type ObjectImplementedUnmarshaler int
|
|
|
|
func (obj *ObjectImplementedUnmarshaler) UnmarshalJSON(s []byte) error {
|
|
val, _ := strconv.ParseInt(string(s[1:len(s)-1]), 10, 64)
|
|
*obj = ObjectImplementedUnmarshaler(val)
|
|
return nil
|
|
}
|
|
|
|
func Test_unmarshaler(t *testing.T) {
|
|
should := require.New(t)
|
|
var obj ObjectImplementedUnmarshaler
|
|
err := json.Unmarshal([]byte(` "100" `), &obj)
|
|
should.Nil(err)
|
|
should.Equal(100, int(obj))
|
|
iter := ParseString(ConfigDefault, ` "100" `)
|
|
iter.ReadVal(&obj)
|
|
should.Nil(err)
|
|
should.Equal(100, int(obj))
|
|
}
|
|
|
|
func Test_unmarshaler_and_decoder(t *testing.T) {
|
|
type TestObject struct {
|
|
Field *ObjectImplementedUnmarshaler
|
|
Field2 string
|
|
}
|
|
ConfigDefault.(*frozenConfig).cleanDecoders()
|
|
should := require.New(t)
|
|
RegisterTypeDecoderFunc("jsoniter.ObjectImplementedUnmarshaler", func(ptr unsafe.Pointer, iter *Iterator) {
|
|
*(*ObjectImplementedUnmarshaler)(ptr) = 10
|
|
iter.Skip()
|
|
})
|
|
obj := TestObject{}
|
|
val := ObjectImplementedUnmarshaler(0)
|
|
obj.Field = &val
|
|
err := json.Unmarshal([]byte(`{"Field":"100"}`), &obj)
|
|
should.Nil(err)
|
|
should.Equal(100, int(*obj.Field))
|
|
err = Unmarshal([]byte(`{"Field":"100"}`), &obj)
|
|
should.Nil(err)
|
|
should.Equal(10, int(*obj.Field))
|
|
}
|
|
|
|
type tmString string
|
|
type tmStruct struct {
|
|
String tmString
|
|
}
|
|
|
|
func (s tmStruct) MarshalJSON() ([]byte, error) {
|
|
var b []byte
|
|
b = append(b, '"')
|
|
b = append(b, s.String...)
|
|
b = append(b, '"')
|
|
return b, nil
|
|
}
|
|
|
|
func Test_marshaler_on_struct(t *testing.T) {
|
|
fixed := tmStruct{"hello"}
|
|
//json.Marshal(fixed)
|
|
Marshal(fixed)
|
|
}
|
|
|
|
type withChan struct {
|
|
F2 chan []byte
|
|
}
|
|
|
|
func (q withChan) MarshalJSON() ([]byte, error) {
|
|
return []byte(`""`), nil
|
|
}
|
|
|
|
func (q *withChan) UnmarshalJSON(value []byte) error {
|
|
return nil
|
|
}
|
|
|
|
func Test_marshal_json_with_chan(t *testing.T) {
|
|
type TestObject struct {
|
|
F1 withChan
|
|
}
|
|
should := require.New(t)
|
|
output, err := MarshalToString(TestObject{})
|
|
should.Nil(err)
|
|
should.Equal(`{"F1":""}`, output)
|
|
}
|
|
|
|
type withTime struct {
|
|
time.Time
|
|
}
|
|
|
|
func (t *withTime) UnmarshalJSON(b []byte) error {
|
|
return nil
|
|
}
|
|
func (t withTime) MarshalJSON() ([]byte, error) {
|
|
return []byte(`"fake"`), nil
|
|
}
|
|
|
|
func Test_marshal_json_with_time(t *testing.T) {
|
|
type S1 struct {
|
|
F1 withTime
|
|
F2 *withTime
|
|
}
|
|
type TestObject struct {
|
|
TF1 S1
|
|
}
|
|
should := require.New(t)
|
|
obj := TestObject{
|
|
S1{
|
|
F1: withTime{
|
|
time.Unix(0, 0),
|
|
},
|
|
F2: &withTime{
|
|
time.Unix(0, 0),
|
|
},
|
|
},
|
|
}
|
|
output, err := json.Marshal(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"TF1":{"F1":"fake","F2":"fake"}}`, string(output))
|
|
output, err = Marshal(obj)
|
|
should.Nil(err)
|
|
should.Equal(`{"TF1":{"F1":"fake","F2":"fake"}}`, string(output))
|
|
obj = TestObject{}
|
|
should.Nil(json.Unmarshal([]byte(`{"TF1":{"F1":"fake","F2":"fake"}}`), &obj))
|
|
should.NotNil(obj.TF1.F2)
|
|
obj = TestObject{}
|
|
should.Nil(Unmarshal([]byte(`{"TF1":{"F1":"fake","F2":"fake"}}`), &obj))
|
|
should.NotNil(obj.TF1.F2)
|
|
}
|
|
|
|
func Test_customize_tag_key(t *testing.T) {
|
|
|
|
type TestObject struct {
|
|
Field string `orm:"field"`
|
|
}
|
|
|
|
should := require.New(t)
|
|
json := Config{TagKey: "orm"}.Froze()
|
|
str, err := json.MarshalToString(TestObject{"hello"})
|
|
should.Nil(err)
|
|
should.Equal(`{"field":"hello"}`, str)
|
|
}
|
|
|
|
func Test_recursive_empty_interface_customization(t *testing.T) {
|
|
t.Skip()
|
|
var obj interface{}
|
|
RegisterTypeDecoderFunc("interface {}", func(ptr unsafe.Pointer, iter *Iterator) {
|
|
switch iter.WhatIsNext() {
|
|
case NumberValue:
|
|
*(*interface{})(ptr) = iter.ReadInt64()
|
|
default:
|
|
*(*interface{})(ptr) = iter.Read()
|
|
}
|
|
})
|
|
should := require.New(t)
|
|
Unmarshal([]byte("[100]"), &obj)
|
|
should.Equal([]interface{}{int64(100)}, obj)
|
|
}
|
|
|
|
type GeoLocation struct {
|
|
Id string `json:"id,omitempty" db:"id"`
|
|
}
|
|
|
|
func (p *GeoLocation) MarshalJSON() ([]byte, error) {
|
|
return []byte(`{}`), nil
|
|
}
|
|
|
|
func (p *GeoLocation) UnmarshalJSON(input []byte) error {
|
|
p.Id = "hello"
|
|
return nil
|
|
}
|
|
|
|
func Test_marshal_and_unmarshal_on_non_pointer(t *testing.T) {
|
|
should := require.New(t)
|
|
locations := []GeoLocation{{"000"}}
|
|
bytes, err := Marshal(locations)
|
|
should.Nil(err)
|
|
should.Equal("[{}]", string(bytes))
|
|
err = Unmarshal([]byte("[1]"), &locations)
|
|
should.Nil(err)
|
|
should.Equal("hello", locations[0].Id)
|
|
}
|