-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfield.go
243 lines (205 loc) · 6.83 KB
/
field.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
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
package opt
import (
"errors"
"fmt"
"net/url"
"reflect"
"regexp"
"strings"
"unicode/utf8"
"github.com/goloop/scs"
)
const (
// The tagNameOption the identifier of the tag that sets
// the main option name.
tagNameOpt = "opt"
// The tagNameAlt the identifier of the tag that sets
// the alternative option name.
tagNameAlt = "alt"
// The tagNameDefValue the identifier of the tag that
// sets the default value.
tagNameDefValue = "def"
// The tagNameHelp the identifier of the tag that sets the help value.
tagNameHelpMsg = "help"
// The tagNameSepList the identifier of the tag that sets the delimiter
// for the tagNameDefValue field if the struct field is a list.
tagNameSepList = "sep"
// The defValueIgnored is the value of the tagNameOption field that
// should be ignored during processing.
defValueIgnored = "-"
// The defSep sets the default delimiter for the tagNameDefValue field
// if a struct field is a list. An empty value indicates that the data
// will not be divided into items, but will be considered as one item
// in the list.
defSep = ""
)
var (
// The orderFlagRgx a regular expression to check if a string is a number.
orderFlagRgx = regexp.MustCompile(`^\d+$`)
// The shortFlagRgx a regular expression to check
// if a string is short option.
shortFlagRgx = regexp.MustCompile(`^(\?|\[\]|[A-Za-z]{1})$`)
// The shortFlagSafeRgx as the shortFlagRgx but without the
// ability to win special tags like: ?, [].
shortFlagSafeRgx = regexp.MustCompile(`^([A-Za-z]{1})$`)
// The longRgx a regular expression to check if a string is long option.
longFlagRgx = regexp.MustCompile(`^[A-Za-z]{1}[A-Za-z\-1-9]{1,}$`)
)
// The tagGroup is the tag group of a field.
type tagGroup struct {
shortFlag string // short flag
longFlag string // long flag
defValue string // default value
helpMsg string // help information about field
sepList string // list delimiter for defValue
isIgnored bool // true if ignore the field
}
// The fieldCast is field data structure.
type fieldCast struct {
fieldName string // just field name
tagGroup *tagGroup // tga group
item *reflect.Value // field instance
}
// The fieldCastList is list of field data structure.
type fieldCastList []*fieldCast
// The flags function returns map of field's flags in opt and alt tags.
func (fcl fieldCastList) flags() map[string]int {
result := make(map[string]int, len(fcl))
for _, fc := range fcl {
if fc.tagGroup.shortFlag != "" {
result[fc.tagGroup.shortFlag]++
}
if fc.tagGroup.longFlag != "" {
result[fc.tagGroup.longFlag]++
}
}
return result
}
// The validateStruct checks whether the object is a pointer to the structure,
// and returns reflect.Type and reflect.Value of the object. If the object is
// not a pointer to the structure or object is nil, it returns an error.
func validateStruct(obj interface{}) (reflect.Type, reflect.Value, error) {
rt, rv, err := reflect.TypeOf(obj), reflect.ValueOf(obj), error(nil)
// Check object type
// Object should be a pointer to a non-empty struct.
if obj == nil {
err = errors.New("obj is nil")
} else if rv.Kind() != reflect.Ptr || rv.IsNil() {
err = errors.New("obj should be a non-nil pointer to a struct")
} else if rv.Type().Elem().Kind() != reflect.Struct {
err = errors.New("obj should be a pointer to a struct")
} else if rv.Elem().NumField() == 0 {
err = errors.New("obj should be a pointer to a non-empty struct")
}
return rt, rv, err
}
// The getFieldCastList parses the structure fields and
// returns list of the fieldCast.
func getFieldCastList(obj interface{}) (fieldCastList, error) {
var result fieldCastList
// Check object type.
rt, rv, err := validateStruct(obj)
if err != nil {
return result, err
}
elem := rv.Elem()
urlS, urlP := reflect.TypeOf(url.URL{}), reflect.TypeOf((*url.URL)(nil))
for i := 0; i < elem.NumField(); i++ {
// Get tag data from the field.
field := rt.Elem().Field(i)
// Get tag group.
tg, err := getTagGroup(
field.Name,
strings.Trim(field.Tag.Get(tagNameOpt), " -"),
strings.Trim(field.Tag.Get(tagNameAlt), " -"),
field.Tag.Get(tagNameDefValue),
field.Tag.Get(tagNameSepList),
field.Tag.Get(tagNameHelpMsg),
)
if err != nil {
return result, err
} else if tg.isIgnored {
continue
}
// Collect fields for further analysis.
item := elem.FieldByName(field.Name)
fc := fieldCast{fieldName: field.Name, tagGroup: &tg, item: &item}
kind := fc.item.Kind()
switch f := fc.tagGroup.shortFlag; {
case f == "?" && kind != reflect.String:
// To load doc, the field must be of the string type.
err = fmt.Errorf("%s field should be a string", fc.fieldName)
case f == "[]" && kind != reflect.Array && kind != reflect.Slice:
// To load positional arguments,
// the field must be of the slice type.
err = fmt.Errorf("%s field should be a list", fc.fieldName)
case kind == reflect.Struct && fc.item.Type() != urlS:
// Supported url.URL struct only.
err = fmt.Errorf("%s field has invalid type", fc.fieldName)
case kind == reflect.Ptr:
// Pointer to the structure *url.URL only.
k, t := fc.item.Type().Elem().Kind(), fc.item.Type()
if k == reflect.Struct && t != urlP {
err = fmt.Errorf("%s field has invalid type", fc.fieldName)
}
}
if err != nil {
return result, err
}
result = append(result, &fc)
}
return result, nil
}
// getTagGroup returns a tagGroup with the specified tag values.
func getTagGroup(
fieldName,
optTagValue,
altTagValue,
defTagValue,
sepTagValue,
helpTagValue string,
) (tagGroup, error) {
var err error
// Create tag group with default values.
tg := tagGroup{
defValue: defTagValue,
sepList: sepTagValue,
helpMsg: helpTagValue,
}
// The fieldName must be used for an empty value of optTagValue.
if optTagValue == "" {
// Convert pascal case to kebab case for long flag name only.
optTagValue = fieldName
if utf8.RuneCountInString(fieldName) > 1 {
if kebab, err := scs.PascalToKebab(fieldName); err == nil {
optTagValue = kebab
}
}
}
msg := "invalid %s tag value %s"
optByte, altByte := []byte(optTagValue), []byte(altTagValue)
if orderFlagRgx.Match(optByte) || shortFlagRgx.Match(optByte) {
tg.shortFlag = optTagValue
if longFlagRgx.Match(altByte) {
tg.longFlag = altTagValue
} else if altTagValue != "" {
err = fmt.Errorf(msg, tagNameAlt, altTagValue)
}
} else if longFlagRgx.Match(optByte) {
tg.longFlag = optTagValue
if shortFlagSafeRgx.Match(altByte) {
tg.shortFlag = altTagValue
} else if altTagValue != "" {
err = fmt.Errorf(msg, tagNameAlt, altTagValue)
}
} else if optTagValue == defValueIgnored {
tg.isIgnored = true
} else {
err = fmt.Errorf(msg, tagNameOpt, optTagValue)
}
tg.longFlag = strings.ToLower(tg.longFlag)
if utf8.RuneCountInString(tg.longFlag) > 32 {
err = fmt.Errorf("%s is a very long name, max 32 chars", tg.longFlag)
}
return tg, err
}