5 "github.com/modern-go/reflect2"
11 func encoderOfStruct(ctx *ctx, typ reflect2.Type) ValEncoder {
12 type bindingTo struct {
17 orderedBindings := []*bindingTo{}
18 structDescriptor := describeStruct(ctx, typ)
19 for _, binding := range structDescriptor.Fields {
20 for _, toName := range binding.ToNames {
25 for _, old := range orderedBindings {
26 if old.toName != toName {
29 old.ignored, new.ignored = resolveConflictBinding(ctx.frozenConfig, old.binding, new.binding)
31 orderedBindings = append(orderedBindings, new)
34 if len(orderedBindings) == 0 {
35 return &emptyStructEncoder{}
37 finalOrderedFields := []structFieldTo{}
38 for _, bindingTo := range orderedBindings {
39 if !bindingTo.ignored {
40 finalOrderedFields = append(finalOrderedFields, structFieldTo{
41 encoder: bindingTo.binding.Encoder.(*structFieldEncoder),
42 toName: bindingTo.toName,
46 return &structEncoder{typ, finalOrderedFields}
49 func createCheckIsEmpty(ctx *ctx, typ reflect2.Type) checkIsEmpty {
50 encoder := createEncoderOfNative(ctx, typ)
56 case reflect.Interface:
57 return &dynamicEncoder{typ}
59 return &structEncoder{typ: typ}
61 return &arrayEncoder{}
63 return &sliceEncoder{}
65 return encoderOfMap(ctx, typ)
67 return &OptionalEncoder{}
69 return &lazyErrorEncoder{err: fmt.Errorf("unsupported type: %v", typ)}
73 func resolveConflictBinding(cfg *frozenConfig, old, new *Binding) (ignoreOld, ignoreNew bool) {
74 newTagged := new.Field.Tag().Get(cfg.getTagKey()) != ""
75 oldTagged := old.Field.Tag().Get(cfg.getTagKey()) != ""
78 if len(old.levels) > len(new.levels) {
80 } else if len(new.levels) > len(old.levels) {
92 if len(old.levels) > len(new.levels) {
94 } else if len(new.levels) > len(old.levels) {
102 type structFieldEncoder struct {
103 field reflect2.StructField
104 fieldEncoder ValEncoder
108 func (encoder *structFieldEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
109 fieldPtr := encoder.field.UnsafeGet(ptr)
110 encoder.fieldEncoder.Encode(fieldPtr, stream)
111 if stream.Error != nil && stream.Error != io.EOF {
112 stream.Error = fmt.Errorf("%s: %s", encoder.field.Name(), stream.Error.Error())
116 func (encoder *structFieldEncoder) IsEmpty(ptr unsafe.Pointer) bool {
117 fieldPtr := encoder.field.UnsafeGet(ptr)
118 return encoder.fieldEncoder.IsEmpty(fieldPtr)
121 func (encoder *structFieldEncoder) IsEmbeddedPtrNil(ptr unsafe.Pointer) bool {
122 isEmbeddedPtrNil, converted := encoder.fieldEncoder.(IsEmbeddedPtrNil)
126 fieldPtr := encoder.field.UnsafeGet(ptr)
127 return isEmbeddedPtrNil.IsEmbeddedPtrNil(fieldPtr)
130 type IsEmbeddedPtrNil interface {
131 IsEmbeddedPtrNil(ptr unsafe.Pointer) bool
134 type structEncoder struct {
136 fields []structFieldTo
139 type structFieldTo struct {
140 encoder *structFieldEncoder
144 func (encoder *structEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
145 stream.WriteObjectStart()
147 for _, field := range encoder.fields {
148 if field.encoder.omitempty && field.encoder.IsEmpty(ptr) {
151 if field.encoder.IsEmbeddedPtrNil(ptr) {
157 stream.WriteObjectField(field.toName)
158 field.encoder.Encode(ptr, stream)
161 stream.WriteObjectEnd()
162 if stream.Error != nil && stream.Error != io.EOF {
163 stream.Error = fmt.Errorf("%v.%s", encoder.typ, stream.Error.Error())
167 func (encoder *structEncoder) IsEmpty(ptr unsafe.Pointer) bool {
171 type emptyStructEncoder struct {
174 func (encoder *emptyStructEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
175 stream.WriteEmptyObject()
178 func (encoder *emptyStructEncoder) IsEmpty(ptr unsafe.Pointer) bool {
182 type stringModeNumberEncoder struct {
183 elemEncoder ValEncoder
186 func (encoder *stringModeNumberEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
187 stream.writeByte('"')
188 encoder.elemEncoder.Encode(ptr, stream)
189 stream.writeByte('"')
192 func (encoder *stringModeNumberEncoder) IsEmpty(ptr unsafe.Pointer) bool {
193 return encoder.elemEncoder.IsEmpty(ptr)
196 type stringModeStringEncoder struct {
197 elemEncoder ValEncoder
201 func (encoder *stringModeStringEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
202 tempStream := encoder.cfg.BorrowStream(nil)
203 defer encoder.cfg.ReturnStream(tempStream)
204 encoder.elemEncoder.Encode(ptr, tempStream)
205 stream.WriteString(string(tempStream.Buffer()))
208 func (encoder *stringModeStringEncoder) IsEmpty(ptr unsafe.Pointer) bool {
209 return encoder.elemEncoder.IsEmpty(ptr)