package serialization import ( "os" "reflect" "strconv" "testing" "github.com/stretchr/testify/require" expect "github.com/yusing/goutils/testing" ) func TestDeserialize(t *testing.T) { type S struct { I int S string IS []int SS []string MSI map[string]int MIS map[int]string } var ( testStruct = S{ I: 1, S: "hello", IS: []int{1, 2, 3}, SS: []string{"a", "b", "c"}, MSI: map[string]int{"a": 1, "b": 2, "c": 3}, MIS: map[int]string{1: "a", 2: "b", 3: "c"}, } testStructSerialized = map[string]any{ "I": 1, "S": "hello", "IS": []int{1, 2, 3}, "SS": []string{"a", "b", "c"}, "MSI": map[string]int{"a": 1, "b": 2, "c": 3}, "MIS": map[int]string{1: "a", 2: "b", 3: "c"}, } ) t.Run("deserialize", func(t *testing.T) { var s2 S err := MapUnmarshalValidate(testStructSerialized, &s2) expect.NoError(t, err) expect.Equal(t, s2, testStruct) }) } func TestDeserializeAnonymousField(t *testing.T) { type Anon struct { A, B int } var s struct { Anon C int } var s2 struct { *Anon C int } // all, anon := extractFields(reflect.TypeOf(s2)) // t.Fatalf("anon %v, all %v", anon, all) err := MapUnmarshalValidate(map[string]any{"a": 1, "b": 2, "c": 3}, &s) expect.NoError(t, err) expect.Equal(t, s.A, 1) expect.Equal(t, s.B, 2) expect.Equal(t, s.C, 3) err = MapUnmarshalValidate(map[string]any{"a": 1, "b": 2, "c": 3}, &s2) expect.NoError(t, err) expect.Equal(t, s2.A, 1) expect.Equal(t, s2.B, 2) expect.Equal(t, s2.C, 3) } func TestPointerPrimitives(t *testing.T) { type testType struct { B *bool `json:"b"` I8 *int8 `json:"i8"` I16 *int16 `json:"i16"` I32 *int32 `json:"i32"` I64 *int64 `json:"i64"` U8 *uint8 `json:"u8"` U16 *uint16 `json:"u16"` U32 *uint32 `json:"u32"` U64 *uint64 `json:"u64"` } var test testType err := MapUnmarshalValidate(map[string]any{"b": true, "i8": int8(127), "i16": int16(127), "i32": int32(127), "i64": int64(127), "u8": uint8(127), "u16": uint16(127), "u32": uint32(127), "u64": uint64(127)}, &test) expect.NoError(t, err) expect.Equal(t, *test.B, true) expect.Equal(t, *test.I8, int8(127)) expect.Equal(t, *test.I16, int16(127)) expect.Equal(t, *test.I32, int32(127)) expect.Equal(t, *test.I64, int64(127)) expect.Equal(t, *test.U8, uint8(127)) expect.Equal(t, *test.U16, uint16(127)) expect.Equal(t, *test.U32, uint32(127)) expect.Equal(t, *test.U64, uint64(127)) // zero values err = MapUnmarshalValidate(map[string]any{"b": false, "i8": int8(0), "i16": int16(0), "i32": int32(0), "i64": int64(0), "u8": uint8(0), "u16": uint16(0), "u32": uint32(0), "u64": uint64(0)}, &test) expect.NoError(t, err) expect.Equal(t, *test.B, false) expect.Equal(t, *test.I8, int8(0)) expect.Equal(t, *test.I16, int16(0)) expect.Equal(t, *test.I32, int32(0)) expect.Equal(t, *test.I64, int64(0)) expect.Equal(t, *test.U8, uint8(0)) expect.Equal(t, *test.U16, uint16(0)) expect.Equal(t, *test.U32, uint32(0)) expect.Equal(t, *test.U64, uint64(0)) // nil values err = MapUnmarshalValidate(map[string]any{"b": true, "i8": int8(127), "i16": int16(127), "i32": int32(127), "i64": int64(127), "u8": uint8(127), "u16": uint16(127), "u32": uint32(127), "u64": uint64(127)}, &test) expect.NoError(t, err) err = MapUnmarshalValidate(map[string]any{"b": nil, "i8": nil, "i16": nil, "i32": nil, "i64": nil, "u8": nil, "u16": nil, "u32": nil, "u64": nil}, &test) expect.NoError(t, err) expect.Equal(t, test.B, nil) expect.Equal(t, test.I8, nil) expect.Equal(t, test.I16, nil) expect.Equal(t, test.I32, nil) expect.Equal(t, test.I64, nil) expect.Equal(t, test.U8, nil) expect.Equal(t, test.U16, nil) expect.Equal(t, test.U32, nil) expect.Equal(t, test.U64, nil) } func TestStringIntConvert(t *testing.T) { s := "127" test := struct { i8 int8 i16 int16 i32 int32 i64 int64 u8 uint8 u16 uint16 u32 uint32 u64 uint64 }{} ok, err := ConvertString(s, reflect.ValueOf(&test.i8)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.i8, int8(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.i16)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.i16, int16(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.i32)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.i32, int32(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.i64)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.i64, int64(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.u8)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.u8, uint8(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.u16)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.u16, uint16(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.u32)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.u32, uint32(127)) ok, err = ConvertString(s, reflect.ValueOf(&test.u64)) expect.True(t, ok) expect.NoError(t, err) expect.Equal(t, test.u64, uint64(127)) } type testModel struct { Test testType Baz string } type testType struct { foo int bar string } func (c *testType) Parse(v string) (err error) { c.bar = v c.foo, err = strconv.Atoi(v) return err } func TestConvertor(t *testing.T) { t.Run("valid", func(t *testing.T) { m := new(testModel) expect.NoError(t, MapUnmarshalValidate(map[string]any{"Test": "123"}, m)) expect.Equal(t, m.Test.foo, 123) expect.Equal(t, m.Test.bar, "123") }) t.Run("int_to_string", func(t *testing.T) { m := new(testModel) expect.NoError(t, MapUnmarshalValidate(map[string]any{"Test": "123"}, m)) expect.Equal(t, m.Test.foo, 123) expect.Equal(t, m.Test.bar, "123") expect.NoError(t, MapUnmarshalValidate(map[string]any{"Baz": 123}, m)) expect.Equal(t, m.Baz, "123") }) t.Run("invalid", func(t *testing.T) { m := new(testModel) err := MapUnmarshalValidate(map[string]any{"Test": struct{ a int }{1}}, m) expect.ErrorIs(t, ErrUnsupportedConversion, err) }) t.Run("set_empty", func(t *testing.T) { m := testModel{ Test: testType{1, "2"}, Baz: "3", } expect.NoError(t, MapUnmarshalValidate(map[string]any{"Test": nil, "Baz": nil}, &m)) expect.Equal(t, m, testModel{}) }) } func TestStringToSlice(t *testing.T) { t.Run("comma_separated", func(t *testing.T) { dst := make([]string, 0) convertible, err := ConvertString("a,b,c", reflect.ValueOf(&dst)) expect.True(t, convertible) expect.NoError(t, err) expect.Equal(t, dst, []string{"a", "b", "c"}) }) t.Run("yaml-like", func(t *testing.T) { dst := make([]string, 0) convertible, err := ConvertString("- a\n- b\n- c", reflect.ValueOf(&dst)) expect.True(t, convertible) expect.NoError(t, err) expect.Equal(t, dst, []string{"a", "b", "c"}) }) t.Run("single-line-yaml-like", func(t *testing.T) { dst := make([]string, 0) convertible, err := ConvertString("- a", reflect.ValueOf(&dst)) expect.True(t, convertible) expect.NoError(t, err) expect.Equal(t, dst, []string{"a"}) }) } func TestStringToMap(t *testing.T) { t.Run("yaml-like", func(t *testing.T) { dst := make(map[string]string) convertible, err := ConvertString(" a: b\n c: d", reflect.ValueOf(&dst)) expect.True(t, convertible) expect.NoError(t, err) expect.Equal(t, dst, map[string]string{"a": "b", "c": "d"}) }) } func TestStringToStruct(t *testing.T) { t.Run("yaml-like", func(t *testing.T) { dst := struct { A string B int }{} convertible, err := ConvertString(" A: a\n B: 123", reflect.ValueOf(&dst)) expect.True(t, convertible) expect.NoError(t, err) expect.Equal(t, dst, struct { A string B int }{"a", 123}) }) } func TestConfigEnvSubstitution(t *testing.T) { os.Setenv("CLOUDFLARE_AUTH_TOKEN", "test") data := []byte(` --- autocert: options: auth_token: ${CLOUDFLARE_AUTH_TOKEN} `) var cfg struct { Autocert struct { Options struct { AuthToken string `yaml:"auth_token"` } `yaml:"options"` } `yaml:"autocert"` } require.NoError(t, UnmarshalValidateYAML(data, &cfg)) require.Equal(t, "test", cfg.Autocert.Options.AuthToken) }