/
struct_to_url_values.go
88 lines (75 loc) · 1.85 KB
/
struct_to_url_values.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
package utils
import (
"fmt"
"net/url"
"reflect"
"strconv"
"time"
)
type mappedURLValue struct {
queryParamSuffix *string
value *string
}
func StructToURLValues(v interface{}) *url.Values {
structValue := reflect.ValueOf(v)
if structValue.IsNil() {
return nil
}
values := url.Values{}
structValue = structValue.Elem()
for i := 0; i < structValue.NumField(); i++ {
slice := fieldToMappedURLValueSlice(structValue.Field(i))
for _, urlValue := range slice {
if urlValue.value != nil {
name := structValue.Type().Field(i).Name
if urlValue.queryParamSuffix != nil {
name = name + "." + *urlValue.queryParamSuffix
}
values.Add(name, *urlValue.value)
}
}
}
return &values
}
func fieldToMappedURLValueSlice(value reflect.Value) []mappedURLValue {
var v reflect.Value
if value.Kind() == reflect.Ptr {
if value.IsNil() {
return nil
}
v = value.Elem()
} else {
v = value
}
switch v.Kind() {
case reflect.Slice:
slice := make([]mappedURLValue, 0)
for i := 0; i < v.Len(); i++ {
slice = append(slice, mappedURLValue{value: fieldToStringPointer(v.Index(i).Interface())})
}
return slice
case reflect.Map:
slice := make([]mappedURLValue, 0)
for _, k := range v.MapKeys() {
slice = append(slice, mappedURLValue{queryParamSuffix: String(k.Interface().(string)), value: fieldToStringPointer(v.MapIndex(k).Interface())})
}
return slice
default:
return []mappedURLValue{{value: fieldToStringPointer(v.Interface())}}
}
}
func fieldToStringPointer(v interface{}) *string {
switch vType := v.(type) {
case string:
return String(v.(string))
case int:
return String(strconv.Itoa(v.(int)))
case bool:
return String(strconv.FormatBool(v.(bool)))
case time.Time:
return String(v.(time.Time).Format(time.RFC3339))
default:
fmt.Printf("Type %v is not supported\n", vType)
return nil
}
}