form_mapping.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. // Copyright 2014 Manu Martinez-Almeida. All rights reserved.
  2. // Use of this source code is governed by a MIT style
  3. // license that can be found in the LICENSE file.
  4. package binding
  5. import (
  6. "errors"
  7. "fmt"
  8. "reflect"
  9. "strconv"
  10. "strings"
  11. "time"
  12. "github.com/gin-gonic/gin/internal/json"
  13. )
  14. var errUnknownType = errors.New("Unknown type")
  15. func mapUri(ptr interface{}, m map[string][]string) error {
  16. return mapFormByTag(ptr, m, "uri")
  17. }
  18. func mapForm(ptr interface{}, form map[string][]string) error {
  19. return mapFormByTag(ptr, form, "form")
  20. }
  21. var emptyField = reflect.StructField{}
  22. func mapFormByTag(ptr interface{}, form map[string][]string, tag string) error {
  23. _, err := mapping(reflect.ValueOf(ptr), emptyField, form, tag)
  24. return err
  25. }
  26. func mapping(value reflect.Value, field reflect.StructField, form map[string][]string, tag string) (bool, error) {
  27. var vKind = value.Kind()
  28. if vKind == reflect.Ptr {
  29. var isNew bool
  30. vPtr := value
  31. if value.IsNil() {
  32. isNew = true
  33. vPtr = reflect.New(value.Type().Elem())
  34. }
  35. isSetted, err := mapping(vPtr.Elem(), field, form, tag)
  36. if err != nil {
  37. return false, err
  38. }
  39. if isNew && isSetted {
  40. value.Set(vPtr)
  41. }
  42. return isSetted, nil
  43. }
  44. ok, err := tryToSetValue(value, field, form, tag)
  45. if err != nil {
  46. return false, err
  47. }
  48. if ok {
  49. return true, nil
  50. }
  51. if vKind == reflect.Struct {
  52. tValue := value.Type()
  53. var isSetted bool
  54. for i := 0; i < value.NumField(); i++ {
  55. if !value.Field(i).CanSet() {
  56. continue
  57. }
  58. ok, err := mapping(value.Field(i), tValue.Field(i), form, tag)
  59. if err != nil {
  60. return false, err
  61. }
  62. isSetted = isSetted || ok
  63. }
  64. return isSetted, nil
  65. }
  66. return false, nil
  67. }
  68. func tryToSetValue(value reflect.Value, field reflect.StructField, form map[string][]string, tag string) (bool, error) {
  69. var tagValue, defaultValue string
  70. var isDefaultExists bool
  71. tagValue = field.Tag.Get(tag)
  72. tagValue, opts := head(tagValue, ",")
  73. if tagValue == "-" { // just ignoring this field
  74. return false, nil
  75. }
  76. if tagValue == "" { // default value is FieldName
  77. tagValue = field.Name
  78. }
  79. if tagValue == "" { // when field is "emptyField" variable
  80. return false, nil
  81. }
  82. var opt string
  83. for len(opts) > 0 {
  84. opt, opts = head(opts, ",")
  85. k, v := head(opt, "=")
  86. switch k {
  87. case "default":
  88. isDefaultExists = true
  89. defaultValue = v
  90. }
  91. }
  92. vs, ok := form[tagValue]
  93. if !ok && !isDefaultExists {
  94. return false, nil
  95. }
  96. switch value.Kind() {
  97. case reflect.Slice:
  98. if !ok {
  99. vs = []string{defaultValue}
  100. }
  101. return true, setSlice(vs, value, field)
  102. case reflect.Array:
  103. if !ok {
  104. vs = []string{defaultValue}
  105. }
  106. if len(vs) != value.Len() {
  107. return false, fmt.Errorf("%q is not valid value for %s", vs, value.Type().String())
  108. }
  109. return true, setArray(vs, value, field)
  110. default:
  111. var val string
  112. if !ok {
  113. val = defaultValue
  114. }
  115. if len(vs) > 0 {
  116. val = vs[0]
  117. }
  118. return true, setWithProperType(val, value, field)
  119. }
  120. }
  121. func setWithProperType(val string, value reflect.Value, field reflect.StructField) error {
  122. switch value.Kind() {
  123. case reflect.Int:
  124. return setIntField(val, 0, value)
  125. case reflect.Int8:
  126. return setIntField(val, 8, value)
  127. case reflect.Int16:
  128. return setIntField(val, 16, value)
  129. case reflect.Int32:
  130. return setIntField(val, 32, value)
  131. case reflect.Int64:
  132. switch value.Interface().(type) {
  133. case time.Duration:
  134. return setTimeDuration(val, value, field)
  135. }
  136. return setIntField(val, 64, value)
  137. case reflect.Uint:
  138. return setUintField(val, 0, value)
  139. case reflect.Uint8:
  140. return setUintField(val, 8, value)
  141. case reflect.Uint16:
  142. return setUintField(val, 16, value)
  143. case reflect.Uint32:
  144. return setUintField(val, 32, value)
  145. case reflect.Uint64:
  146. return setUintField(val, 64, value)
  147. case reflect.Bool:
  148. return setBoolField(val, value)
  149. case reflect.Float32:
  150. return setFloatField(val, 32, value)
  151. case reflect.Float64:
  152. return setFloatField(val, 64, value)
  153. case reflect.String:
  154. value.SetString(val)
  155. case reflect.Struct:
  156. switch value.Interface().(type) {
  157. case time.Time:
  158. return setTimeField(val, field, value)
  159. }
  160. return json.Unmarshal([]byte(val), value.Addr().Interface())
  161. case reflect.Map:
  162. return json.Unmarshal([]byte(val), value.Addr().Interface())
  163. default:
  164. return errUnknownType
  165. }
  166. return nil
  167. }
  168. func setIntField(val string, bitSize int, field reflect.Value) error {
  169. if val == "" {
  170. val = "0"
  171. }
  172. intVal, err := strconv.ParseInt(val, 10, bitSize)
  173. if err == nil {
  174. field.SetInt(intVal)
  175. }
  176. return err
  177. }
  178. func setUintField(val string, bitSize int, field reflect.Value) error {
  179. if val == "" {
  180. val = "0"
  181. }
  182. uintVal, err := strconv.ParseUint(val, 10, bitSize)
  183. if err == nil {
  184. field.SetUint(uintVal)
  185. }
  186. return err
  187. }
  188. func setBoolField(val string, field reflect.Value) error {
  189. if val == "" {
  190. val = "false"
  191. }
  192. boolVal, err := strconv.ParseBool(val)
  193. if err == nil {
  194. field.SetBool(boolVal)
  195. }
  196. return err
  197. }
  198. func setFloatField(val string, bitSize int, field reflect.Value) error {
  199. if val == "" {
  200. val = "0.0"
  201. }
  202. floatVal, err := strconv.ParseFloat(val, bitSize)
  203. if err == nil {
  204. field.SetFloat(floatVal)
  205. }
  206. return err
  207. }
  208. func setTimeField(val string, structField reflect.StructField, value reflect.Value) error {
  209. timeFormat := structField.Tag.Get("time_format")
  210. if timeFormat == "" {
  211. timeFormat = time.RFC3339
  212. }
  213. if val == "" {
  214. value.Set(reflect.ValueOf(time.Time{}))
  215. return nil
  216. }
  217. l := time.Local
  218. if isUTC, _ := strconv.ParseBool(structField.Tag.Get("time_utc")); isUTC {
  219. l = time.UTC
  220. }
  221. if locTag := structField.Tag.Get("time_location"); locTag != "" {
  222. loc, err := time.LoadLocation(locTag)
  223. if err != nil {
  224. return err
  225. }
  226. l = loc
  227. }
  228. t, err := time.ParseInLocation(timeFormat, val, l)
  229. if err != nil {
  230. return err
  231. }
  232. value.Set(reflect.ValueOf(t))
  233. return nil
  234. }
  235. func setArray(vals []string, value reflect.Value, field reflect.StructField) error {
  236. for i, s := range vals {
  237. err := setWithProperType(s, value.Index(i), field)
  238. if err != nil {
  239. return err
  240. }
  241. }
  242. return nil
  243. }
  244. func setSlice(vals []string, value reflect.Value, field reflect.StructField) error {
  245. slice := reflect.MakeSlice(value.Type(), len(vals), len(vals))
  246. err := setArray(vals, slice, field)
  247. if err != nil {
  248. return err
  249. }
  250. value.Set(slice)
  251. return nil
  252. }
  253. func setTimeDuration(val string, value reflect.Value, field reflect.StructField) error {
  254. d, err := time.ParseDuration(val)
  255. if err != nil {
  256. return err
  257. }
  258. value.Set(reflect.ValueOf(d))
  259. return nil
  260. }
  261. func head(str, sep string) (head string, tail string) {
  262. idx := strings.Index(str, sep)
  263. if idx < 0 {
  264. return str, ""
  265. }
  266. return str[:idx], str[idx+len(sep):]
  267. }