forked from liip/sheriff
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsheriff.go
288 lines (251 loc) · 7.59 KB
/
sheriff.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
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
package sheriff
import (
"encoding"
"encoding/json"
"fmt"
"reflect"
"strings"
"github.com/hashicorp/go-version"
)
// Options determine which struct fields are being added to the output map.
type Options struct {
// Groups determine which fields are getting marshalled based on the groups tag.
// A field with multiple groups (comma-separated) will result in marshalling of that
// field if one of their groups is specified.
Groups []string
// ApiVersion sets the API version to use when marshalling.
// The tags `since` and `until` use the API version setting.
// Specifying the API version as "1.0.0" and having an until setting of "2"
// will result in the field being marshalled.
// Specifying a since setting of "2" with the same API version specified,
// will not marshal the field.
ApiVersion *version.Version
// IncludeEmptyTag determines whether a field without the
// `groups` tag should be marshalled ot not.
// This option is false by default.
IncludeEmptyTag bool
// This is used internally so that we can propagate anonymous fields groups tag to all child field.
nestedGroupsMap map[string][]string
}
// MarshalInvalidTypeError is an error returned to indicate the wrong type has been
// passed to Marshal.
type MarshalInvalidTypeError struct {
// t reflects the type of the data
t reflect.Kind
// data contains the passed data itself
data interface{}
}
func (e MarshalInvalidTypeError) Error() string {
return fmt.Sprintf("marshaller: Unable to marshal type %s. Struct required.", e.t)
}
// Marshaller is the interface models have to implement in order to conform to marshalling.
type Marshaller interface {
Marshal(options *Options) (interface{}, error)
}
// Marshal encodes the passed data into a map which can be used to pass to json.Marshal().
//
// If the passed argument `data` is a struct, the return value will be of type `map[string]interface{}`.
// In all other cases we can't derive the type in a meaningful way and is therefore an `interface{}`.
func Marshal(options *Options, data interface{}) (interface{}, error) {
v := reflect.ValueOf(data)
if !v.IsValid() {
return data, nil
}
t := v.Type()
// Initialise nestedGroupsMap,
// TODO: this may impact the performance, find a better place for this.
if options.nestedGroupsMap == nil {
options.nestedGroupsMap = make(map[string][]string)
}
checkGroups := len(options.Groups) > 0
if t.Kind() == reflect.Ptr {
// follow pointer
t = t.Elem()
}
if v.Kind() == reflect.Ptr {
// follow pointer
v = v.Elem()
}
if t.Kind() != reflect.Struct {
return marshalValue(options, v)
}
dest := make(map[string]interface{})
for i := 0; i < t.NumField(); i++ {
field := t.Field(i)
val := v.Field(i)
jsonTag, jsonOpts := parseTag(field.Tag.Get("json"))
// If no json tag is provided, use the field Name
if jsonTag == "" {
jsonTag = field.Name
}
if jsonTag == "-" {
continue
}
if jsonOpts.Contains("omitempty") && isEmptyValue(val) {
continue
}
// skip unexported fields
if !val.IsValid() || !val.CanInterface() {
continue
}
// if there is an anonymous field which is a struct
// we want the childs exposed at the toplevel to be
// consistent with the embedded json marshaller
if val.Kind() == reflect.Ptr {
val = val.Elem()
}
// we can skip the group checkif if the field is a composition field
isEmbeddedField := field.Anonymous && val.Kind() == reflect.Struct
if isEmbeddedField && field.Type.Kind() == reflect.Struct {
tt := field.Type
parentGroups := strings.Split(field.Tag.Get("groups"), ",")
for i := 0; i < tt.NumField(); i++ {
nestedField := tt.Field(i)
options.nestedGroupsMap[nestedField.Name] = parentGroups
}
}
if !isEmbeddedField {
if checkGroups {
var groups []string
if field.Tag.Get("groups") != "" {
groups = strings.Split(field.Tag.Get("groups"), ",")
}
if len(groups) == 0 && options.nestedGroupsMap[field.Name] != nil {
groups = append(groups, options.nestedGroupsMap[field.Name]...)
}
// Marshall the field if
// - it has at least one of the requested groups
// or
// - it has no group and 'IncludeEmptyTag' is set to true
shouldShow := listContains(groups, options.Groups) || (len(groups) == 0 && options.IncludeEmptyTag)
// Prevent marshalling of the field if
// - it should not be shown (above)
// or
// - it has no groups and 'IncludeEmptyTag' is set to false
shouldHide := !shouldShow || (len(groups) == 0 && !options.IncludeEmptyTag)
if shouldHide {
// skip this field
continue
}
}
if since := field.Tag.Get("since"); since != "" {
sinceVersion, err := version.NewVersion(since)
if err != nil {
return nil, err
}
if options.ApiVersion.LessThan(sinceVersion) {
continue
}
}
if until := field.Tag.Get("until"); until != "" {
untilVersion, err := version.NewVersion(until)
if err != nil {
return nil, err
}
if options.ApiVersion.GreaterThan(untilVersion) {
continue
}
}
}
v, err := marshalValue(options, val)
if err != nil {
return nil, err
}
// when a composition field we want to bring the child
// nodes to the top
nestedVal, ok := v.(map[string]interface{})
if isEmbeddedField && ok {
for key, value := range nestedVal {
dest[key] = value
}
} else {
dest[jsonTag] = v
}
}
return dest, nil
}
// marshalValue is being used for getting the actual value of a field.
//
// There is support for types implementing the Marshaller interface, arbitrary structs, slices, maps and base types.
func marshalValue(options *Options, v reflect.Value) (interface{}, error) {
// return nil on nil pointer struct fields
if !v.IsValid() || !v.CanInterface() {
return nil, nil
}
val := v.Interface()
if marshaller, ok := val.(Marshaller); ok {
return marshaller.Marshal(options)
}
// types which are e.g. structs, slices or maps and implement one of the following interfaces should not be
// marshalled by sheriff because they'll be correctly marshalled by json.Marshal instead.
// Otherwise (e.g. net.IP) a byte slice may be output as a list of uints instead of as an IP string.
switch val.(type) {
case json.Marshaler, encoding.TextMarshaler, fmt.Stringer:
return val, nil
}
k := v.Kind()
switch k {
case reflect.Interface, reflect.Map, reflect.Ptr:
if v.IsNil() {
return val, nil
}
}
if k == reflect.Ptr {
v = v.Elem()
val = v.Interface()
k = v.Kind()
}
if k == reflect.Interface || k == reflect.Struct {
return Marshal(options, val)
}
if k == reflect.Slice {
l := v.Len()
dest := make([]interface{}, l)
for i := 0; i < l; i++ {
d, err := marshalValue(options, v.Index(i))
if err != nil {
return nil, err
}
dest[i] = d
}
return dest, nil
}
if k == reflect.Map {
mapKeys := v.MapKeys()
if len(mapKeys) == 0 {
return nil, nil
}
if mapKeys[0].Kind() != reflect.String {
return nil, MarshalInvalidTypeError{t: mapKeys[0].Kind(), data: val}
}
dest := make(map[string]interface{})
for _, key := range mapKeys {
d, err := marshalValue(options, v.MapIndex(key))
if err != nil {
return nil, err
}
dest[key.String()] = d
}
return dest, nil
}
return val, nil
}
// contains check if a given key is contained in a slice of strings.
func contains(key string, list []string) bool {
for _, innerKey := range list {
if key == innerKey {
return true
}
}
return false
}
// listContains operates on two string slices and checks if one of the strings in `a`
// is contained in `b`.
func listContains(a []string, b []string) bool {
for _, key := range a {
if contains(key, b) {
return true
}
}
return false
}