feature_reflect_extension.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. package jsoniter
  2. import (
  3. "reflect"
  4. "fmt"
  5. "unsafe"
  6. "strings"
  7. "unicode"
  8. )
  9. var typeDecoders = map[string]ValDecoder{}
  10. var fieldDecoders = map[string]ValDecoder{}
  11. var typeEncoders = map[string]ValEncoder{}
  12. var fieldEncoders = map[string]ValEncoder{}
  13. var extensions = []Extension{}
  14. type StructDescriptor struct {
  15. Type reflect.Type
  16. Fields []*Binding
  17. }
  18. func (structDescriptor *StructDescriptor) GetField(fieldName string) *Binding {
  19. for _, binding := range structDescriptor.Fields {
  20. if binding.Field.Name == fieldName {
  21. return binding
  22. }
  23. }
  24. return nil
  25. }
  26. type Binding struct {
  27. Field *reflect.StructField
  28. FromNames []string
  29. ToNames []string
  30. Encoder ValEncoder
  31. Decoder ValDecoder
  32. }
  33. type Extension interface {
  34. UpdateStructDescriptor(structDescriptor *StructDescriptor)
  35. CreateDecoder(typ reflect.Type) ValDecoder
  36. CreateEncoder(typ reflect.Type) ValEncoder
  37. }
  38. type DummyExtension struct {
  39. }
  40. func (extension *DummyExtension) UpdateStructDescriptor(structDescriptor *StructDescriptor) {
  41. }
  42. func (extension *DummyExtension) CreateDecoder(typ reflect.Type) ValDecoder {
  43. return nil
  44. }
  45. func (extension *DummyExtension) CreateEncoder(typ reflect.Type) ValEncoder {
  46. return nil
  47. }
  48. type funcDecoder struct {
  49. fun DecoderFunc
  50. }
  51. func RegisterTypeDecoderFunc(typ string, fun DecoderFunc) {
  52. typeDecoders[typ] = &funcDecoder{fun}
  53. }
  54. func RegisterTypeDecoder(typ string, decoder ValDecoder) {
  55. typeDecoders[typ] = decoder
  56. }
  57. func RegisterFieldDecoderFunc(typ string, field string, fun DecoderFunc) {
  58. RegisterFieldDecoder(typ, field, &funcDecoder{fun})
  59. }
  60. func RegisterFieldDecoder(typ string, field string, decoder ValDecoder) {
  61. fieldDecoders[fmt.Sprintf("%s/%s", typ, field)] = decoder
  62. }
  63. func RegisterTypeEncoderFunc(typ string, fun EncoderFunc, isEmptyFunc func(unsafe.Pointer) bool) {
  64. typeEncoders[typ] = &funcEncoder{fun, isEmptyFunc}
  65. }
  66. func RegisterTypeEncoder(typ string, encoder ValEncoder) {
  67. typeEncoders[typ] = encoder
  68. }
  69. func RegisterFieldEncoderFunc(typ string, field string, fun EncoderFunc, isEmptyFunc func(unsafe.Pointer) bool) {
  70. RegisterFieldEncoder(typ, field, &funcEncoder{fun, isEmptyFunc})
  71. }
  72. func RegisterFieldEncoder(typ string, field string, encoder ValEncoder) {
  73. fieldEncoders[fmt.Sprintf("%s/%s", typ, field)] = encoder
  74. }
  75. func RegisterExtension(extension Extension) {
  76. extensions = append(extensions, extension)
  77. }
  78. func getTypeDecoderFromExtension(typ reflect.Type) ValDecoder {
  79. for _, extension := range extensions {
  80. decoder := extension.CreateDecoder(typ)
  81. if decoder != nil {
  82. return decoder
  83. }
  84. }
  85. typeName := typ.String()
  86. decoder := typeDecoders[typeName]
  87. if decoder != nil {
  88. return decoder
  89. }
  90. if typ.Kind() == reflect.Ptr {
  91. decoder := typeDecoders[typ.Elem().String()]
  92. if decoder != nil {
  93. return &optionalDecoder{typ.Elem(), decoder}
  94. }
  95. }
  96. return nil
  97. }
  98. func getTypeEncoderFromExtension(typ reflect.Type) ValEncoder {
  99. for _, extension := range extensions {
  100. encoder := extension.CreateEncoder(typ)
  101. if encoder != nil {
  102. return encoder
  103. }
  104. }
  105. typeName := typ.String()
  106. encoder := typeEncoders[typeName]
  107. if encoder != nil {
  108. return encoder
  109. }
  110. if typ.Kind() == reflect.Ptr {
  111. encoder := typeEncoders[typ.Elem().String()]
  112. if encoder != nil {
  113. return &optionalEncoder{encoder}
  114. }
  115. }
  116. return nil
  117. }
  118. func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, error) {
  119. bindings := []*Binding{}
  120. for i := 0; i < typ.NumField(); i++ {
  121. field := typ.Field(i)
  122. if field.Anonymous {
  123. if field.Type.Kind() == reflect.Struct {
  124. structDescriptor, err := describeStruct(cfg, field.Type)
  125. if err != nil {
  126. return nil, err
  127. }
  128. for _, binding := range structDescriptor.Fields {
  129. bindings = append(bindings, binding)
  130. }
  131. } else if field.Type.Kind() == reflect.Ptr && field.Type.Elem().Kind() == reflect.Struct {
  132. structDescriptor, err := describeStruct(cfg, field.Type.Elem())
  133. if err != nil {
  134. return nil, err
  135. }
  136. for _, binding := range structDescriptor.Fields {
  137. binding.Encoder = &optionalEncoder{binding.Encoder}
  138. binding.Encoder = &structFieldEncoder{&field, binding.Encoder, false}
  139. binding.Decoder = &optionalDecoder{field.Type, binding.Decoder}
  140. binding.Decoder = &structFieldDecoder{&field, binding.Decoder}
  141. bindings = append(bindings, binding)
  142. }
  143. }
  144. } else {
  145. tagParts := strings.Split(field.Tag.Get("json"), ",")
  146. fieldNames := calcFieldNames(field.Name, tagParts[0])
  147. fieldCacheKey := fmt.Sprintf("%s/%s", typ.String(), field.Name)
  148. decoder := fieldDecoders[fieldCacheKey]
  149. if decoder == nil && len(fieldNames) > 0 {
  150. var err error
  151. decoder, err = decoderOfType(cfg, field.Type)
  152. if err != nil {
  153. return nil, err
  154. }
  155. }
  156. encoder := fieldEncoders[fieldCacheKey]
  157. if encoder == nil && len(fieldNames) > 0 {
  158. var err error
  159. encoder, err = encoderOfType(cfg, field.Type)
  160. if err != nil {
  161. return nil, err
  162. }
  163. // map is stored as pointer in the struct
  164. if field.Type.Kind() == reflect.Map {
  165. encoder = &optionalEncoder{encoder}
  166. }
  167. }
  168. binding := &Binding{
  169. Field: &field,
  170. FromNames: fieldNames,
  171. ToNames: fieldNames,
  172. Decoder: decoder,
  173. Encoder: encoder,
  174. }
  175. shouldOmitEmpty := false
  176. for _, tagPart := range tagParts[1:] {
  177. if tagPart == "omitempty" {
  178. shouldOmitEmpty = true
  179. } else if tagPart == "string" {
  180. binding.Decoder = &stringModeDecoder{binding.Decoder}
  181. binding.Encoder = &stringModeEncoder{binding.Encoder}
  182. }
  183. }
  184. binding.Decoder = &structFieldDecoder{&field, binding.Decoder}
  185. binding.Encoder = &structFieldEncoder{&field, binding.Encoder, shouldOmitEmpty}
  186. bindings = append(bindings, binding)
  187. }
  188. }
  189. structDescriptor := &StructDescriptor{
  190. Type: typ,
  191. Fields: bindings,
  192. }
  193. for _, extension := range extensions {
  194. extension.UpdateStructDescriptor(structDescriptor)
  195. }
  196. return structDescriptor, nil
  197. }
  198. func listStructFields(typ reflect.Type) []*reflect.StructField {
  199. fields := []*reflect.StructField{}
  200. return fields
  201. }
  202. func calcFieldNames(originalFieldName string, tagProvidedFieldName string) []string {
  203. // tag => exported? => original
  204. isNotExported := unicode.IsLower(rune(originalFieldName[0]))
  205. var fieldNames []string
  206. /// tagParts[0] always present, even if no tags
  207. switch tagProvidedFieldName {
  208. case "":
  209. if isNotExported {
  210. fieldNames = []string{}
  211. } else {
  212. fieldNames = []string{originalFieldName}
  213. }
  214. case "-":
  215. fieldNames = []string{}
  216. default:
  217. fieldNames = []string{tagProvidedFieldName}
  218. }
  219. return fieldNames
  220. }