forked from mirror/go-json
Support int and uint types as map key type
This commit is contained in:
parent
699ab766f5
commit
85577616f8
|
@ -214,6 +214,8 @@ func (t opType) fieldToStringTagField() opType {
|
|||
"uint", "uint8", "uint16", "uint32", "uint64",
|
||||
"float32", "float64", "bool", "string", "bytes",
|
||||
"array", "map", "mapLoad", "slice", "struct", "MarshalJSON", "MarshalText", "recursive",
|
||||
"intString", "int8String", "int16String", "int32String", "int64String",
|
||||
"uintString", "uint8String", "uint16String", "uint32String", "uint64String",
|
||||
}
|
||||
primitiveTypesUpper := []string{}
|
||||
for _, typ := range primitiveTypes {
|
||||
|
|
|
@ -75,6 +75,27 @@ func (d *Decoder) compile(typ *rtype, structName, fieldName string) (decoder, er
|
|||
return nil, &UnsupportedTypeError{Type: rtype2type(typ)}
|
||||
}
|
||||
|
||||
func (d *Decoder) compileMapKey(typ *rtype, structName, fieldName string) (decoder, error) {
|
||||
dec, err := d.compile(typ, structName, fieldName)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
for {
|
||||
switch t := dec.(type) {
|
||||
case *stringDecoder, *interfaceDecoder, *unmarshalJSONDecoder, *unmarshalTextDecoder:
|
||||
return dec, nil
|
||||
case *boolDecoder, *intDecoder, *uintDecoder, *numberDecoder:
|
||||
return newWrappedStringDecoder(dec, structName, fieldName), nil
|
||||
case *ptrDecoder:
|
||||
dec = t.dec
|
||||
default:
|
||||
goto ERROR
|
||||
}
|
||||
}
|
||||
ERROR:
|
||||
return nil, &UnsupportedTypeError{Type: rtype2type(typ)}
|
||||
}
|
||||
|
||||
func (d *Decoder) compilePtr(typ *rtype, structName, fieldName string) (decoder, error) {
|
||||
dec, err := d.compile(typ.Elem(), structName, fieldName)
|
||||
if err != nil {
|
||||
|
@ -186,7 +207,7 @@ func (d *Decoder) compileArray(typ *rtype, structName, fieldName string) (decode
|
|||
}
|
||||
|
||||
func (d *Decoder) compileMap(typ *rtype, structName, fieldName string) (decoder, error) {
|
||||
keyDec, err := d.compile(typ.Key(), structName, fieldName)
|
||||
keyDec, err := d.compileMapKey(typ.Key(), structName, fieldName)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
|
|
@ -129,6 +129,26 @@ func (e *Encoder) compileKey(ctx *encodeCompileContext) (*opcode, error) {
|
|||
return e.compileInterface(ctx)
|
||||
case reflect.String:
|
||||
return e.compileString(ctx)
|
||||
case reflect.Int:
|
||||
return e.compileIntString(ctx)
|
||||
case reflect.Int8:
|
||||
return e.compileInt8String(ctx)
|
||||
case reflect.Int16:
|
||||
return e.compileInt16String(ctx)
|
||||
case reflect.Int32:
|
||||
return e.compileInt32String(ctx)
|
||||
case reflect.Int64:
|
||||
return e.compileInt64String(ctx)
|
||||
case reflect.Uint:
|
||||
return e.compileUintString(ctx)
|
||||
case reflect.Uint8:
|
||||
return e.compileUint8String(ctx)
|
||||
case reflect.Uint16:
|
||||
return e.compileUint16String(ctx)
|
||||
case reflect.Uint32:
|
||||
return e.compileUint32String(ctx)
|
||||
case reflect.Uint64:
|
||||
return e.compileUint64String(ctx)
|
||||
}
|
||||
return nil, &UnsupportedTypeError{Type: rtype2type(typ)}
|
||||
}
|
||||
|
@ -238,6 +258,66 @@ func (e *Encoder) compileUint64(ctx *encodeCompileContext) (*opcode, error) {
|
|||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileIntString(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opIntString)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileInt8String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opInt8String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileInt16String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opInt16String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileInt32String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opInt32String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileInt64String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opInt64String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileUintString(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opUintString)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileUint8String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opUint8String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileUint16String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opUint16String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileUint32String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opUint32String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileUint64String(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opUint64String)
|
||||
ctx.incIndex()
|
||||
return code, nil
|
||||
}
|
||||
|
||||
func (e *Encoder) compileFloat32(ctx *encodeCompileContext) (*opcode, error) {
|
||||
code := newOpCode(ctx, opFloat32)
|
||||
ctx.incIndex()
|
||||
|
|
4928
encode_optype.go
4928
encode_optype.go
File diff suppressed because it is too large
Load Diff
80
encode_vm.go
80
encode_vm.go
|
@ -143,6 +143,86 @@ func (e *Encoder) run(ctx *encodeRuntimeContext, code *opcode) error {
|
|||
e.encodeUint64(e.ptrToUint64(load(ctxptr, code.idx)))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opIntString:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opIntStringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opInt8String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt8(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opInt8StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt8(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opInt16String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt16(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opInt16StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt16(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opInt32String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt32(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opInt32StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt32(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opInt64String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt64(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opInt64StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToInt64(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opUintString:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opUintStringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opUint8String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint8(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opUint8StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint8(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opUint16String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint16(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opUint16StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint16(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opUint32String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint32(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opUint32StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint32(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opUint64String:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint64(load(ctxptr, code.idx))))
|
||||
e.encodeByte(',')
|
||||
code = code.next
|
||||
case opUint64StringIndent:
|
||||
e.encodeString(fmt.Sprint(e.ptrToUint64(load(ctxptr, code.idx))))
|
||||
e.encodeBytes([]byte{',', '\n'})
|
||||
code = code.next
|
||||
case opFloat32:
|
||||
e.encodeFloat32(e.ptrToFloat32(load(ctxptr, code.idx)))
|
||||
e.encodeByte(',')
|
||||
|
|
Loading…
Reference in New Issue