-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy paths2s.go
167 lines (140 loc) · 5.14 KB
/
s2s.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
// Package s2s maps structs to structs using reflection
// It does this in an opinionated way by default, but is configurable
package s2s
import (
"errors"
"fmt"
"reflect"
)
var ErrArgumentsInvalid = errors.New("input or output argument not pointer to struct")
var ErrInvalidConversion = errors.New("ValueMapper return can't be assigned to targetType")
var ErrMissingField = errors.New("Destination struct is missing field")
// Maps field names to allow flexibility in mapping
type NameMapper = func(string) string
// Expected to map `value` to a value of either `targetType` or `*targetType`
// Can be chained using [CompositeMapper]
type ValueMapper = func(value reflect.Value, targetType reflect.Type) interface{}
type MapperConfig struct {
// This function if set will be called every time a field name is used
// allowing you to map handle conversions like CamelCase -> snake_case
NameMapper NameMapper
// This function if set will be called when mapping an input value
// to an output fields type.
// For convenience you are allowed to return both a value directly or
// behind a pointer.
// Also if `MapNilToZeroImplicit` is set (default), returned typed nil
// pointers will be implicitly mapped to the zero value if the output
// field is not of pointer type
ValueMapper ValueMapper
// If set (default) the mapper will not error on fields that can't be
// mapped to an output field
SkipMissingField bool
// If set (default) the mapper will not error if a conversion from input
// field type to output field type fails
SkipFailedConversion bool
// If set (default) the mapper will implicitly convert typed nil pointers
// to zero values of the type as needed
MapNilToZeroImplicit bool
}
var DefaultConfig = MapperConfig{
NameMapper: nil,
ValueMapper: nil,
SkipMissingField: true,
SkipFailedConversion: true,
MapNilToZeroImplicit: true,
}
// MapStruct takes a pointer to an input struct and a pointer to an output struct.
// It will perform a mapping using the default options
func MapStruct(input interface{}, output interface{}) error {
return mapStruct(DefaultConfig, input, output)
}
// MapStructEx takes an additional argument compared to [MapStruct] for configuration.
func MapStructEx(cfg MapperConfig, input interface{}, output interface{}) error {
return mapStruct(cfg, input, output)
}
func mapStruct(cfg MapperConfig, input interface{}, output interface{}) error {
iTyp, oTyp := reflect.TypeOf(input), reflect.TypeOf(output)
// Handle untyped nil
if iTyp == nil || oTyp == nil {
return nil
}
if iTyp.Kind() != reflect.Pointer || oTyp.Kind() != reflect.Pointer {
return ErrArgumentsInvalid
}
iTyp, oTyp = iTyp.Elem(), oTyp.Elem()
if iTyp.Kind() != reflect.Struct || oTyp.Kind() != reflect.Struct {
return ErrArgumentsInvalid
}
iVal := reflect.ValueOf(input).Elem()
oVal := reflect.ValueOf(output).Elem()
// Handle typed nil
if !iVal.IsValid() || !oVal.IsValid() {
return ErrArgumentsInvalid
}
fieldMap := map[string][]int{}
oVisibleFields := reflect.VisibleFields(oTyp)
for _, f := range oVisibleFields {
// Setting unexported fields is a no-no
if !f.IsExported() {
continue
}
mappedName := f.Name
if cfg.NameMapper != nil {
mappedName = cfg.NameMapper(mappedName)
}
fieldMap[mappedName] = f.Index
}
iVisibleFields := reflect.VisibleFields(iTyp)
for _, iField := range iVisibleFields {
// Reading unexported fields is also a no-no
if !iField.IsExported() {
continue
}
iFieldVal := iVal.FieldByIndex(iField.Index)
mappedName := iField.Name
if cfg.NameMapper != nil {
mappedName = cfg.NameMapper(mappedName)
}
oIdx, ok := fieldMap[mappedName]
if !ok {
if !cfg.SkipMissingField {
return fmt.Errorf("%w: %q", ErrMissingField, mappedName)
}
continue
}
oField := oTyp.FieldByIndex(oIdx)
oFieldVal := oVal.FieldByIndex(oIdx)
mappedInput := iFieldVal.Interface()
if cfg.ValueMapper != nil {
mappedInput = cfg.ValueMapper(iFieldVal, oField.Type)
}
mappedInputVal := reflect.ValueOf(mappedInput)
//Decision: Don't modify anything if the returned type is directly assignable
if !mappedInputVal.Type().AssignableTo(oFieldVal.Type()) {
if mappedInputVal.Kind() == reflect.Pointer && mappedInputVal.Type().Elem().AssignableTo(oFieldVal.Type()) {
//Convenience: Allow ValueMapper to return either oField.Type or *oField.Type
if mappedInputVal.IsNil() {
if !cfg.MapNilToZeroImplicit {
return fmt.Errorf("%w: Can't map <nil>(%s) to (%s)", ErrInvalidConversion, mappedInputVal.Type().Name(), oFieldVal.Type())
}
mappedInputVal = reflect.Zero(oFieldVal.Type())
} else {
mappedInputVal = mappedInputVal.Elem()
}
} else if reflect.PointerTo(mappedInputVal.Type()).AssignableTo(oFieldVal.Type()) {
//Decision: Do not modify the things that the output object already points to
//Convenience: If oField has type *mappedInputVal.Type, allocate a new one
valHolder := reflect.New(mappedInputVal.Type())
valHolder.Elem().Set(mappedInputVal)
mappedInputVal = valHolder
} else {
if cfg.SkipFailedConversion {
continue
}
return ErrInvalidConversion
}
}
oFieldVal.Set(mappedInputVal)
}
return nil
}