mirror of
https://github.com/chaisql/chai.git
synced 2025-10-06 16:18:14 +08:00

All new error handling code now rely on internal/errors package which provides a compilation time toggle that enables to capture stacktraces for easier debugging while developing. It also comes with a new testutil/assert package which replaces the require package when it comes to checking or comparing errors and printing the stack traces if needed. Finally, the test target of the Makefile uses the debug build tag by default. A testnodebug target is also provided for convenience and to make sure no tests are broken due to not having used the internal/errors or testutil/assert package. See #431 for more details
224 lines
6.1 KiB
Go
224 lines
6.1 KiB
Go
// Package encodingtest provides a test suite for testing codec implementations.
|
|
package encodingtest
|
|
|
|
import (
|
|
"bytes"
|
|
"testing"
|
|
|
|
"github.com/genjidb/genji/document"
|
|
"github.com/genjidb/genji/document/encoding"
|
|
"github.com/genjidb/genji/internal/testutil/assert"
|
|
"github.com/genjidb/genji/types"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
// TestCodec runs a list of tests on the given codec.
|
|
func TestCodec(t *testing.T, codecBuilder func() encoding.Codec) {
|
|
tests := []struct {
|
|
name string
|
|
test func(*testing.T, func() encoding.Codec)
|
|
}{
|
|
{"EncodeDecode", testEncodeDecode},
|
|
{"NewDocument", testDecodeDocument},
|
|
{"Document/GetByField", testDocumentGetByField},
|
|
{"Array/GetByIndex", testArrayGetByIndex},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
test.test(t, codecBuilder)
|
|
})
|
|
}
|
|
}
|
|
|
|
func testEncodeDecode(t *testing.T, codecBuilder func() encoding.Codec) {
|
|
userMapDoc, err := document.NewFromMap(map[string]interface{}{
|
|
"age": 10,
|
|
"name": "john",
|
|
})
|
|
assert.NoError(t, err)
|
|
|
|
addressMapDoc, err := document.NewFromMap(map[string]string{
|
|
"city": "Ajaccio",
|
|
"country": "France",
|
|
})
|
|
assert.NoError(t, err)
|
|
|
|
complexArray := document.NewValueBuffer().
|
|
Append(types.NewBoolValue(true)).
|
|
Append(types.NewIntegerValue(-40)).
|
|
Append(types.NewDoubleValue(-3.14)).
|
|
Append(types.NewDoubleValue(3)).
|
|
Append(types.NewBlobValue([]byte("blob"))).
|
|
Append(types.NewTextValue("hello")).
|
|
Append(types.NewDocumentValue(addressMapDoc)).
|
|
Append(types.NewArrayValue(document.NewValueBuffer().Append(types.NewIntegerValue(11))))
|
|
|
|
tests := []struct {
|
|
name string
|
|
d types.Document
|
|
expected string
|
|
}{
|
|
{
|
|
"document.FieldBuffer",
|
|
document.NewFieldBuffer().
|
|
Add("age", types.NewIntegerValue(10)).
|
|
Add("name", types.NewTextValue("john")),
|
|
`{"age": 10, "name": "john"}`,
|
|
},
|
|
{
|
|
"Map",
|
|
userMapDoc,
|
|
`{"age": 10, "name": "john"}`,
|
|
},
|
|
{
|
|
"Nested types.Document",
|
|
document.NewFieldBuffer().
|
|
Add("age", types.NewIntegerValue(10)).
|
|
Add("name", types.NewTextValue("john")).
|
|
Add("address", types.NewDocumentValue(addressMapDoc)).
|
|
Add("array", types.NewArrayValue(complexArray)),
|
|
`{"age": 10, "name": "john", "address": {"city": "Ajaccio", "country": "France"}, "array": [true, -40, -3.14, 3, "YmxvYg==", "hello", {"city": "Ajaccio", "country": "France"}, [11]]}`,
|
|
},
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
buf.Reset()
|
|
codec := codecBuilder()
|
|
err := codec.NewEncoder(&buf).EncodeDocument(test.d)
|
|
assert.NoError(t, err)
|
|
ok, err := types.IsEqual(types.NewDocumentValue(test.d), types.NewDocumentValue(codec.NewDecoder(buf.Bytes())))
|
|
assert.NoError(t, err)
|
|
require.True(t, ok)
|
|
data, err := document.MarshalJSON(codec.NewDecoder(buf.Bytes()))
|
|
assert.NoError(t, err)
|
|
require.JSONEq(t, test.expected, string(data))
|
|
})
|
|
}
|
|
}
|
|
|
|
func testDocumentGetByField(t *testing.T, codecBuilder func() encoding.Codec) {
|
|
codec := codecBuilder()
|
|
|
|
fb := document.NewFieldBuffer().
|
|
Add("a", types.NewIntegerValue(10)).
|
|
Add("b", types.NewNullValue()).
|
|
Add("c", types.NewTextValue("john"))
|
|
|
|
var buf bytes.Buffer
|
|
|
|
err := codec.NewEncoder(&buf).EncodeDocument(fb)
|
|
assert.NoError(t, err)
|
|
|
|
d := codec.NewDecoder(buf.Bytes())
|
|
|
|
v, err := d.GetByField("a")
|
|
assert.NoError(t, err)
|
|
|
|
require.Equal(t, types.NewIntegerValue(10), v)
|
|
|
|
v, err = d.GetByField("b")
|
|
assert.NoError(t, err)
|
|
require.Equal(t, types.NewNullValue(), v)
|
|
|
|
v, err = d.GetByField("c")
|
|
assert.NoError(t, err)
|
|
require.Equal(t, types.NewTextValue("john"), v)
|
|
|
|
v, err = d.GetByField("d")
|
|
assert.ErrorIs(t, err, document.ErrFieldNotFound)
|
|
}
|
|
|
|
func testArrayGetByIndex(t *testing.T, codecBuilder func() encoding.Codec) {
|
|
codec := codecBuilder()
|
|
|
|
arr := document.NewValueBuffer().
|
|
Append(types.NewIntegerValue(10)).
|
|
Append(types.NewNullValue()).
|
|
Append(types.NewTextValue("john"))
|
|
|
|
var buf bytes.Buffer
|
|
|
|
err := codec.NewEncoder(&buf).EncodeDocument(document.NewFieldBuffer().Add("a", types.NewArrayValue(arr)))
|
|
assert.NoError(t, err)
|
|
|
|
d := codec.NewDecoder(buf.Bytes())
|
|
v, err := d.GetByField("a")
|
|
assert.NoError(t, err)
|
|
|
|
require.Equal(t, types.ArrayValue, v.Type())
|
|
a := v.V().(types.Array)
|
|
v, err = a.GetByIndex(0)
|
|
assert.NoError(t, err)
|
|
|
|
require.Equal(t, types.NewIntegerValue(10), v)
|
|
|
|
v, err = a.GetByIndex(1)
|
|
assert.NoError(t, err)
|
|
require.Equal(t, types.NewNullValue(), v)
|
|
|
|
v, err = a.GetByIndex(2)
|
|
assert.NoError(t, err)
|
|
require.Equal(t, types.NewTextValue("john"), v)
|
|
|
|
_, err = a.GetByIndex(1000)
|
|
assert.ErrorIs(t, err, document.ErrValueNotFound)
|
|
}
|
|
|
|
func testDecodeDocument(t *testing.T, codecBuilder func() encoding.Codec) {
|
|
codec := codecBuilder()
|
|
|
|
mapDoc, err := document.NewFromMap(map[string]string{
|
|
"city": "Ajaccio",
|
|
"country": "France",
|
|
})
|
|
assert.NoError(t, err)
|
|
|
|
doc := document.NewFieldBuffer().
|
|
Add("age", types.NewIntegerValue(10)).
|
|
Add("name", types.NewTextValue("john")).
|
|
Add("address", types.NewDocumentValue(mapDoc))
|
|
|
|
var buf bytes.Buffer
|
|
|
|
enc := codec.NewEncoder(&buf)
|
|
defer enc.Close()
|
|
|
|
err = enc.EncodeDocument(doc)
|
|
assert.NoError(t, err)
|
|
|
|
ec := codec.NewDecoder(buf.Bytes())
|
|
v, err := ec.GetByField("age")
|
|
assert.NoError(t, err)
|
|
require.Equal(t, types.NewIntegerValue(10), v)
|
|
v, err = ec.GetByField("address")
|
|
assert.NoError(t, err)
|
|
expected, err := document.MarshalJSON(document.NewFieldBuffer().Add("address", types.NewDocumentValue(mapDoc)))
|
|
assert.NoError(t, err)
|
|
actual, err := document.MarshalJSON(document.NewFieldBuffer().Add("address", v))
|
|
assert.NoError(t, err)
|
|
require.JSONEq(t, string(expected), string(actual))
|
|
|
|
var i int
|
|
err = ec.Iterate(func(f string, v types.Value) error {
|
|
switch f {
|
|
case "age":
|
|
require.Equal(t, types.NewIntegerValue(10), v)
|
|
case "address":
|
|
expected, err := document.MarshalJSON(document.NewFieldBuffer().Add("address", types.NewDocumentValue(mapDoc)))
|
|
assert.NoError(t, err)
|
|
actual, err := document.MarshalJSON(document.NewFieldBuffer().Add(f, v))
|
|
assert.NoError(t, err)
|
|
require.JSONEq(t, string(expected), string(actual))
|
|
case "name":
|
|
require.Equal(t, types.NewTextValue("john"), v)
|
|
}
|
|
i++
|
|
return nil
|
|
})
|
|
assert.NoError(t, err)
|
|
require.Equal(t, 3, i)
|
|
}
|