feature_reflect_map.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. package jsoniter
  2. import (
  3. "encoding"
  4. "encoding/json"
  5. "reflect"
  6. "sort"
  7. "strconv"
  8. "unsafe"
  9. )
  10. type mapDecoder struct {
  11. mapType reflect.Type
  12. keyType reflect.Type
  13. elemType reflect.Type
  14. elemDecoder ValDecoder
  15. mapInterface emptyInterface
  16. }
  17. func (decoder *mapDecoder) Decode(ptr unsafe.Pointer, iter *Iterator) {
  18. // dark magic to cast unsafe.Pointer back to interface{} using reflect.Type
  19. mapInterface := decoder.mapInterface
  20. mapInterface.word = ptr
  21. realInterface := (*interface{})(unsafe.Pointer(&mapInterface))
  22. realVal := reflect.ValueOf(*realInterface).Elem()
  23. if iter.ReadNil() {
  24. realVal.Set(reflect.Zero(decoder.mapType))
  25. return
  26. }
  27. if realVal.IsNil() {
  28. realVal.Set(reflect.MakeMap(realVal.Type()))
  29. }
  30. iter.ReadMapCB(func(iter *Iterator, keyStr string) bool {
  31. elem := reflect.New(decoder.elemType)
  32. decoder.elemDecoder.Decode(unsafe.Pointer(elem.Pointer()), iter)
  33. // to put into map, we have to use reflection
  34. keyType := decoder.keyType
  35. switch {
  36. case keyType.Kind() == reflect.String:
  37. realVal.SetMapIndex(reflect.ValueOf(keyStr).Convert(keyType), elem.Elem())
  38. return true
  39. case keyType.Implements(textUnmarshalerType):
  40. textUnmarshaler := reflect.New(keyType.Elem()).Interface().(encoding.TextUnmarshaler)
  41. err := textUnmarshaler.UnmarshalText([]byte(keyStr))
  42. if err != nil {
  43. iter.ReportError("read map key as TextUnmarshaler", err.Error())
  44. return false
  45. }
  46. realVal.SetMapIndex(reflect.ValueOf(textUnmarshaler), elem.Elem())
  47. return true
  48. default:
  49. switch keyType.Kind() {
  50. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  51. n, err := strconv.ParseInt(keyStr, 10, 64)
  52. if err != nil || reflect.Zero(keyType).OverflowInt(n) {
  53. iter.ReportError("read map key as int64", "read int64 failed")
  54. return false
  55. }
  56. realVal.SetMapIndex(reflect.ValueOf(n).Convert(keyType), elem.Elem())
  57. return true
  58. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  59. n, err := strconv.ParseUint(keyStr, 10, 64)
  60. if err != nil || reflect.Zero(keyType).OverflowUint(n) {
  61. iter.ReportError("read map key as uint64", "read uint64 failed")
  62. return false
  63. }
  64. realVal.SetMapIndex(reflect.ValueOf(n).Convert(keyType), elem.Elem())
  65. return true
  66. }
  67. }
  68. iter.ReportError("read map key", "unexpected map key type "+keyType.String())
  69. return true
  70. })
  71. }
  72. type mapEncoder struct {
  73. mapType reflect.Type
  74. elemType reflect.Type
  75. elemEncoder ValEncoder
  76. mapInterface emptyInterface
  77. }
  78. func (encoder *mapEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
  79. mapInterface := encoder.mapInterface
  80. mapInterface.word = ptr
  81. realInterface := (*interface{})(unsafe.Pointer(&mapInterface))
  82. realVal := reflect.ValueOf(*realInterface)
  83. stream.WriteObjectStart()
  84. for i, key := range realVal.MapKeys() {
  85. if i != 0 {
  86. stream.WriteMore()
  87. }
  88. encodeMapKey(key, stream)
  89. stream.writeByte(':')
  90. val := realVal.MapIndex(key).Interface()
  91. encoder.elemEncoder.EncodeInterface(val, stream)
  92. }
  93. stream.WriteObjectEnd()
  94. }
  95. func encodeMapKey(key reflect.Value, stream *Stream) {
  96. if key.Kind() == reflect.String {
  97. stream.WriteString(key.String())
  98. return
  99. }
  100. if tm, ok := key.Interface().(encoding.TextMarshaler); ok {
  101. buf, err := tm.MarshalText()
  102. if err != nil {
  103. stream.Error = err
  104. return
  105. }
  106. stream.writeByte('"')
  107. stream.Write(buf)
  108. stream.writeByte('"')
  109. return
  110. }
  111. switch key.Kind() {
  112. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  113. stream.writeByte('"')
  114. stream.WriteInt64(key.Int())
  115. stream.writeByte('"')
  116. return
  117. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  118. stream.writeByte('"')
  119. stream.WriteUint64(key.Uint())
  120. stream.writeByte('"')
  121. return
  122. }
  123. stream.Error = &json.UnsupportedTypeError{key.Type()}
  124. }
  125. func (encoder *mapEncoder) EncodeInterface(val interface{}, stream *Stream) {
  126. WriteToStream(val, stream, encoder)
  127. }
  128. func (encoder *mapEncoder) IsEmpty(ptr unsafe.Pointer) bool {
  129. mapInterface := encoder.mapInterface
  130. mapInterface.word = ptr
  131. realInterface := (*interface{})(unsafe.Pointer(&mapInterface))
  132. realVal := reflect.ValueOf(*realInterface)
  133. return realVal.Len() == 0
  134. }
  135. type sortKeysMapEncoder struct {
  136. mapType reflect.Type
  137. elemType reflect.Type
  138. elemEncoder ValEncoder
  139. mapInterface emptyInterface
  140. }
  141. func (encoder *sortKeysMapEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
  142. mapInterface := encoder.mapInterface
  143. mapInterface.word = ptr
  144. realInterface := (*interface{})(unsafe.Pointer(&mapInterface))
  145. realVal := reflect.ValueOf(*realInterface)
  146. // Extract and sort the keys.
  147. keys := realVal.MapKeys()
  148. sv := make([]reflectWithString, len(keys))
  149. for i, v := range keys {
  150. sv[i].v = v
  151. if err := sv[i].resolve(); err != nil {
  152. stream.Error = err
  153. return
  154. }
  155. }
  156. sort.Slice(sv, func(i, j int) bool { return sv[i].s < sv[j].s })
  157. stream.WriteObjectStart()
  158. for i, key := range sv {
  159. if i != 0 {
  160. stream.WriteMore()
  161. }
  162. stream.WriteVal(key.s) // might need html escape, so can not WriteString directly
  163. stream.writeByte(':')
  164. val := realVal.MapIndex(key.v).Interface()
  165. encoder.elemEncoder.EncodeInterface(val, stream)
  166. }
  167. stream.WriteObjectEnd()
  168. }
  169. // stringValues is a slice of reflect.Value holding *reflect.StringValue.
  170. // It implements the methods to sort by string.
  171. type stringValues []reflectWithString
  172. type reflectWithString struct {
  173. v reflect.Value
  174. s string
  175. }
  176. func (w *reflectWithString) resolve() error {
  177. if w.v.Kind() == reflect.String {
  178. w.s = w.v.String()
  179. return nil
  180. }
  181. if tm, ok := w.v.Interface().(encoding.TextMarshaler); ok {
  182. buf, err := tm.MarshalText()
  183. w.s = string(buf)
  184. return err
  185. }
  186. switch w.v.Kind() {
  187. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  188. w.s = strconv.FormatInt(w.v.Int(), 10)
  189. return nil
  190. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  191. w.s = strconv.FormatUint(w.v.Uint(), 10)
  192. return nil
  193. }
  194. return &json.UnsupportedTypeError{w.v.Type()}
  195. }
  196. func (sv stringValues) Len() int { return len(sv) }
  197. func (sv stringValues) Swap(i, j int) { sv[i], sv[j] = sv[j], sv[i] }
  198. func (sv stringValues) Less(i, j int) bool { return sv[i].s < sv[j].s }
  199. func (encoder *sortKeysMapEncoder) EncodeInterface(val interface{}, stream *Stream) {
  200. WriteToStream(val, stream, encoder)
  201. }
  202. func (encoder *sortKeysMapEncoder) IsEmpty(ptr unsafe.Pointer) bool {
  203. mapInterface := encoder.mapInterface
  204. mapInterface.word = ptr
  205. realInterface := (*interface{})(unsafe.Pointer(&mapInterface))
  206. realVal := reflect.ValueOf(*realInterface)
  207. return realVal.Len() == 0
  208. }