form_mapping.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  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. "log"
  8. "reflect"
  9. "strconv"
  10. )
  11. func mapForm(ptr interface{}, form map[string][]string) error {
  12. typ := reflect.TypeOf(ptr).Elem()
  13. val := reflect.ValueOf(ptr).Elem()
  14. for i := 0; i < typ.NumField(); i++ {
  15. typeField := typ.Field(i)
  16. structField := val.Field(i)
  17. if !structField.CanSet() {
  18. continue
  19. }
  20. inputFieldName := typeField.Tag.Get("form")
  21. if inputFieldName == "" {
  22. inputFieldName = typeField.Name
  23. }
  24. inputValue, exists := form[inputFieldName]
  25. if !exists {
  26. continue
  27. }
  28. numElems := len(inputValue)
  29. if structField.Kind() == reflect.Slice && numElems > 0 {
  30. sliceOf := structField.Type().Elem().Kind()
  31. slice := reflect.MakeSlice(structField.Type(), numElems, numElems)
  32. for i := 0; i < numElems; i++ {
  33. if err := setWithProperType(sliceOf, inputValue[i], slice.Index(i)); err != nil {
  34. return err
  35. }
  36. }
  37. val.Field(i).Set(slice)
  38. } else {
  39. if err := setWithProperType(typeField.Type.Kind(), inputValue[0], structField); err != nil {
  40. return err
  41. }
  42. }
  43. }
  44. return nil
  45. }
  46. func setWithProperType(valueKind reflect.Kind, val string, structField reflect.Value) error {
  47. switch valueKind {
  48. case reflect.Int:
  49. return setIntField(val, 0, structField)
  50. case reflect.Int8:
  51. return setIntField(val, 8, structField)
  52. case reflect.Int16:
  53. return setIntField(val, 16, structField)
  54. case reflect.Int32:
  55. return setIntField(val, 32, structField)
  56. case reflect.Int64:
  57. return setIntField(val, 64, structField)
  58. case reflect.Uint:
  59. return setUintField(val, 0, structField)
  60. case reflect.Uint8:
  61. return setUintField(val, 8, structField)
  62. case reflect.Uint16:
  63. return setUintField(val, 16, structField)
  64. case reflect.Uint32:
  65. return setUintField(val, 32, structField)
  66. case reflect.Uint64:
  67. return setUintField(val, 64, structField)
  68. case reflect.Bool:
  69. return setBoolField(val, structField)
  70. case reflect.Float32:
  71. return setFloatField(val, 32, structField)
  72. case reflect.Float64:
  73. return setFloatField(val, 64, structField)
  74. case reflect.String:
  75. structField.SetString(val)
  76. default:
  77. return errors.New("Unknown type")
  78. }
  79. return nil
  80. }
  81. func setIntField(val string, bitSize int, field reflect.Value) error {
  82. if val == "" {
  83. val = "0"
  84. }
  85. intVal, err := strconv.ParseInt(val, 10, bitSize)
  86. if err == nil {
  87. field.SetInt(intVal)
  88. }
  89. return err
  90. }
  91. func setUintField(val string, bitSize int, field reflect.Value) error {
  92. if val == "" {
  93. val = "0"
  94. }
  95. uintVal, err := strconv.ParseUint(val, 10, bitSize)
  96. if err == nil {
  97. field.SetUint(uintVal)
  98. }
  99. return err
  100. }
  101. func setBoolField(val string, field reflect.Value) error {
  102. if val == "" {
  103. val = "false"
  104. }
  105. boolVal, err := strconv.ParseBool(val)
  106. if err == nil {
  107. field.SetBool(boolVal)
  108. }
  109. return nil
  110. }
  111. func setFloatField(val string, bitSize int, field reflect.Value) error {
  112. if val == "" {
  113. val = "0.0"
  114. }
  115. floatVal, err := strconv.ParseFloat(val, bitSize)
  116. if err == nil {
  117. field.SetFloat(floatVal)
  118. }
  119. return err
  120. }
  121. // Don't pass in pointers to bind to. Can lead to bugs. See:
  122. // https://github.com/codegangsta/martini-contrib/issues/40
  123. // https://github.com/codegangsta/martini-contrib/pull/34#issuecomment-29683659
  124. func ensureNotPointer(obj interface{}) {
  125. if reflect.TypeOf(obj).Kind() == reflect.Ptr {
  126. log.Panic("Pointers are not accepted as binding models")
  127. }
  128. }