summaryrefslogtreecommitdiff
path: root/compat
diff options
context:
space:
mode:
authorLuke Shumaker <lukeshu@lukeshu.com>2023-02-17 19:21:37 -0700
committerLuke Shumaker <lukeshu@lukeshu.com>2023-02-19 14:51:04 -0700
commit49ee8be679add0bd3cf08a2669331b3be7a835f8 (patch)
treeb3d3f889ed25084fe33ed9e01554d6ca51104bb5 /compat
parent00187950437a10952b82353405e5ba4b4515fb29 (diff)
compat/json: Correctly handle syntax-error-in-decode
Diffstat (limited to 'compat')
-rw-r--r--compat/json/compat.go82
-rw-r--r--compat/json/compat_test.go78
2 files changed, 151 insertions, 9 deletions
diff --git a/compat/json/compat.go b/compat/json/compat.go
index 3a9bd6c..695c1a8 100644
--- a/compat/json/compat.go
+++ b/compat/json/compat.go
@@ -237,14 +237,14 @@ func Valid(data []byte) bool {
// Decode wrappers ///////////////////////////////////////////////////
-func convertDecodeError(err error) error {
+func convertDecodeError(err error, isUnmarshal bool) error {
if derr, ok := err.(*lowmemjson.DecodeError); ok {
switch terr := derr.Err.(type) {
case *lowmemjson.DecodeSyntaxError:
switch {
case errors.Is(terr.Err, io.EOF):
err = io.EOF
- case errors.Is(terr.Err, io.ErrUnexpectedEOF):
+ case errors.Is(terr.Err, io.ErrUnexpectedEOF) && isUnmarshal:
err = &SyntaxError{
msg: "unexpected end of JSON input",
Offset: terr.Offset,
@@ -284,13 +284,66 @@ func convertDecodeError(err error) error {
return err
}
+type decodeValidator struct{}
+
+func (*decodeValidator) DecodeJSON(r io.RuneScanner) error {
+ for {
+ if _, _, err := r.ReadRune(); err != nil {
+
+ if err == io.EOF {
+ return nil
+ }
+ return err
+ }
+ }
+}
+
+var _ lowmemjson.Decodable = (*decodeValidator)(nil)
+
func Unmarshal(data []byte, ptr any) error {
- return convertDecodeError(lowmemjson.NewDecoder(bytes.NewReader(data)).DecodeThenEOF(ptr))
+ if err := convertDecodeError(lowmemjson.NewDecoder(bytes.NewReader(data)).DecodeThenEOF(&decodeValidator{}), true); err != nil {
+ return err
+ }
+ if err := convertDecodeError(lowmemjson.NewDecoder(bytes.NewReader(data)).DecodeThenEOF(ptr), true); err != nil {
+ return err
+ }
+ return nil
+}
+
+type teeRuneScanner struct {
+ src io.RuneScanner
+ dst *bytes.Buffer
+ lastSize int
+}
+
+func (tee *teeRuneScanner) ReadRune() (r rune, size int, err error) {
+ r, size, err = tee.src.ReadRune()
+ if err == nil {
+ if _, err := tee.dst.WriteRune(r); err != nil {
+ return 0, 0, err
+ }
+ }
+
+ tee.lastSize = size
+ return
+}
+
+func (tee *teeRuneScanner) UnreadRune() error {
+ if tee.lastSize == 0 {
+ return lowmemjson.ErrInvalidUnreadRune
+ }
+ _ = tee.src.UnreadRune()
+ tee.dst.Truncate(tee.dst.Len() - tee.lastSize)
+ tee.lastSize = 0
+ return nil
}
type Decoder struct {
+ validatorBuf *bufio.Reader
+ validator *lowmemjson.Decoder
+
+ decoderBuf bytes.Buffer
*lowmemjson.Decoder
- buf *bufio.Reader
}
func NewDecoder(r io.Reader) *Decoder {
@@ -298,18 +351,29 @@ func NewDecoder(r io.Reader) *Decoder {
if !ok {
br = bufio.NewReader(r)
}
- return &Decoder{
- Decoder: lowmemjson.NewDecoder(br),
- buf: br,
+ ret := &Decoder{
+ validatorBuf: br,
}
+ ret.validator = lowmemjson.NewDecoder(&teeRuneScanner{
+ src: ret.validatorBuf,
+ dst: &ret.decoderBuf,
+ })
+ ret.Decoder = lowmemjson.NewDecoder(&ret.decoderBuf)
+ return ret
}
func (dec *Decoder) Decode(ptr any) error {
- return convertDecodeError(dec.Decoder.Decode(ptr))
+ if err := convertDecodeError(dec.validator.Decode(&decodeValidator{}), false); err != nil {
+ return err
+ }
+ if err := convertDecodeError(dec.Decoder.Decode(ptr), false); err != nil {
+ return err
+ }
+ return nil
}
func (dec *Decoder) Buffered() io.Reader {
- dat, _ := dec.buf.Peek(dec.buf.Buffered())
+ dat, _ := dec.validatorBuf.Peek(dec.validatorBuf.Buffered())
return bytes.NewReader(dat)
}
diff --git a/compat/json/compat_test.go b/compat/json/compat_test.go
index 29a8b37..df9d387 100644
--- a/compat/json/compat_test.go
+++ b/compat/json/compat_test.go
@@ -6,6 +6,8 @@ package json
import (
"bytes"
+ "reflect"
+ "strings"
"testing"
"github.com/stretchr/testify/assert"
@@ -161,3 +163,79 @@ func TestCompatMarshal(t *testing.T) {
})
}
}
+
+func TestCompatUnmarshal(t *testing.T) {
+ t.Parallel()
+ type testcase struct {
+ In string
+ InPtr any
+ ExpOut any
+ ExpErr string
+ }
+ testcases := map[string]testcase{
+ "empty-obj": {In: `{}`, ExpOut: map[string]any{}},
+ "partial-obj": {In: `{"foo":"bar",`, ExpOut: nil, ExpErr: `unexpected end of JSON input`},
+ "existing-obj": {In: `{"baz":"quz"}`, InPtr: &map[string]string{"foo": "bar"}, ExpOut: map[string]string{"foo": "bar", "baz": "quz"}},
+ "existing-obj-partial": {In: `{"baz":"quz"`, InPtr: &map[string]string{"foo": "bar"}, ExpOut: map[string]string{"foo": "bar"}, ExpErr: "unexpected end of JSON input"},
+ "empty-ary": {In: `[]`, ExpOut: []any{}},
+ "two-objs": {In: `{} {}`, ExpOut: nil, ExpErr: `invalid character '{' after top-level value`},
+ "two-numbers1": {In: `00`, ExpOut: nil, ExpErr: `invalid character '0' after top-level value`},
+ "two-numbers2": {In: `1 2`, ExpOut: nil, ExpErr: `invalid character '2' after top-level value`},
+ }
+ for tcName, tc := range testcases {
+ tc := tc
+ t.Run(tcName, func(t *testing.T) {
+ t.Parallel()
+ ptr := tc.InPtr
+ if ptr == nil {
+ var out any
+ ptr = &out
+ }
+ err := Unmarshal([]byte(tc.In), ptr)
+ assert.Equal(t, tc.ExpOut, reflect.ValueOf(ptr).Elem().Interface())
+ if tc.ExpErr == "" {
+ assert.NoError(t, err)
+ } else {
+ assert.EqualError(t, err, tc.ExpErr)
+ }
+ })
+ }
+}
+
+func TestCompatDecode(t *testing.T) {
+ t.Parallel()
+ type testcase struct {
+ In string
+ InPtr any
+ ExpOut any
+ ExpErr string
+ }
+ testcases := map[string]testcase{
+ "empty-obj": {In: `{}`, ExpOut: map[string]any{}},
+ "partial-obj": {In: `{"foo":"bar",`, ExpOut: nil, ExpErr: `unexpected EOF`},
+ "existing-obj": {In: `{"baz":"quz"}`, InPtr: &map[string]string{"foo": "bar"}, ExpOut: map[string]string{"foo": "bar", "baz": "quz"}},
+ "existing-obj-partial": {In: `{"baz":"quz"`, InPtr: &map[string]string{"foo": "bar"}, ExpOut: map[string]string{"foo": "bar"}, ExpErr: "unexpected EOF"},
+ "empty-ary": {In: `[]`, ExpOut: []any{}},
+ "two-objs": {In: `{} {}`, ExpOut: map[string]any{}},
+ "two-numbers1": {In: `00`, ExpOut: float64(0)},
+ "two-numbers2": {In: `1 2`, ExpOut: float64(1)},
+ }
+ for tcName, tc := range testcases {
+ tc := tc
+ t.Run(tcName, func(t *testing.T) {
+ t.Parallel()
+ ptr := tc.InPtr
+ if ptr == nil {
+ var out any
+ ptr = &out
+ }
+ err := NewDecoder(strings.NewReader(tc.In)).Decode(ptr)
+ assert.Equal(t, tc.ExpOut, reflect.ValueOf(ptr).Elem().Interface())
+ if tc.ExpErr == "" {
+ assert.NoError(t, err)
+ } else {
+ assert.EqualError(t, err, tc.ExpErr)
+ }
+ })
+ }
+}