1
0
mirror of https://github.com/json-iterator/go.git synced 2025-06-30 23:23:49 +02:00
Files
json-iterator/extension_tests/extension_test.go

194 lines
4.5 KiB
Go
Raw Normal View History

2018-02-13 23:49:40 +08:00
package test
import (
"github.com/json-iterator/go"
2018-02-28 17:09:30 +08:00
"github.com/modern-go/reflect2"
"github.com/stretchr/testify/require"
"reflect"
2018-02-24 22:04:41 +08:00
"strconv"
"testing"
"unsafe"
2018-02-13 23:49:40 +08:00
)
type TestObject1 struct {
Field1 string
}
type testExtension struct {
jsoniter.DummyExtension
}
func (extension *testExtension) UpdateStructDescriptor(structDescriptor *jsoniter.StructDescriptor) {
if structDescriptor.Type.String() != "test.TestObject1" {
return
}
binding := structDescriptor.GetField("Field1")
binding.Encoder = &funcEncoder{fun: func(ptr unsafe.Pointer, stream *jsoniter.Stream) {
str := *((*string)(ptr))
val, _ := strconv.Atoi(str)
stream.WriteInt(val)
}}
binding.Decoder = &funcDecoder{func(ptr unsafe.Pointer, iter *jsoniter.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)
cfg := jsoniter.Config{}.Froze()
cfg.RegisterExtension(&testExtension{})
obj := TestObject1{}
err := cfg.UnmarshalFromString(`{"field-1": 100}`, &obj)
should.Nil(err)
should.Equal("100", obj.Field1)
str, err := cfg.MarshalToString(obj)
should.Nil(err)
should.Equal(`{"field-1":100}`, str)
}
func Test_customize_map_key_encoder(t *testing.T) {
should := require.New(t)
cfg := jsoniter.Config{}.Froze()
cfg.RegisterExtension(&testMapKeyExtension{})
m := map[int]int{1: 2}
output, err := cfg.MarshalToString(m)
should.NoError(err)
should.Equal(`{"2":2}`, output)
m = map[int]int{}
should.NoError(cfg.UnmarshalFromString(output, &m))
should.Equal(map[int]int{1: 2}, m)
}
// Test using custom encoder with sorted map keys.
// Keys should be numerically sorted AFTER the custom key encoder runs.
func Test_customize_map_key_encoder_with_sorted_keys(t *testing.T) {
should := require.New(t)
cfg := jsoniter.Config{
SortMapKeys: true,
}.Froze()
cfg.RegisterExtension(&testMapKeyExtension{})
m := map[int]int{
3: 3,
1: 9,
}
output, err := cfg.MarshalToString(m)
should.NoError(err)
should.Equal(`{"2":9,"4":3}`, output)
m2 := map[int]int{}
should.NoError(cfg.UnmarshalFromString(output, &m2))
should.Equal(map[int]int{
1: 9,
3: 3,
}, m2)
}
func Test_customize_map_key_sorter(t *testing.T) {
should := require.New(t)
cfg := jsoniter.Config{
SortMapKeys: true,
}.Froze()
cfg.RegisterExtension(&testMapKeySorterExtension{
sorter: &testKeySorter{},
})
m := map[string]int{
"a": 1,
"foo": 2,
"b": 3,
}
output, err := cfg.MarshalToString(m)
should.NoError(err)
should.Equal(`{"foo":2,"a":1,"b":3}`, output)
m = map[string]int{}
should.NoError(cfg.UnmarshalFromString(output, &m))
should.Equal(map[string]int{
"foo": 2,
"a": 1,
"b": 3,
}, m)
}
type testKeySorter struct {
}
func (sorter *testKeySorter) Sort(keyA string, keyB string) bool {
// Prioritize "foo" over other keys, otherwise alpha-sort
if keyA == "foo" {
return true
} else if keyB == "foo" {
return false
} else {
return keyA < keyB
}
}
type testMapKeySorterExtension struct {
jsoniter.DummyExtension
sorter jsoniter.MapKeySorter
}
func (extension *testMapKeySorterExtension) CreateMapKeySorter() jsoniter.MapKeySorter {
return extension.sorter
}
type testMapKeyExtension struct {
jsoniter.DummyExtension
}
func (extension *testMapKeyExtension) CreateMapKeyEncoder(typ reflect2.Type) jsoniter.ValEncoder {
if typ.Kind() == reflect.Int {
return &funcEncoder{
fun: func(ptr unsafe.Pointer, stream *jsoniter.Stream) {
stream.WriteRaw(`"`)
stream.WriteInt(*(*int)(ptr) + 1)
stream.WriteRaw(`"`)
},
}
}
return nil
}
func (extension *testMapKeyExtension) CreateMapKeyDecoder(typ reflect2.Type) jsoniter.ValDecoder {
if typ.Kind() == reflect.Int {
return &funcDecoder{
fun: func(ptr unsafe.Pointer, iter *jsoniter.Iterator) {
i, err := strconv.Atoi(iter.ReadString())
if err != nil {
iter.ReportError("read map key", err.Error())
return
}
i--
*(*int)(ptr) = i
},
}
}
return nil
}
2018-02-13 23:49:40 +08:00
type funcDecoder struct {
fun jsoniter.DecoderFunc
}
func (decoder *funcDecoder) Decode(ptr unsafe.Pointer, iter *jsoniter.Iterator) {
decoder.fun(ptr, iter)
}
type funcEncoder struct {
fun jsoniter.EncoderFunc
isEmptyFunc func(ptr unsafe.Pointer) bool
}
func (encoder *funcEncoder) Encode(ptr unsafe.Pointer, stream *jsoniter.Stream) {
encoder.fun(ptr, stream)
}
func (encoder *funcEncoder) IsEmpty(ptr unsafe.Pointer) bool {
if encoder.isEmptyFunc == nil {
return false
}
return encoder.isEmptyFunc(ptr)
}