2
0
Fork 0
mirror of https://github.com/ii64/sonic.git synced 2026-06-21 00:46:43 +08:00
sonic/decoder/stream_test.go
Yi Duan f9d033d6f8
feat: support streaming IO (#225)
* feat (decoder): support streaming IO

* opt: fine tune buffer size factors

* feat (encoder): support streaming IO

* doc: refactor format and add comments

* fix: io.EOF dead loop

* build: adjust CI files

* doc: update README.md
2022-05-24 10:38:43 +08:00

374 lines
No EOL
10 KiB
Go

/*
* Copyright 2021 ByteDance Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package decoder
import (
`encoding/json`
`io`
`io/ioutil`
`strings`
`testing`
jsoniter `github.com/json-iterator/go`
`github.com/stretchr/testify/assert`
`github.com/stretchr/testify/require`
)
var (
_Single_JSON = `{"aaaaa":"` + strings.Repeat("b",1024) + `"} { `
_Double_JSON = `{"aaaaa":"` + strings.Repeat("b",1024) + `"} {"11111":"` + strings.Repeat("2",1024) + `"}`
_Triple_JSON = `{"aaaaa":"` + strings.Repeat("b",1024) + `"}{ } {"11111":"` +
strings.Repeat("2",1024)+`"} b {}`
)
func TestDecodeSingle(t *testing.T) {
var str = _Single_JSON
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
var d1 = jsoniter.NewDecoder(r1)
var r2 = strings.NewReader(str)
var v2 map[string]interface{}
var d2 = NewStreamDecoder(r2)
require.Equal(t, d1.More(), d2.More())
es1 := d1.Decode(&v1)
ee1 := d2.Decode(&v2)
assert.Equal(t, es1, ee1)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es3 := d1.Decode(&v1)
assert.NotNil(t, es3)
ee3 := d2.Decode(&v2)
assert.NotNil(t, ee3)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
}
func TestDecodeMulti(t *testing.T) {
var str = _Triple_JSON
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
var d1 = jsoniter.NewDecoder(r1)
var r2 = strings.NewReader(str)
var v2 map[string]interface{}
var d2 = NewStreamDecoder(r2)
require.Equal(t, d1.More(), d2.More())
es1 := d1.Decode(&v1)
ee1 := d2.Decode(&v2)
assert.Equal(t, es1, ee1)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es4 := d1.Decode(&v1)
ee4 := d2.Decode(&v2)
assert.Equal(t, es4, ee4)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es2 := d1.Decode(&v1)
ee2 := d2.Decode(&v2)
assert.Equal(t, es2, ee2)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
// fmt.Printf("v:%#v\n", v1)
require.Equal(t, d1.More(), d2.More())
es3 := d1.Decode(&v1)
assert.NotNil(t, es3)
ee3 := d2.Decode(&v2)
assert.NotNil(t, ee3)
require.Equal(t, d1.More(), d2.More())
es5 := d1.Decode(&v1)
assert.NotNil(t, es5)
ee5 := d2.Decode(&v2)
assert.NotNil(t, ee5)
}
type HaltReader struct {
halts map[int]bool
buf string
p int
}
func NewHaltReader(buf string, halts map[int]bool) *HaltReader {
return &HaltReader{
halts: halts,
buf: buf,
p: 0,
}
}
func (self *HaltReader) Read(p []byte) (int, error) {
t := 0
for ; t < len(p); {
if self.p >= len(self.buf) {
return t, io.EOF
}
if b, ok := self.halts[self.p]; b {
self.halts[self.p] = false
return t, nil
} else if ok {
delete(self.halts, self.p)
return 0, nil
}
p[t] = self.buf[self.p]
self.p++
t++
}
return t, nil
}
func (self *HaltReader) Reset(buf string) {
self.p = 0
self.buf = buf
}
var testHalts = func () map[int]bool {
return map[int]bool{
1: true,
10:true,
20: true}
}
func TestDecodeHalt(t *testing.T) {
var str = _Triple_JSON
var r1 = NewHaltReader(str, testHalts())
var r2 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
var v2 map[string]interface{}
var d1 = jsoniter.NewDecoder(r1)
var d2 = NewStreamDecoder(r2)
require.Equal(t, d1.More(), d2.More())
err1 := d1.Decode(&v1)
err2 := d2.Decode(&v2)
assert.Equal(t, err1, err2)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es4 := d1.Decode(&v1)
ee4 := d2.Decode(&v2)
assert.Equal(t, es4, ee4)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es2 := d1.Decode(&v1)
ee2 := d2.Decode(&v2)
assert.Equal(t, es2, ee2)
assert.Equal(t, v1, v2)
// assert.Equal(t, d1.InputOffset(), d2.InputOffset())
require.Equal(t, d1.More(), d2.More())
es3 := d1.Decode(&v1)
assert.NotNil(t, es3)
ee3 := d2.Decode(&v2)
assert.NotNil(t, ee3)
require.Equal(t, d1.More(), d2.More())
es5 := d1.Decode(&v1)
assert.NotNil(t, es5)
ee5 := d2.Decode(&v2)
assert.NotNil(t, ee5)
}
func TestBuffered(t *testing.T) {
var str = _Triple_JSON
var r1 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
var d1 = json.NewDecoder(r1)
require.Nil(t, d1.Decode(&v1))
var r2 = NewHaltReader(str, testHalts())
var v2 map[string]interface{}
var d2 = NewStreamDecoder(r2)
require.Nil(t, d2.Decode(&v2))
left1, err1 := ioutil.ReadAll(d1.Buffered())
require.Nil(t, err1)
left2, err2 := ioutil.ReadAll(d2.Buffered())
require.Nil(t, err2)
require.Equal(t, d1.InputOffset(), d2.InputOffset())
min := len(left1)
if min > len(left2) {
min = len(left2)
}
require.Equal(t, left1[:min], left2[:min])
es4 := d1.Decode(&v1)
ee4 := d2.Decode(&v2)
assert.Equal(t, es4, ee4)
assert.Equal(t, d1.InputOffset(), d2.InputOffset())
es2 := d1.Decode(&v1)
ee2 := d2.Decode(&v2)
assert.Equal(t, es2, ee2)
assert.Equal(t, d1.InputOffset(), d2.InputOffset())
}
func TestMore(t *testing.T) {
var str = _Triple_JSON
var r2 = NewHaltReader(str, testHalts())
var v2 map[string]interface{}
var d2 = NewStreamDecoder(r2)
var r1 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
var d1 = jsoniter.NewDecoder(r1)
require.Nil(t, d1.Decode(&v1))
require.Nil(t, d2.Decode(&v2))
require.Equal(t, d1.More(), d2.More())
es4 := d1.Decode(&v1)
ee4 := d2.Decode(&v2)
assert.Equal(t, es4, ee4)
assert.Equal(t, v1, v2)
require.Equal(t, d1.More(), d2.More())
es2 := d1.Decode(&v1)
ee2 := d2.Decode(&v2)
assert.Equal(t, es2, ee2)
assert.Equal(t, v1, v2)
require.Equal(t, d1.More(), d2.More())
es3 := d1.Decode(&v1)
assert.NotNil(t, es3)
ee3 := d2.Decode(&v2)
assert.NotNil(t, ee3)
require.Equal(t, d1.More(), d2.More())
es5 := d1.Decode(&v1)
assert.NotNil(t, es5)
ee5 := d2.Decode(&v2)
assert.NotNil(t, ee5)
require.Equal(t, d1.More(), d2.More())
}
func BenchmarkDecodeStream_Std(b *testing.B) {
b.Run("single", func (b *testing.B) {
var str = _Single_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := json.NewDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("double", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := json.NewDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("halt", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
dc := json.NewDecoder(r1)
_ = dc.Decode(&v1)
}
})
}
func BenchmarkDecodeStream_Jsoniter(b *testing.B) {
b.Run("single", func (b *testing.B) {
var str = _Single_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := jsoniter.NewDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("double", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := jsoniter.NewDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("halt", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
dc := jsoniter.NewDecoder(r1)
_ = dc.Decode(&v1)
}
})
}
// func BenchmarkDecodeError_Sonic(b *testing.B) {
// var str = `\b测试1234`
// for i:=0; i<b.N; i++ {
// var v1 map[string]interface{}
// _ = NewDecoder(str).Decode(&v1)
// }
// }
func BenchmarkDecodeStream_Sonic(b *testing.B) {
b.Run("single", func (b *testing.B) {
var str = _Single_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := NewStreamDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("double", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = strings.NewReader(str)
var v1 map[string]interface{}
dc := NewStreamDecoder(r1)
_ = dc.Decode(&v1)
_ = dc.Decode(&v1)
}
})
b.Run("halt", func (b *testing.B) {
var str = _Double_JSON
for i:=0; i<b.N; i++ {
var r1 = NewHaltReader(str, testHalts())
var v1 map[string]interface{}
dc := NewStreamDecoder(r1)
_ = dc.Decode(&v1)
}
})
}