reflect.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. package reflect
  2. import (
  3. "errors"
  4. "fmt"
  5. "reflect"
  6. "strconv"
  7. "strings"
  8. )
  9. var (
  10. allowTags = []string{"json", "yaml", "xml", "name"}
  11. )
  12. var (
  13. ErrValueAssociated = errors.New("value cannot be associated")
  14. )
  15. func findField(v reflect.Value, field string) reflect.Value {
  16. var (
  17. pos int
  18. tagValue string
  19. refType reflect.Type
  20. fieldType reflect.StructField
  21. )
  22. refType = v.Type()
  23. for i := 0; i < refType.NumField(); i++ {
  24. fieldType = refType.Field(i)
  25. for _, tagName := range allowTags {
  26. tagValue = fieldType.Tag.Get(tagName)
  27. if tagValue == "" {
  28. continue
  29. }
  30. if pos = strings.IndexByte(tagValue, ','); pos != -1 {
  31. tagValue = tagValue[:pos]
  32. }
  33. if tagValue == field {
  34. return v.Field(i)
  35. }
  36. }
  37. }
  38. return v.FieldByName(field)
  39. }
  40. func Assign(variable reflect.Value, value interface{}) (err error) {
  41. return safeAssignment(variable, value)
  42. }
  43. func safeAssignment(variable reflect.Value, value interface{}) (err error) {
  44. var (
  45. n int64
  46. un uint64
  47. fn float64
  48. kind reflect.Kind
  49. )
  50. rv := reflect.ValueOf(value)
  51. kind = variable.Kind()
  52. if kind != reflect.Slice && kind != reflect.Array && kind != reflect.Map && kind == rv.Kind() {
  53. variable.Set(rv)
  54. return
  55. }
  56. switch kind {
  57. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  58. switch rv.Kind() {
  59. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  60. variable.SetInt(rv.Int())
  61. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  62. variable.SetInt(int64(rv.Uint()))
  63. case reflect.Float32, reflect.Float64:
  64. variable.SetInt(int64(rv.Float()))
  65. case reflect.String:
  66. if n, err = strconv.ParseInt(rv.String(), 10, 64); err == nil {
  67. variable.SetInt(n)
  68. }
  69. default:
  70. err = fmt.Errorf("integer value can not assign %s", rv.Kind())
  71. }
  72. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  73. switch rv.Kind() {
  74. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  75. variable.SetUint(uint64(rv.Int()))
  76. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  77. variable.SetUint(rv.Uint())
  78. case reflect.Float32, reflect.Float64:
  79. variable.SetUint(uint64(rv.Float()))
  80. case reflect.String:
  81. if un, err = strconv.ParseUint(rv.String(), 10, 64); err == nil {
  82. variable.SetUint(un)
  83. }
  84. default:
  85. err = fmt.Errorf("unsigned integer value can not assign %s", rv.Kind())
  86. }
  87. case reflect.Float32, reflect.Float64:
  88. switch rv.Kind() {
  89. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  90. variable.SetFloat(float64(rv.Int()))
  91. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  92. variable.SetFloat(float64(rv.Uint()))
  93. case reflect.Float32, reflect.Float64:
  94. variable.SetFloat(rv.Float())
  95. case reflect.String:
  96. if fn, err = strconv.ParseFloat(rv.String(), 64); err == nil {
  97. variable.SetFloat(fn)
  98. }
  99. default:
  100. err = fmt.Errorf("decimal value can not assign %s", rv.Kind())
  101. }
  102. case reflect.String:
  103. switch rv.Kind() {
  104. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  105. variable.SetString(strconv.FormatInt(rv.Int(), 10))
  106. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
  107. variable.SetString(strconv.FormatUint(rv.Uint(), 10))
  108. case reflect.Float32, reflect.Float64:
  109. variable.SetString(strconv.FormatFloat(rv.Float(), 'f', -1, 64))
  110. case reflect.String:
  111. variable.SetString(rv.String())
  112. default:
  113. variable.SetString(fmt.Sprint(value))
  114. }
  115. case reflect.Interface:
  116. variable.Set(rv)
  117. default:
  118. err = fmt.Errorf("unsupported kind %s", kind)
  119. }
  120. return
  121. }
  122. func Set(hacky interface{}, field string, value interface{}) (err error) {
  123. var (
  124. n int
  125. refField reflect.Value
  126. )
  127. refVal := reflect.ValueOf(hacky)
  128. if refVal.Kind() == reflect.Ptr {
  129. refVal = reflect.Indirect(refVal)
  130. }
  131. if refVal.Kind() != reflect.Struct {
  132. return fmt.Errorf("%s kind is %v", refVal.Type().String(), refField.Kind())
  133. }
  134. refField = findField(refVal, field)
  135. if !refField.IsValid() {
  136. return fmt.Errorf("%s field `%s` not found", refVal.Type(), field)
  137. }
  138. rv := reflect.ValueOf(value)
  139. fieldKind := refField.Kind()
  140. if fieldKind != reflect.Slice && fieldKind != reflect.Array && fieldKind != reflect.Map && fieldKind == rv.Kind() {
  141. refField.Set(rv)
  142. return
  143. }
  144. switch fieldKind {
  145. case reflect.Struct:
  146. if rv.Kind() != reflect.Map {
  147. return ErrValueAssociated
  148. }
  149. keys := rv.MapKeys()
  150. subVal := reflect.New(refField.Type())
  151. for _, key := range keys {
  152. pv := rv.MapIndex(key)
  153. if key.Kind() == reflect.String {
  154. if err = Set(subVal.Interface(), key.String(), pv.Interface()); err != nil {
  155. return err
  156. }
  157. }
  158. }
  159. refField.Set(subVal.Elem())
  160. case reflect.Ptr:
  161. elemType := refField.Type()
  162. if elemType.Elem().Kind() != reflect.Struct {
  163. return ErrValueAssociated
  164. } else {
  165. if rv.Kind() != reflect.Map {
  166. return ErrValueAssociated
  167. }
  168. keys := rv.MapKeys()
  169. subVal := reflect.New(elemType.Elem())
  170. for _, key := range keys {
  171. pv := rv.MapIndex(key)
  172. if key.Kind() == reflect.String {
  173. if err = Set(subVal.Interface(), key.String(), pv.Interface()); err != nil {
  174. return err
  175. }
  176. }
  177. }
  178. refField.Set(subVal)
  179. }
  180. case reflect.Map:
  181. if rv.Kind() != reflect.Map {
  182. return ErrValueAssociated
  183. }
  184. targetValue := reflect.MakeMap(refField.Type())
  185. keys := rv.MapKeys()
  186. for _, key := range keys {
  187. pv := rv.MapIndex(key)
  188. kVal := reflect.New(refField.Type().Key())
  189. eVal := reflect.New(refField.Type().Elem())
  190. if err = safeAssignment(kVal.Elem(), key.Interface()); err != nil {
  191. return ErrValueAssociated
  192. }
  193. if refField.Type().Elem().Kind() == reflect.Struct {
  194. if pv.Elem().Kind() != reflect.Map {
  195. return ErrValueAssociated
  196. }
  197. subKeys := pv.Elem().MapKeys()
  198. for _, subKey := range subKeys {
  199. subVal := pv.Elem().MapIndex(subKey)
  200. if subKey.Kind() == reflect.String {
  201. if err = Set(eVal.Interface(), subKey.String(), subVal.Interface()); err != nil {
  202. return err
  203. }
  204. }
  205. }
  206. targetValue.SetMapIndex(kVal.Elem(), eVal.Elem())
  207. } else {
  208. if err = safeAssignment(eVal.Elem(), pv.Interface()); err != nil {
  209. return ErrValueAssociated
  210. }
  211. targetValue.SetMapIndex(kVal.Elem(), eVal.Elem())
  212. }
  213. }
  214. refField.Set(targetValue)
  215. case reflect.Array, reflect.Slice:
  216. n = 0
  217. innerType := refField.Type().Elem()
  218. if rv.Kind() == reflect.Array || rv.Kind() == reflect.Slice {
  219. if innerType.Kind() == reflect.Struct {
  220. sliceVar := reflect.MakeSlice(refField.Type(), rv.Len(), rv.Len())
  221. for i := 0; i < rv.Len(); i++ {
  222. srcVal := rv.Index(i)
  223. if srcVal.Kind() != reflect.Map {
  224. return ErrValueAssociated
  225. }
  226. dstVal := reflect.New(innerType)
  227. keys := srcVal.MapKeys()
  228. for _, key := range keys {
  229. kv := srcVal.MapIndex(key)
  230. if key.Kind() == reflect.String {
  231. if err = Set(dstVal.Interface(), key.String(), kv.Interface()); err != nil {
  232. return
  233. }
  234. }
  235. }
  236. sliceVar.Index(n).Set(dstVal.Elem())
  237. n++
  238. }
  239. refField.Set(sliceVar.Slice(0, n))
  240. } else if innerType.Kind() == reflect.Ptr {
  241. sliceVar := reflect.MakeSlice(refField.Type(), rv.Len(), rv.Len())
  242. for i := 0; i < rv.Len(); i++ {
  243. srcVal := rv.Index(i)
  244. if srcVal.Kind() != reflect.Map {
  245. return ErrValueAssociated
  246. }
  247. dstVal := reflect.New(innerType.Elem())
  248. keys := srcVal.MapKeys()
  249. for _, key := range keys {
  250. kv := srcVal.MapIndex(key)
  251. if key.Kind() == reflect.String {
  252. if err = Set(dstVal.Interface(), key.String(), kv.Interface()); err != nil {
  253. return
  254. }
  255. }
  256. }
  257. sliceVar.Index(n).Set(dstVal)
  258. n++
  259. }
  260. refField.Set(sliceVar.Slice(0, n))
  261. } else {
  262. sliceVar := reflect.MakeSlice(refField.Type(), rv.Len(), rv.Len())
  263. for i := 0; i < rv.Len(); i++ {
  264. srcVal := rv.Index(i)
  265. dstVal := reflect.New(innerType).Elem()
  266. if err = safeAssignment(dstVal, srcVal.Interface()); err != nil {
  267. return
  268. }
  269. sliceVar.Index(n).Set(dstVal)
  270. n++
  271. }
  272. refField.Set(sliceVar.Slice(0, n))
  273. }
  274. }
  275. default:
  276. err = safeAssignment(refField, value)
  277. }
  278. return
  279. }