/
common.go
104 lines (93 loc) · 2.19 KB
/
common.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
package cfgur
import (
"errors"
"github.com/spf13/cast"
"reflect"
"strings"
)
type fieldWithTag struct {
field reflect.Value
tag string
defValue interface{}
}
func parseValue(kind reflect.Kind, val string) interface{} {
switch kind {
case reflect.String:
return val
case reflect.Int:
return cast.ToInt(val)
case reflect.Int8:
return cast.ToInt8(val)
case reflect.Int16:
return cast.ToInt16(val)
case reflect.Int32:
return cast.ToInt32(val)
case reflect.Int64:
return cast.ToInt64(val)
case reflect.Bool:
return cast.ToBool(val)
default:
return val
}
}
func getFieldWithTag(val interface{}) ([]fieldWithTag, error) {
v := reflect.ValueOf(val)
if v.Kind() != reflect.Ptr || v.IsNil() {
return nil, errors.New("cfgur unmarshal failed: expecting a non-nil pointer")
}
e := v.Elem()
t := e.Type()
fields := make([]fieldWithTag, 0)
for i := 0; i < e.NumField(); i++ {
f := e.Field(i)
tagVal := t.Field(i).Tag.Get("cfgur")
if tagVal != "" {
tagParts := strings.Split(tagVal, ",")
tagKey := tagParts[0]
var defValue interface{}
if len(tagParts) > 1 && strings.HasPrefix(tagParts[1], "default=") {
defValue = parseValue(f.Kind(), strings.TrimPrefix(tagParts[1], "default="))
} else {
defValue = reflect.Zero(f.Type()).Interface()
}
fields = append(fields, fieldWithTag{
field: f,
tag: tagKey,
defValue: defValue,
})
}
}
return fields, nil
}
type commonMarshaller struct {
ConfigProvider
}
func (m *commonMarshaller) Unmarshal(val interface{}) error {
fields, err := getFieldWithTag(val)
if err != nil {
return err
}
for _, field := range fields {
switch field.field.Kind() {
case reflect.String:
if v, ok := m.ConfigProvider.GetStringE(field.tag); ok {
field.field.SetString(v)
} else {
field.field.SetString(field.defValue.(string))
}
case reflect.Int:
if v, ok := m.ConfigProvider.GetIntE(field.tag); ok {
field.field.SetInt(v)
} else {
field.field.SetInt(int64(field.defValue.(int)))
}
case reflect.Bool:
if v, ok := m.ConfigProvider.GetBoolE(field.tag); ok {
field.field.SetBool(v)
} else {
field.field.SetBool(field.defValue.(bool))
}
}
}
return nil
}