Commit 89d4f00e authored by Kirill Smelkov's avatar Kirill Smelkov Committed by Kamil Kisiel

tests: Split TestDecode into main driver + worker that tests 1 particular pickle

We are going to modify the main tests table and for every case (e.g.
int(1) add pickles with various encoding that all represent the same
object when decoded. The main TestDecode driver will iterate over all
those input data and hand it over for particular testing to testDecode
worker.

Use t.Run for spawning each case - it is less typing (we do not need to
manually print test.name on errors, etc), and it allows to select which
subtests to run via e.g. `go test -run Decode/int`
parent 81e5b5e7
...@@ -4,6 +4,7 @@ import ( ...@@ -4,6 +4,7 @@ import (
"bytes" "bytes"
"encoding/hex" "encoding/hex"
"errors" "errors"
"fmt"
"io" "io"
"math/big" "math/big"
"reflect" "reflect"
...@@ -91,69 +92,79 @@ var tests = []struct { ...@@ -91,69 +92,79 @@ var tests = []struct {
{"STACK_GLOBAL opcode", "S'foo'\nS'bar'\n\x93.", Class{Module: "foo", Name: "bar"}}, {"STACK_GLOBAL opcode", "S'foo'\nS'bar'\n\x93.", Class{Module: "foo", Name: "bar"}},
} }
// TestDecode verifies ogórek decoder.
func TestDecode(t *testing.T) { func TestDecode(t *testing.T) {
for _, test := range tests { for _, test := range tests {
// decode(input) -> expected t.Run(fmt.Sprintf("%s/%q", test.name, test.input), func(t *testing.T) {
buf := bytes.NewBufferString(test.input) testDecode(t, test.expected, test.input)
})
}
}
// testDecode decodes input and verifies it is == object.
//
// It also verifies decoder robustness - via feeding it various kinds of
// corrupt data derived from input.
func testDecode(t *testing.T, object interface{}, input string) {
// decode(input) -> expected
buf := bytes.NewBufferString(input)
dec := NewDecoder(buf)
v, err := dec.Decode()
if err != nil {
t.Error(err)
}
if !reflect.DeepEqual(v, object) {
t.Errorf("decode:\nhave: %#v\nwant: %#v", v, object)
}
// decode more -> EOF
v, err = dec.Decode()
if !(v == nil && err == io.EOF) {
t.Errorf("decode: no EOF at end: v = %#v err = %#v", v, err)
}
// expected (= decoded(input)) -> encode -> decode = identity
buf.Reset()
enc := NewEncoder(buf)
err = enc.Encode(object)
if err != nil {
t.Errorf("encode(expected): %v", err)
} else {
dec := NewDecoder(buf) dec := NewDecoder(buf)
v, err := dec.Decode() v, err := dec.Decode()
if err != nil { if err != nil {
t.Error(err) t.Error(err)
} }
if !reflect.DeepEqual(v, test.expected) { if !reflect.DeepEqual(v, object) {
t.Errorf("%s: decode:\nhave: %#v\nwant: %#v", test.name, v, test.expected) t.Errorf("expected -> decode -> encode != identity\nhave: %#v\nwant: %#v", v, object)
}
// decode more -> EOF
v, err = dec.Decode()
if !(v == nil && err == io.EOF) {
t.Errorf("%s: decode: no EOF at end: v = %#v err = %#v", test.name, v, err)
}
// expected (= decoded(input)) -> encode -> decode = identity
buf.Reset()
enc := NewEncoder(buf)
err = enc.Encode(test.expected)
if err != nil {
t.Errorf("%s: encode(expected): %v", test.name, err)
} else {
dec := NewDecoder(buf)
v, err := dec.Decode()
if err != nil {
t.Error(err)
}
if !reflect.DeepEqual(v, test.expected) {
t.Errorf("%s: expected -> decode -> encode != identity\nhave: %#v\nwant: %#v", test.name, v, test.expected)
}
} }
}
// for truncated input io.ErrUnexpectedEOF must be returned // decode(truncated input) -> must return io.ErrUnexpectedEOF
for l := len(test.input) - 1; l > 0; l-- { for l := len(input) - 1; l > 0; l-- {
buf := bytes.NewBufferString(test.input[:l]) buf := bytes.NewBufferString(input[:l])
dec := NewDecoder(buf) dec := NewDecoder(buf)
//println(test.name, l) v, err := dec.Decode()
v, err := dec.Decode() if !(v == nil && err == io.ErrUnexpectedEOF) {
if !(v == nil && err == io.ErrUnexpectedEOF) { t.Errorf("no ErrUnexpectedEOF on [:%d] truncated stream: v = %#v err = %#v", l, v, err)
t.Errorf("%s: no ErrUnexpectedEOF on [:%d] truncated stream: v = %#v err = %#v", test.name, l, v, err)
}
} }
}
// by using input with omitted prefix we can test how code handles pickle stack overflow: // decode(input with omitted prefix) - tests how code handles pickle stack overflow:
// it must not panic // it must not panic.
for i := 0; i < len(test.input); i++ { for i := 0; i < len(input); i++ {
buf := bytes.NewBufferString(test.input[i:]) buf := bytes.NewBufferString(input[i:])
dec := NewDecoder(buf) dec := NewDecoder(buf)
func() { func() {
defer func() { defer func() {
if r := recover(); r != nil { if r := recover(); r != nil {
t.Errorf("%s: panic on input[%d:]: %v", test.name, i, r) t.Errorf("panic on input[%d:]: %v", i, r)
} }
}()
dec.Decode()
}() }()
} dec.Decode()
}()
} }
} }
......
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