forked from TrueCloudLab/neoneo-go
compiler: speed up boolean expression short-circuit test
The amount of tests done is comparable to all other tests in compiler package (~2k). After we moved to a new x/tools package this became a bottleneck. In this commit we reduce the amount of compiled files by combining multiple tests in a single file. Signed-off-by: Evgeniy Stratonikov <evgeniy@nspcc.ru>
This commit is contained in:
parent
9871dc8f5a
commit
548a6a06f1
2 changed files with 106 additions and 54 deletions
|
@ -1,9 +1,15 @@
|
||||||
package compiler_test
|
package compiler_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
var binaryExprTestCases = []testCase{
|
var binaryExprTestCases = []testCase{
|
||||||
|
@ -264,34 +270,55 @@ func TestBinaryExprs(t *testing.T) {
|
||||||
runTestCases(t, binaryExprTestCases)
|
runTestCases(t, binaryExprTestCases)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getBoolExprTestFunc(val bool, cond string) func(t *testing.T) {
|
func addBoolExprTestFunc(testCases []testCase, b *bytes.Buffer, val bool, cond string) []testCase {
|
||||||
srcTmpl := `package foo
|
n := len(testCases)
|
||||||
var s = "str"
|
b.WriteString(fmt.Sprintf(`
|
||||||
var v = 9
|
func F%d_expr() int {
|
||||||
var cond = %s
|
var cond%d = %s
|
||||||
func Main() int {
|
if cond%d { return 42 }
|
||||||
if %s {
|
|
||||||
return 42
|
|
||||||
} %s
|
|
||||||
return 17
|
return 17
|
||||||
%s
|
}
|
||||||
}`
|
func F%d_cond() int {
|
||||||
|
if %s { return 42 }
|
||||||
|
return 17
|
||||||
|
}
|
||||||
|
func F%d_else() int {
|
||||||
|
if %s { return 42 } else { return 17 }
|
||||||
|
}
|
||||||
|
`, n, n, cond, n, n, cond, n, cond))
|
||||||
|
|
||||||
res := big.NewInt(42)
|
res := big.NewInt(42)
|
||||||
if !val {
|
if !val {
|
||||||
res.SetInt64(17)
|
res.SetInt64(17)
|
||||||
}
|
}
|
||||||
return func(t *testing.T) {
|
|
||||||
|
return append(testCases, testCase{
|
||||||
|
name: cond,
|
||||||
|
result: res,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func runBooleanCases(t *testing.T, testCases []testCase, src string) {
|
||||||
|
ne, di, err := compiler.CompileWithOptions("file.go", strings.NewReader(src), nil)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
for i, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
t.Run("AsExpression", func(t *testing.T) {
|
t.Run("AsExpression", func(t *testing.T) {
|
||||||
src := fmt.Sprintf(srcTmpl, cond, "cond", "", "")
|
v := vm.New()
|
||||||
eval(t, src, res)
|
invokeMethod(t, fmt.Sprintf("F%d_expr", i), ne.Script, v, di)
|
||||||
|
runAndCheck(t, v, tc.result)
|
||||||
})
|
})
|
||||||
t.Run("InCondition", func(t *testing.T) {
|
t.Run("InCondition", func(t *testing.T) {
|
||||||
src := fmt.Sprintf(srcTmpl, "true", cond, "", "")
|
v := vm.New()
|
||||||
eval(t, src, res)
|
invokeMethod(t, fmt.Sprintf("F%d_cond", i), ne.Script, v, di)
|
||||||
|
runAndCheck(t, v, tc.result)
|
||||||
})
|
})
|
||||||
t.Run("InConditionWithElse", func(t *testing.T) {
|
t.Run("InConditionWithElse", func(t *testing.T) {
|
||||||
src := fmt.Sprintf(srcTmpl, "true", cond, " else {", "}")
|
v := vm.New()
|
||||||
eval(t, src, res)
|
invokeMethod(t, fmt.Sprintf("F%d_else", i), ne.Script, v, di)
|
||||||
|
runAndCheck(t, v, tc.result)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -299,24 +326,32 @@ func getBoolExprTestFunc(val bool, cond string) func(t *testing.T) {
|
||||||
// TestBooleanExprs enumerates a lot of possible combinations of boolean expressions
|
// TestBooleanExprs enumerates a lot of possible combinations of boolean expressions
|
||||||
// and tests if the result matches to that of Go.
|
// and tests if the result matches to that of Go.
|
||||||
func TestBooleanExprs(t *testing.T) {
|
func TestBooleanExprs(t *testing.T) {
|
||||||
t.Skip() // FIXME this test takes more than 2 minutes to complete
|
header := `package foo
|
||||||
|
var s = "str"
|
||||||
|
var v = 9
|
||||||
|
`
|
||||||
|
|
||||||
|
srcBuilder := bytes.NewBuffer([]byte(header))
|
||||||
|
|
||||||
|
var testCases []testCase
|
||||||
|
|
||||||
trueExpr := []string{"true", "v < 10", "v <= 9", "v > 8", "v >= 9", "v == 9", "v != 8", `s == "str"`}
|
trueExpr := []string{"true", "v < 10", "v <= 9", "v > 8", "v >= 9", "v == 9", "v != 8", `s == "str"`}
|
||||||
falseExpr := []string{"false", "v > 9", "v >= 10", "v < 9", "v <= 8", "v == 8", "v != 9", `s == "a"`}
|
falseExpr := []string{"false", "v > 9", "v >= 10", "v < 9", "v <= 8", "v == 8", "v != 9", `s == "a"`}
|
||||||
t.Run("Single", func(t *testing.T) {
|
t.Run("Single", func(t *testing.T) {
|
||||||
for _, s := range trueExpr {
|
for _, s := range trueExpr {
|
||||||
t.Run(s, getBoolExprTestFunc(true, s))
|
testCases = addBoolExprTestFunc(testCases, srcBuilder, true, s)
|
||||||
}
|
}
|
||||||
for _, s := range falseExpr {
|
for _, s := range falseExpr {
|
||||||
t.Run(s, getBoolExprTestFunc(false, s))
|
testCases = addBoolExprTestFunc(testCases, srcBuilder, false, s)
|
||||||
}
|
}
|
||||||
|
runBooleanCases(t, testCases, srcBuilder.String())
|
||||||
})
|
})
|
||||||
|
|
||||||
type arg struct {
|
type arg struct {
|
||||||
val bool
|
val bool
|
||||||
s string
|
s string
|
||||||
}
|
}
|
||||||
t.Run("Combine", func(t *testing.T) {
|
|
||||||
var double []arg
|
var double []arg
|
||||||
for _, e := range trueExpr {
|
for _, e := range trueExpr {
|
||||||
double = append(double, arg{true, e + " || false"})
|
double = append(double, arg{true, e + " || false"})
|
||||||
|
@ -326,9 +361,16 @@ func TestBooleanExprs(t *testing.T) {
|
||||||
double = append(double, arg{false, e + " && true"})
|
double = append(double, arg{false, e + " && true"})
|
||||||
double = append(double, arg{false, e + " || false"})
|
double = append(double, arg{false, e + " || false"})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
t.Run("Double", func(t *testing.T) {
|
||||||
|
testCases = testCases[:0]
|
||||||
|
srcBuilder.Reset()
|
||||||
|
srcBuilder.WriteString(header)
|
||||||
for i := range double {
|
for i := range double {
|
||||||
t.Run(double[i].s, getBoolExprTestFunc(double[i].val, double[i].s))
|
testCases = addBoolExprTestFunc(testCases, srcBuilder, double[i].val, double[i].s)
|
||||||
}
|
}
|
||||||
|
runBooleanCases(t, testCases, srcBuilder.String())
|
||||||
|
})
|
||||||
|
|
||||||
var triple []arg
|
var triple []arg
|
||||||
for _, a1 := range double {
|
for _, a1 := range double {
|
||||||
|
@ -337,8 +379,17 @@ func TestBooleanExprs(t *testing.T) {
|
||||||
triple = append(triple, arg{a1.val && a2.val, fmt.Sprintf("(%s) && (%s)", a1.s, a2.s)})
|
triple = append(triple, arg{a1.val && a2.val, fmt.Sprintf("(%s) && (%s)", a1.s, a2.s)})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for i := range triple {
|
|
||||||
t.Run(triple[i].s, getBoolExprTestFunc(triple[i].val, triple[i].s))
|
t.Run("Triple", func(t *testing.T) {
|
||||||
|
const step = 256 // empirically found value to make script less than 65536 in size
|
||||||
|
for start := 0; start < len(triple); start += step {
|
||||||
|
testCases = testCases[:0]
|
||||||
|
srcBuilder.Reset()
|
||||||
|
srcBuilder.WriteString(header)
|
||||||
|
for i := start; i < start+step && i < len(triple); i++ {
|
||||||
|
testCases = addBoolExprTestFunc(testCases, srcBuilder, triple[i].val, triple[i].s)
|
||||||
|
}
|
||||||
|
runBooleanCases(t, testCases, srcBuilder.String())
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,10 +34,14 @@ func runTestCases(t *testing.T, tcases []testCase) {
|
||||||
|
|
||||||
func eval(t *testing.T, src string, result interface{}) {
|
func eval(t *testing.T, src string, result interface{}) {
|
||||||
vm, _ := vmAndCompileInterop(t, src)
|
vm, _ := vmAndCompileInterop(t, src)
|
||||||
err := vm.Run()
|
runAndCheck(t, vm, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
func runAndCheck(t *testing.T, v *vm.VM, result interface{}) {
|
||||||
|
err := v.Run()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, 1, vm.Estack().Len(), "stack contains unexpected items")
|
assert.Equal(t, 1, v.Estack().Len(), "stack contains unexpected items")
|
||||||
assertResult(t, vm, result)
|
assertResult(t, v, result)
|
||||||
}
|
}
|
||||||
|
|
||||||
func evalWithArgs(t *testing.T, src string, op []byte, args []stackitem.Item, result interface{}) {
|
func evalWithArgs(t *testing.T, src string, op []byte, args []stackitem.Item, result interface{}) {
|
||||||
|
@ -48,10 +52,7 @@ func evalWithArgs(t *testing.T, src string, op []byte, args []stackitem.Item, re
|
||||||
if op != nil {
|
if op != nil {
|
||||||
vm.Estack().PushVal(op)
|
vm.Estack().PushVal(op)
|
||||||
}
|
}
|
||||||
err := vm.Run()
|
runAndCheck(t, vm, result)
|
||||||
require.NoError(t, err)
|
|
||||||
assert.Equal(t, 1, vm.Estack().Len(), "stack contains unexpected items")
|
|
||||||
assertResult(t, vm, result)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func assertResult(t *testing.T, vm *vm.VM, result interface{}) {
|
func assertResult(t *testing.T, vm *vm.VM, result interface{}) {
|
||||||
|
|
Loading…
Reference in a new issue