Commit 7b2b8ede authored by Russ Cox's avatar Russ Cox

encoding/json: fix handling of null with ,string fields

Fixes #8587.

LGTM=bradfitz
R=golang-codereviews, bradfitz
CC=golang-codereviews, iant, r
https://golang.org/cl/152270044
parent 18172c42
...@@ -173,7 +173,6 @@ type decodeState struct { ...@@ -173,7 +173,6 @@ type decodeState struct {
scan scanner scan scanner
nextscan scanner // for calls to nextValue nextscan scanner // for calls to nextValue
savedError error savedError error
tempstr string // scratch space to avoid some allocations
useNumber bool useNumber bool
} }
...@@ -293,6 +292,32 @@ func (d *decodeState) value(v reflect.Value) { ...@@ -293,6 +292,32 @@ func (d *decodeState) value(v reflect.Value) {
} }
} }
type unquotedValue struct{}
// valueQuoted is like value but decodes a
// quoted string literal or literal null into an interface value.
// If it finds anything other than a quoted string literal or null,
// valueQuoted returns unquotedValue{}.
func (d *decodeState) valueQuoted() interface{} {
switch op := d.scanWhile(scanSkipSpace); op {
default:
d.error(errPhase)
case scanBeginArray:
d.array(reflect.Value{})
case scanBeginObject:
d.object(reflect.Value{})
case scanBeginLiteral:
switch v := d.literalInterface().(type) {
case nil, string:
return v
}
}
return unquotedValue{}
}
// indirect walks down v allocating pointers as needed, // indirect walks down v allocating pointers as needed,
// until it gets to a non-pointer. // until it gets to a non-pointer.
// if it encounters an Unmarshaler, indirect stops and returns that. // if it encounters an Unmarshaler, indirect stops and returns that.
...@@ -444,6 +469,8 @@ func (d *decodeState) array(v reflect.Value) { ...@@ -444,6 +469,8 @@ func (d *decodeState) array(v reflect.Value) {
} }
} }
var nullLiteral = []byte("null")
// object consumes an object from d.data[d.off-1:], decoding into the value v. // object consumes an object from d.data[d.off-1:], decoding into the value v.
// the first byte ('{') of the object has been read already. // the first byte ('{') of the object has been read already.
func (d *decodeState) object(v reflect.Value) { func (d *decodeState) object(v reflect.Value) {
...@@ -566,9 +593,14 @@ func (d *decodeState) object(v reflect.Value) { ...@@ -566,9 +593,14 @@ func (d *decodeState) object(v reflect.Value) {
// Read value. // Read value.
if destring { if destring {
d.value(reflect.ValueOf(&d.tempstr)) switch qv := d.valueQuoted().(type) {
d.literalStore([]byte(d.tempstr), subv, true) case nil:
d.tempstr = "" // Zero scratch space for successive values. d.literalStore(nullLiteral, subv, false)
case string:
d.literalStore([]byte(qv), subv, true)
default:
d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal unquoted value into %v", item, v.Type()))
}
} else { } else {
d.value(subv) d.value(subv)
} }
......
...@@ -1070,18 +1070,25 @@ func TestEmptyString(t *testing.T) { ...@@ -1070,18 +1070,25 @@ func TestEmptyString(t *testing.T) {
} }
} }
// Test that the returned error is non-nil when trying to unmarshal null string into int, for successive ,string option // Test that a null for ,string is not replaced with the previous quoted string (issue 7046).
// Issue 7046 // It should also not be an error (issue 2540, issue 8587).
func TestNullString(t *testing.T) { func TestNullString(t *testing.T) {
type T struct { type T struct {
A int `json:",string"` A int `json:",string"`
B int `json:",string"` B int `json:",string"`
C *int `json:",string"`
} }
data := []byte(`{"A": "1", "B": null}`) data := []byte(`{"A": "1", "B": null, "C": null}`)
var s T var s T
s.B = 1
s.C = new(int)
*s.C = 2
err := Unmarshal(data, &s) err := Unmarshal(data, &s)
if err == nil { if err != nil {
t.Fatalf("expected error; got %v", s) t.Fatalf("Unmarshal: %v")
}
if s.B != 1 || s.C != nil {
t.Fatalf("after Unmarshal, s.B=%d, s.C=%p, want 1, nil", s.B, s.C)
} }
} }
......
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