|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397 |
- package serder
-
- import (
- "fmt"
- "reflect"
- "testing"
-
- . "github.com/smartystreets/goconvey/convey"
- myreflect "gitlink.org.cn/cloudream/common/utils/reflect"
- )
-
- type FromAnyString struct {
- Str string
- }
-
- func (a *FromAnyString) FromAny(val any) (bool, error) {
- if str, ok := val.(string); ok {
- a.Str = "@" + str
- return true, nil
- }
-
- return false, nil
- }
-
- type ToAnyString struct {
- Str string
- }
-
- func (a *ToAnyString) ToAny(typ reflect.Type) (val any, ok bool, err error) {
- if typ == myreflect.TypeOf[map[string]any]() {
- return map[string]any{
- "str": "@" + a.Str,
- }, true, nil
- }
-
- return nil, false, nil
- }
-
- type FromAnySt struct {
- Value string
- }
-
- func (a *FromAnySt) FromAny(val any) (bool, error) {
- if st, ok := val.(ToAnySt); ok {
- a.Value = "From:" + st.Value
- return true, nil
- }
-
- return false, nil
- }
-
- type ToAnySt struct {
- Value string
- }
-
- func (a *ToAnySt) ToAny(typ reflect.Type) (val any, ok bool, err error) {
- if typ == myreflect.TypeOf[FromAnySt]() {
- return FromAnySt{
- Value: "To:" + a.Value,
- }, true, nil
- }
-
- return nil, false, nil
- }
-
- type DirToAnySt struct {
- Value string
- }
-
- func (a DirToAnySt) ToAny(typ reflect.Type) (val any, ok bool, err error) {
- if typ == myreflect.TypeOf[FromAnySt]() {
- return FromAnySt{
- Value: "DirTo:" + a.Value,
- }, true, nil
- }
-
- return nil, false, nil
- }
-
- func Test_AnyToAny(t *testing.T) {
- Convey("包含用字符串保存的int数据", t, func() {
- type Struct struct {
- A string `json:"a"`
- B int `json:"b"`
- C int64 `json:"c,string"`
- }
-
- mp := map[string]any{
- "a": "a",
- "b": 1,
- "c": "1234",
- }
-
- var st Struct
-
- err := AnyToAny(mp, &st)
- So(err, ShouldBeNil)
-
- So(st.A, ShouldEqual, "a")
- So(st.B, ShouldEqual, 1)
- So(st.C, ShouldEqual, 1234)
- })
-
- Convey("只有FromAny", t, func() {
- type Struct struct {
- Special FromAnyString `json:"str"`
- }
-
- mp := map[string]any{
- "str": "test",
- }
-
- var ret Struct
- err := AnyToAny(mp, &ret)
- So(err, ShouldBeNil)
-
- So(ret.Special.Str, ShouldEqual, "@test")
- })
-
- Convey("字段类型直接实现了FromAny", t, func() {
- type Struct struct {
- Special *FromAnyString `json:"str"`
- }
-
- mp := map[string]any{
- "str": "test",
- }
-
- var ret Struct
- err := AnyToAny(mp, &ret)
- So(err, ShouldBeNil)
-
- So(ret.Special.Str, ShouldEqual, "@test")
- })
-
- Convey("只有ToAny", t, func() {
- st := struct {
- Special ToAnyString `json:"str"`
- }{
- Special: ToAnyString{
- Str: "test",
- },
- }
-
- ret := map[string]any{}
-
- err := AnyToAny(st, &ret)
- So(err, ShouldBeNil)
-
- So(ret["str"].(map[string]any)["str"], ShouldEqual, "@test")
- })
-
- Convey("优先使用ToAny", t, func() {
- st1 := ToAnySt{
- Value: "test",
- }
-
- st2 := FromAnySt{}
-
- err := AnyToAny(st1, &st2)
- So(err, ShouldBeNil)
-
- So(st2.Value, ShouldEqual, "To:test")
- })
-
- Convey("使用Convertor", t, func() {
- type Struct1 struct {
- Value string
- }
-
- type Struct2 struct {
- Value string
- }
-
- st1 := Struct1{
- Value: "test",
- }
-
- st2 := Struct2{}
-
- err := AnyToAny(st1, &st2, AnyToAnyOption{
- Converters: []Converter{func(srcType reflect.Type, dstType reflect.Type, data interface{}) (interface{}, error) {
- if srcType == myreflect.TypeOf[Struct1]() && dstType == myreflect.TypeOf[Struct2]() {
- s1 := data.(Struct1)
- return Struct2{
- Value: "@" + s1.Value,
- }, nil
- }
-
- return nil, fmt.Errorf("should not arrive here!")
- }},
- })
- So(err, ShouldBeNil)
-
- So(st2.Value, ShouldEqual, "@test")
- })
- }
-
- func Test_TypedMapToObject(t *testing.T) {
- type Struct struct {
- A string `json:"a"`
- B int `json:"b"`
- C int64 `json:"c,string"`
- }
-
- nameResovler := NewTypeNameResolver(true)
- nameResovler.Register(myreflect.TypeOf[Struct]())
-
- Convey("结构体", t, func() {
- st := Struct{
- A: "a",
- B: 1,
- C: 2,
- }
-
- mp, err := ObjectToTypedMap(st, TypedSerderOption{
- TypeResolver: &nameResovler,
- TypeFieldName: "@type",
- })
-
- So(err, ShouldBeNil)
-
- st2Ptr, err := TypedMapToObject(mp, TypedSerderOption{
- TypeResolver: &nameResovler,
- TypeFieldName: "@type",
- })
- So(err, ShouldBeNil)
-
- st2, ok := st2Ptr.(Struct)
- So(ok, ShouldBeTrue)
- So(st2, ShouldHaveSameTypeAs, st)
- So(st2, ShouldResemble, st)
- })
-
- }
-
- func Test_MapToObject(t *testing.T) {
- type Base struct {
- Int int
- Bool bool
- String string
- Float float32
- }
- type ArraryStruct struct {
- IntArr []int
- StArr []Base
- ArrArr [][]int
- Nil []Base
- }
- type MapStruct struct {
- StrMap map[string]string
- StMap map[string]Base
- MapMap map[string]map[string]string
- Nil map[string]Base
- }
-
- type Top struct {
- ArrSt ArraryStruct
- MapSt *MapStruct
- BaseIf any
- NilPtr *Base
- }
- Convey("结构体递归转换成map[string]any", t, func() {
- val := Top{
- ArrSt: ArraryStruct{
- IntArr: []int{1, 2, 3},
- StArr: []Base{
- {
- Int: 1,
- Bool: true,
- String: "test",
- Float: 1,
- },
- {
- Int: 2,
- Bool: false,
- String: "test2",
- Float: 2,
- },
- },
- ArrArr: [][]int{
- {1, 2, 3},
- {},
- nil,
- },
- Nil: nil,
- },
- MapSt: &MapStruct{
- StrMap: map[string]string{
- "a": "1",
- "b": "2",
- },
- StMap: map[string]Base{
- "a": {
- Int: 1,
- Bool: true,
- String: "test",
- Float: 1,
- },
- "b": {
- Int: 2,
- Bool: false,
- String: "test2",
- Float: 2,
- },
- },
- MapMap: map[string]map[string]string{
- "a": {
- "a": "1",
- "b": "2",
- },
- "b": nil,
- },
- Nil: nil,
- },
- BaseIf: Base{
- Int: 1,
- Bool: true,
- String: "test",
- Float: 1,
- },
- NilPtr: nil,
- }
-
- retMp, err := ObjectToMap(val)
- So(err, ShouldBeNil)
-
- exceptMap := map[string]any{
- "ArrSt": map[string]any{
- "IntArr": []any{1, 2, 3},
- "StArr": []any{
- map[string]any{
- "Int": 1,
- "Bool": true,
- "String": "test",
- "Float": 1,
- },
- map[string]any{
- "Int": 2,
- "Bool": false,
- "String": "test2",
- "Float": 2,
- },
- },
- "ArrArr": []any{
- []any{1, 2, 3},
- []any{},
- []int(nil),
- },
- "Nil": []Base(nil),
- },
- "MapSt": map[string]any{
- "StrMap": map[string]any{
- "a": "1",
- "b": "2",
- },
- "StMap": map[string]any{
- "a": map[string]any{
- "Int": 1,
- "Bool": true,
- "String": "test",
- "Float": 1,
- },
- "b": map[string]any{
- "Int": 2,
- "Bool": false,
- "String": "test2",
- "Float": 2,
- },
- },
- "MapMap": map[string]any{
- "a": map[string]any{
- "a": "1",
- "b": "2",
- },
- "b": map[string]string(nil),
- },
- "Nil": map[string]Base(nil),
- },
- "BaseIf": map[string]any{
- "Int": 1,
- "Bool": true,
- "String": "test",
- "Float": 1,
- },
- "NilPtr": (*Base)(nil),
- }
-
- mpRetJson, err := ObjectToJSON(retMp)
- So(err, ShouldBeNil)
-
- exceptMapJson, err := ObjectToJSON(exceptMap)
- So(err, ShouldBeNil)
-
- So(string(mpRetJson), ShouldEqualJSON, string(exceptMapJson))
- })
- }
|