Skip to content

Commit f023623

Browse files
EHfivexiaokangwang
authored andcommitted
Fix: json.Reader: fill output bytes as much as possible
Fix JSON parsing state machine so Read([]byte) can accept byte array of any length. This fixes a infinite read with io.ReadFull().
1 parent ddc6312 commit f023623

File tree

2 files changed

+29
-12
lines changed

2 files changed

+29
-12
lines changed

infra/conf/json/reader.go

+27-11
Original file line numberDiff line numberDiff line change
@@ -27,18 +27,29 @@ const (
2727
type Reader struct {
2828
io.Reader
2929

30-
state State
31-
br *buf.BufferedReader
30+
state State
31+
pending []byte
32+
br *buf.BufferedReader
3233
}
3334

34-
// Read implements io.Reader.Read(). Buffer must be at least 3 bytes.
35+
// Read implements io.Reader.Read().
3536
func (v *Reader) Read(b []byte) (int, error) {
3637
if v.br == nil {
3738
v.br = &buf.BufferedReader{Reader: buf.NewReader(v.Reader)}
3839
}
3940

4041
p := b[:0]
41-
for len(p) < len(b)-2 {
42+
for len(p) < len(b) {
43+
if len(v.pending) > 0 {
44+
max := len(b) - len(p)
45+
if max > len(v.pending) {
46+
max = len(v.pending)
47+
}
48+
p = append(p, v.pending[:max]...)
49+
v.pending = v.pending[max:]
50+
continue
51+
}
52+
4253
x, err := v.br.ReadByte()
4354
if err != nil {
4455
if len(p) == 0 {
@@ -57,6 +68,7 @@ func (v *Reader) Read(b []byte) (int, error) {
5768
p = append(p, x)
5869
case '\\':
5970
v.state = StateEscape
71+
p = append(p, x)
6072
case '#':
6173
v.state = StateComment
6274
case '/':
@@ -65,7 +77,7 @@ func (v *Reader) Read(b []byte) (int, error) {
6577
p = append(p, x)
6678
}
6779
case StateEscape:
68-
p = append(p, '\\', x)
80+
p = append(p, x)
6981
v.state = StateContent
7082
case StateDoubleQuote:
7183
switch x {
@@ -74,11 +86,12 @@ func (v *Reader) Read(b []byte) (int, error) {
7486
p = append(p, x)
7587
case '\\':
7688
v.state = StateDoubleQuoteEscape
89+
p = append(p, x)
7790
default:
7891
p = append(p, x)
7992
}
8093
case StateDoubleQuoteEscape:
81-
p = append(p, '\\', x)
94+
p = append(p, x)
8295
v.state = StateDoubleQuote
8396
case StateSingleQuote:
8497
switch x {
@@ -87,16 +100,17 @@ func (v *Reader) Read(b []byte) (int, error) {
87100
p = append(p, x)
88101
case '\\':
89102
v.state = StateSingleQuoteEscape
103+
p = append(p, x)
90104
default:
91105
p = append(p, x)
92106
}
93107
case StateSingleQuoteEscape:
94-
p = append(p, '\\', x)
108+
p = append(p, x)
95109
v.state = StateSingleQuote
96110
case StateComment:
97111
if x == '\n' {
98112
v.state = StateContent
99-
p = append(p, '\n')
113+
p = append(p, x)
100114
}
101115
case StateSlash:
102116
switch x {
@@ -105,14 +119,16 @@ func (v *Reader) Read(b []byte) (int, error) {
105119
case '*':
106120
v.state = StateMultilineComment
107121
default:
108-
p = append(p, '/', x)
122+
v.state = StateContent
123+
v.pending = append(v.pending, x)
124+
p = append(p, '/')
109125
}
110126
case StateMultilineComment:
111127
switch x {
112128
case '*':
113129
v.state = StateMultilineCommentStar
114130
case '\n':
115-
p = append(p, '\n')
131+
p = append(p, x)
116132
}
117133
case StateMultilineCommentStar:
118134
switch x {
@@ -121,7 +137,7 @@ func (v *Reader) Read(b []byte) (int, error) {
121137
case '*':
122138
// Stay
123139
case '\n':
124-
p = append(p, '\n')
140+
p = append(p, x)
125141
default:
126142
v.state = StateMultilineComment
127143
}

infra/conf/json/reader_test.go

+2-1
Original file line numberDiff line numberDiff line change
@@ -62,14 +62,15 @@ func TestReader1(t *testing.T) {
6262
output string
6363
}
6464

65-
bufLen := 8
65+
bufLen := 1
6666

6767
data := []dataStruct{
6868
{"loooooooooooooooooooooooooooooooooooooooog", "loooooooooooooooooooooooooooooooooooooooog"},
6969
{`{"t": "\/testlooooooooooooooooooooooooooooong"}`, `{"t": "\/testlooooooooooooooooooooooooooooong"}`},
7070
{`{"t": "\/test"}`, `{"t": "\/test"}`},
7171
{`"\// fake comment"`, `"\// fake comment"`},
7272
{`"\/\/\/\/\/"`, `"\/\/\/\/\/"`},
73+
{`/test/test`, `/test/test`},
7374
}
7475

7576
for _, testCase := range data {

0 commit comments

Comments
 (0)