form_mapping.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  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. "net/url"
  8. "reflect"
  9. "strconv"
  10. "time"
  11. )
  12. // MapForm form values map to struct
  13. func MapForm(ptr interface{}, values url.Values) error {
  14. return mapForm(ptr, values)
  15. }
  16. func mapForm(ptr interface{}, form map[string][]string) error {
  17. typ := reflect.TypeOf(ptr).Elem()
  18. val := reflect.ValueOf(ptr).Elem()
  19. for i := 0; i < typ.NumField(); i++ {
  20. typeField := typ.Field(i)
  21. structField := val.Field(i)
  22. if !structField.CanSet() {
  23. continue
  24. }
  25. structFieldKind := structField.Kind()
  26. inputFieldName := typeField.Tag.Get("form")
  27. if inputFieldName == "" {
  28. inputFieldName = typeField.Name
  29. // if "form" tag is nil, we inspect if the field is a struct.
  30. // this would not make sense for JSON parsing but it does for a form
  31. // since data is flatten
  32. if structFieldKind == reflect.Struct {
  33. err := mapForm(structField.Addr().Interface(), form)
  34. if err != nil {
  35. return err
  36. }
  37. continue
  38. }
  39. }
  40. inputValue, exists := form[inputFieldName]
  41. if !exists {
  42. continue
  43. }
  44. numElems := len(inputValue)
  45. if structFieldKind == reflect.Slice && numElems > 0 {
  46. sliceOf := structField.Type().Elem().Kind()
  47. slice := reflect.MakeSlice(structField.Type(), numElems, numElems)
  48. for i := 0; i < numElems; i++ {
  49. if err := setWithProperType(sliceOf, inputValue[i], slice.Index(i)); err != nil {
  50. return err
  51. }
  52. }
  53. val.Field(i).Set(slice)
  54. } else {
  55. if _, isTime := structField.Interface().(time.Time); isTime {
  56. if err := setTimeField(inputValue[0], typeField, structField); err != nil {
  57. return err
  58. }
  59. continue
  60. }
  61. if err := setWithProperType(typeField.Type.Kind(), inputValue[0], structField); err != nil {
  62. return err
  63. }
  64. }
  65. }
  66. return nil
  67. }
  68. func setWithProperType(valueKind reflect.Kind, val string, structField reflect.Value) error {
  69. switch valueKind {
  70. case reflect.Int:
  71. return setIntField(val, 0, structField)
  72. case reflect.Int8:
  73. return setIntField(val, 8, structField)
  74. case reflect.Int16:
  75. return setIntField(val, 16, structField)
  76. case reflect.Int32:
  77. return setIntField(val, 32, structField)
  78. case reflect.Int64:
  79. return setIntField(val, 64, structField)
  80. case reflect.Uint:
  81. return setUintField(val, 0, structField)
  82. case reflect.Uint8:
  83. return setUintField(val, 8, structField)
  84. case reflect.Uint16:
  85. return setUintField(val, 16, structField)
  86. case reflect.Uint32:
  87. return setUintField(val, 32, structField)
  88. case reflect.Uint64:
  89. return setUintField(val, 64, structField)
  90. case reflect.Bool:
  91. return setBoolField(val, structField)
  92. case reflect.Float32:
  93. return setFloatField(val, 32, structField)
  94. case reflect.Float64:
  95. return setFloatField(val, 64, structField)
  96. case reflect.String:
  97. structField.SetString(val)
  98. default:
  99. return errors.New("Unknown type")
  100. }
  101. return nil
  102. }
  103. func setIntField(val string, bitSize int, field reflect.Value) error {
  104. if val == "" {
  105. val = "0"
  106. }
  107. intVal, err := strconv.ParseInt(val, 10, bitSize)
  108. if err == nil {
  109. field.SetInt(intVal)
  110. }
  111. return err
  112. }
  113. func setUintField(val string, bitSize int, field reflect.Value) error {
  114. if val == "" {
  115. val = "0"
  116. }
  117. uintVal, err := strconv.ParseUint(val, 10, bitSize)
  118. if err == nil {
  119. field.SetUint(uintVal)
  120. }
  121. return err
  122. }
  123. func setBoolField(val string, field reflect.Value) error {
  124. if val == "" {
  125. val = "false"
  126. }
  127. boolVal, err := strconv.ParseBool(val)
  128. if err == nil {
  129. field.SetBool(boolVal)
  130. }
  131. return nil
  132. }
  133. func setFloatField(val string, bitSize int, field reflect.Value) error {
  134. if val == "" {
  135. val = "0.0"
  136. }
  137. floatVal, err := strconv.ParseFloat(val, bitSize)
  138. if err == nil {
  139. field.SetFloat(floatVal)
  140. }
  141. return err
  142. }
  143. func setTimeField(val string, structField reflect.StructField, value reflect.Value) error {
  144. timeFormat := structField.Tag.Get("time_format")
  145. if timeFormat == "" {
  146. return errors.New("Blank time format")
  147. }
  148. if val == "" {
  149. value.Set(reflect.ValueOf(time.Time{}))
  150. return nil
  151. }
  152. l := time.Local
  153. if isUTC, _ := strconv.ParseBool(structField.Tag.Get("time_utc")); isUTC {
  154. l = time.UTC
  155. }
  156. t, err := time.ParseInLocation(timeFormat, val, l)
  157. if err != nil {
  158. return err
  159. }
  160. value.Set(reflect.ValueOf(t))
  161. return nil
  162. }
  163. // Don't pass in pointers to bind to. Can lead to bugs. See:
  164. // https://github.com/codegangsta/martini-contrib/issues/40
  165. // https://github.com/codegangsta/martini-contrib/pull/34#issuecomment-29683659
  166. func ensureNotPointer(obj interface{}) {
  167. if reflect.TypeOf(obj).Kind() == reflect.Ptr {
  168. panic("Pointers are not accepted as binding models")
  169. }
  170. }