Commit a257ffb1 authored by Alex Plugaru's avatar Alex Plugaru Committed by Brad Fitzpatrick

encoding/json: add UnmarshalTypeError.Offset

Fixes #9693

Change-Id: Ibf07199729bfc883b2a7e051cafd98185f912acd
Reviewed-on: https://go-review.googlesource.com/3283Reviewed-by: default avatarRuss Cox <rsc@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent bd043d86
...@@ -90,8 +90,9 @@ type Unmarshaler interface { ...@@ -90,8 +90,9 @@ type Unmarshaler interface {
// An UnmarshalTypeError describes a JSON value that was // An UnmarshalTypeError describes a JSON value that was
// not appropriate for a value of a specific Go type. // not appropriate for a value of a specific Go type.
type UnmarshalTypeError struct { type UnmarshalTypeError struct {
Value string // description of JSON value - "bool", "array", "number -5" Value string // description of JSON value - "bool", "array", "number -5"
Type reflect.Type // type of Go value it could not be assigned to Type reflect.Type // type of Go value it could not be assigned to
Offset int64 // error occurred after reading Offset bytes
} }
func (e *UnmarshalTypeError) Error() string { func (e *UnmarshalTypeError) Error() string {
...@@ -377,7 +378,7 @@ func (d *decodeState) array(v reflect.Value) { ...@@ -377,7 +378,7 @@ func (d *decodeState) array(v reflect.Value) {
return return
} }
if ut != nil { if ut != nil {
d.saveError(&UnmarshalTypeError{"array", v.Type()}) d.saveError(&UnmarshalTypeError{"array", v.Type(), int64(d.off)})
d.off-- d.off--
d.next() d.next()
return return
...@@ -396,7 +397,7 @@ func (d *decodeState) array(v reflect.Value) { ...@@ -396,7 +397,7 @@ func (d *decodeState) array(v reflect.Value) {
// Otherwise it's invalid. // Otherwise it's invalid.
fallthrough fallthrough
default: default:
d.saveError(&UnmarshalTypeError{"array", v.Type()}) d.saveError(&UnmarshalTypeError{"array", v.Type(), int64(d.off)})
d.off-- d.off--
d.next() d.next()
return return
...@@ -485,7 +486,7 @@ func (d *decodeState) object(v reflect.Value) { ...@@ -485,7 +486,7 @@ func (d *decodeState) object(v reflect.Value) {
return return
} }
if ut != nil { if ut != nil {
d.saveError(&UnmarshalTypeError{"object", v.Type()}) d.saveError(&UnmarshalTypeError{"object", v.Type(), int64(d.off)})
d.off-- d.off--
d.next() // skip over { } in input d.next() // skip over { } in input
return return
...@@ -504,7 +505,7 @@ func (d *decodeState) object(v reflect.Value) { ...@@ -504,7 +505,7 @@ func (d *decodeState) object(v reflect.Value) {
// map must have string kind // map must have string kind
t := v.Type() t := v.Type()
if t.Key().Kind() != reflect.String { if t.Key().Kind() != reflect.String {
d.saveError(&UnmarshalTypeError{"object", v.Type()}) d.saveError(&UnmarshalTypeError{"object", v.Type(), int64(d.off)})
d.off-- d.off--
d.next() // skip over { } in input d.next() // skip over { } in input
return return
...@@ -515,7 +516,7 @@ func (d *decodeState) object(v reflect.Value) { ...@@ -515,7 +516,7 @@ func (d *decodeState) object(v reflect.Value) {
case reflect.Struct: case reflect.Struct:
default: default:
d.saveError(&UnmarshalTypeError{"object", v.Type()}) d.saveError(&UnmarshalTypeError{"object", v.Type(), int64(d.off)})
d.off-- d.off--
d.next() // skip over { } in input d.next() // skip over { } in input
return return
...@@ -646,7 +647,7 @@ func (d *decodeState) convertNumber(s string) (interface{}, error) { ...@@ -646,7 +647,7 @@ func (d *decodeState) convertNumber(s string) (interface{}, error) {
} }
f, err := strconv.ParseFloat(s, 64) f, err := strconv.ParseFloat(s, 64)
if err != nil { if err != nil {
return nil, &UnmarshalTypeError{"number " + s, reflect.TypeOf(0.0)} return nil, &UnmarshalTypeError{"number " + s, reflect.TypeOf(0.0), int64(d.off)}
} }
return f, nil return f, nil
} }
...@@ -679,7 +680,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -679,7 +680,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
if fromQuoted { if fromQuoted {
d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()))
} else { } else {
d.saveError(&UnmarshalTypeError{"string", v.Type()}) d.saveError(&UnmarshalTypeError{"string", v.Type(), int64(d.off)})
} }
} }
s, ok := unquoteBytes(item) s, ok := unquoteBytes(item)
...@@ -713,7 +714,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -713,7 +714,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
if fromQuoted { if fromQuoted {
d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()))
} else { } else {
d.saveError(&UnmarshalTypeError{"bool", v.Type()}) d.saveError(&UnmarshalTypeError{"bool", v.Type(), int64(d.off)})
} }
case reflect.Bool: case reflect.Bool:
v.SetBool(value) v.SetBool(value)
...@@ -721,7 +722,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -721,7 +722,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
if v.NumMethod() == 0 { if v.NumMethod() == 0 {
v.Set(reflect.ValueOf(value)) v.Set(reflect.ValueOf(value))
} else { } else {
d.saveError(&UnmarshalTypeError{"bool", v.Type()}) d.saveError(&UnmarshalTypeError{"bool", v.Type(), int64(d.off)})
} }
} }
...@@ -736,10 +737,10 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -736,10 +737,10 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
} }
switch v.Kind() { switch v.Kind() {
default: default:
d.saveError(&UnmarshalTypeError{"string", v.Type()}) d.saveError(&UnmarshalTypeError{"string", v.Type(), int64(d.off)})
case reflect.Slice: case reflect.Slice:
if v.Type() != byteSliceType { if v.Type() != byteSliceType {
d.saveError(&UnmarshalTypeError{"string", v.Type()}) d.saveError(&UnmarshalTypeError{"string", v.Type(), int64(d.off)})
break break
} }
b := make([]byte, base64.StdEncoding.DecodedLen(len(s))) b := make([]byte, base64.StdEncoding.DecodedLen(len(s)))
...@@ -755,7 +756,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -755,7 +756,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
if v.NumMethod() == 0 { if v.NumMethod() == 0 {
v.Set(reflect.ValueOf(string(s))) v.Set(reflect.ValueOf(string(s)))
} else { } else {
d.saveError(&UnmarshalTypeError{"string", v.Type()}) d.saveError(&UnmarshalTypeError{"string", v.Type(), int64(d.off)})
} }
} }
...@@ -777,7 +778,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -777,7 +778,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
if fromQuoted { if fromQuoted {
d.error(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) d.error(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()))
} else { } else {
d.error(&UnmarshalTypeError{"number", v.Type()}) d.error(&UnmarshalTypeError{"number", v.Type(), int64(d.off)})
} }
case reflect.Interface: case reflect.Interface:
n, err := d.convertNumber(s) n, err := d.convertNumber(s)
...@@ -786,7 +787,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -786,7 +787,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
break break
} }
if v.NumMethod() != 0 { if v.NumMethod() != 0 {
d.saveError(&UnmarshalTypeError{"number", v.Type()}) d.saveError(&UnmarshalTypeError{"number", v.Type(), int64(d.off)})
break break
} }
v.Set(reflect.ValueOf(n)) v.Set(reflect.ValueOf(n))
...@@ -794,7 +795,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -794,7 +795,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
n, err := strconv.ParseInt(s, 10, 64) n, err := strconv.ParseInt(s, 10, 64)
if err != nil || v.OverflowInt(n) { if err != nil || v.OverflowInt(n) {
d.saveError(&UnmarshalTypeError{"number " + s, v.Type()}) d.saveError(&UnmarshalTypeError{"number " + s, v.Type(), int64(d.off)})
break break
} }
v.SetInt(n) v.SetInt(n)
...@@ -802,7 +803,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -802,7 +803,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
n, err := strconv.ParseUint(s, 10, 64) n, err := strconv.ParseUint(s, 10, 64)
if err != nil || v.OverflowUint(n) { if err != nil || v.OverflowUint(n) {
d.saveError(&UnmarshalTypeError{"number " + s, v.Type()}) d.saveError(&UnmarshalTypeError{"number " + s, v.Type(), int64(d.off)})
break break
} }
v.SetUint(n) v.SetUint(n)
...@@ -810,7 +811,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool ...@@ -810,7 +811,7 @@ func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool
case reflect.Float32, reflect.Float64: case reflect.Float32, reflect.Float64:
n, err := strconv.ParseFloat(s, v.Type().Bits()) n, err := strconv.ParseFloat(s, v.Type().Bits())
if err != nil || v.OverflowFloat(n) { if err != nil || v.OverflowFloat(n) {
d.saveError(&UnmarshalTypeError{"number " + s, v.Type()}) d.saveError(&UnmarshalTypeError{"number " + s, v.Type(), int64(d.off)})
break break
} }
v.SetFloat(n) v.SetFloat(n)
......
...@@ -231,7 +231,7 @@ var unmarshalTests = []unmarshalTest{ ...@@ -231,7 +231,7 @@ var unmarshalTests = []unmarshalTest{
{in: `"g-clef: \uD834\uDD1E"`, ptr: new(string), out: "g-clef: \U0001D11E"}, {in: `"g-clef: \uD834\uDD1E"`, ptr: new(string), out: "g-clef: \U0001D11E"},
{in: `"invalid: \uD834x\uDD1E"`, ptr: new(string), out: "invalid: \uFFFDx\uFFFD"}, {in: `"invalid: \uD834x\uDD1E"`, ptr: new(string), out: "invalid: \uFFFDx\uFFFD"},
{in: "null", ptr: new(interface{}), out: nil}, {in: "null", ptr: new(interface{}), out: nil},
{in: `{"X": [1,2,3], "Y": 4}`, ptr: new(T), out: T{Y: 4}, err: &UnmarshalTypeError{"array", reflect.TypeOf("")}}, {in: `{"X": [1,2,3], "Y": 4}`, ptr: new(T), out: T{Y: 4}, err: &UnmarshalTypeError{"array", reflect.TypeOf(""), 7}},
{in: `{"x": 1}`, ptr: new(tx), out: tx{}}, {in: `{"x": 1}`, ptr: new(tx), out: tx{}},
{in: `{"F1":1,"F2":2,"F3":3}`, ptr: new(V), out: V{F1: float64(1), F2: int32(2), F3: Number("3")}}, {in: `{"F1":1,"F2":2,"F3":3}`, ptr: new(V), out: V{F1: float64(1), F2: int32(2), F3: Number("3")}},
{in: `{"F1":1,"F2":2,"F3":3}`, ptr: new(V), out: V{F1: Number("1"), F2: int32(2), F3: Number("3")}, useNumber: true}, {in: `{"F1":1,"F2":2,"F3":3}`, ptr: new(V), out: V{F1: Number("1"), F2: int32(2), F3: Number("3")}, useNumber: true},
...@@ -411,7 +411,7 @@ var unmarshalTests = []unmarshalTest{ ...@@ -411,7 +411,7 @@ var unmarshalTests = []unmarshalTest{
{ {
in: `{"2009-11-10T23:00:00Z": "hello world"}`, in: `{"2009-11-10T23:00:00Z": "hello world"}`,
ptr: &map[time.Time]string{}, ptr: &map[time.Time]string{},
err: &UnmarshalTypeError{"object", reflect.TypeOf(map[time.Time]string{})}, err: &UnmarshalTypeError{"object", reflect.TypeOf(map[time.Time]string{}), 1},
}, },
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment