304 lines
6.7 KiB
Go
304 lines
6.7 KiB
Go
package assert
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/d5/tengo/compiler"
|
|
"github.com/d5/tengo/compiler/source"
|
|
"github.com/d5/tengo/compiler/token"
|
|
"github.com/d5/tengo/objects"
|
|
)
|
|
|
|
func NoError(t *testing.T, err error, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if err == nil {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "no error", err, msg...)
|
|
}
|
|
|
|
func Error(t *testing.T, err error, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if err != nil {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "error", err, msg...)
|
|
}
|
|
|
|
func Nil(t *testing.T, v interface{}, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if v == nil {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "nil", v, msg...)
|
|
}
|
|
|
|
func True(t *testing.T, v bool, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if v {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "true", v, msg...)
|
|
}
|
|
|
|
func False(t *testing.T, v bool, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if !v {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "false", v, msg...)
|
|
}
|
|
|
|
func NotNil(t *testing.T, v interface{}, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if v != nil {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, "not nil", v, msg...)
|
|
}
|
|
|
|
func IsType(t *testing.T, expected, actual interface{}, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if reflect.TypeOf(expected) == reflect.TypeOf(actual) {
|
|
return true
|
|
}
|
|
|
|
return failExpectedActual(t, reflect.TypeOf(expected), reflect.TypeOf(actual), msg...)
|
|
}
|
|
|
|
func Equal(t *testing.T, expected, actual interface{}, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
if expected == nil {
|
|
return Nil(t, actual, "expected nil, but got not nil")
|
|
}
|
|
if !NotNil(t, actual, "expected not nil, but got nil") {
|
|
return false
|
|
}
|
|
if !IsType(t, expected, actual) {
|
|
return false
|
|
}
|
|
|
|
switch expected := expected.(type) {
|
|
case int:
|
|
if expected != actual.(int) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case int64:
|
|
if expected != actual.(int64) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case float64:
|
|
if expected != actual.(float64) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case string:
|
|
if expected != actual.(string) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case []byte:
|
|
if bytes.Compare(expected, actual.([]byte)) != 0 {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case []int:
|
|
if !equalIntSlice(expected, actual.([]int)) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case bool:
|
|
if expected != actual.(bool) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case rune:
|
|
if expected != actual.(rune) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case compiler.Symbol:
|
|
if !equalSymbol(expected, actual.(compiler.Symbol)) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case source.Pos:
|
|
if expected != actual.(source.Pos) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case token.Token:
|
|
if expected != actual.(token.Token) {
|
|
return failExpectedActual(t, expected, actual, msg...)
|
|
}
|
|
case []objects.Object:
|
|
return equalObjectSlice(t, expected, actual.([]objects.Object))
|
|
case *objects.Int:
|
|
return Equal(t, expected.Value, actual.(*objects.Int).Value)
|
|
case *objects.Float:
|
|
return Equal(t, expected.Value, actual.(*objects.Float).Value)
|
|
case *objects.String:
|
|
return Equal(t, expected.Value, actual.(*objects.String).Value)
|
|
case *objects.Char:
|
|
return Equal(t, expected.Value, actual.(*objects.Char).Value)
|
|
case *objects.Bool:
|
|
return Equal(t, expected.Value, actual.(*objects.Bool).Value)
|
|
case *objects.ReturnValue:
|
|
return Equal(t, expected.Value, actual.(objects.ReturnValue).Value)
|
|
case *objects.Array:
|
|
return equalArray(t, expected, actual.(*objects.Array))
|
|
case *objects.Map:
|
|
return equalMap(t, expected, actual.(*objects.Map))
|
|
case *objects.CompiledFunction:
|
|
return equalCompiledFunction(t, expected, actual.(*objects.CompiledFunction))
|
|
case *objects.Closure:
|
|
return equalClosure(t, expected, actual.(*objects.Closure))
|
|
case *objects.Undefined:
|
|
return true
|
|
default:
|
|
panic(fmt.Errorf("type not implemented: %T", expected))
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func Fail(t *testing.T, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
t.Logf("\nError trace:\n\t%s\n%s", strings.Join(errorTrace(), "\n\t"), message(msg...))
|
|
|
|
t.Fail()
|
|
|
|
return false
|
|
}
|
|
|
|
func failExpectedActual(t *testing.T, expected, actual interface{}, msg ...interface{}) bool {
|
|
t.Helper()
|
|
|
|
var addMsg string
|
|
if len(msg) > 0 {
|
|
addMsg = "\nMessage: " + message(msg...)
|
|
}
|
|
|
|
t.Logf("\nError trace:\n\t%s\nExpected: %v\nActual: %v%s",
|
|
strings.Join(errorTrace(), "\n\t"),
|
|
expected, actual,
|
|
addMsg)
|
|
|
|
t.Fail()
|
|
|
|
return false
|
|
}
|
|
|
|
func message(formatArgs ...interface{}) string {
|
|
var format string
|
|
var args []interface{}
|
|
if len(formatArgs) > 0 {
|
|
format = formatArgs[0].(string)
|
|
}
|
|
if len(formatArgs) > 1 {
|
|
args = formatArgs[1:]
|
|
}
|
|
|
|
return fmt.Sprintf(format, args...)
|
|
}
|
|
|
|
func equalIntSlice(a, b []int) bool {
|
|
if len(a) != len(b) {
|
|
return false
|
|
}
|
|
|
|
for i := 0; i < len(a); i++ {
|
|
if a[i] != b[i] {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func equalSymbol(a, b compiler.Symbol) bool {
|
|
return a.Name == b.Name &&
|
|
a.Index == b.Index &&
|
|
a.Scope == b.Scope
|
|
}
|
|
|
|
func equalArray(t *testing.T, expected, actual objects.Object) bool {
|
|
expectedT := expected.(*objects.Array).Value
|
|
actualT := actual.(*objects.Array).Value
|
|
|
|
return equalObjectSlice(t, expectedT, actualT)
|
|
}
|
|
|
|
func equalObjectSlice(t *testing.T, expected, actual []objects.Object) bool {
|
|
// TODO: this test does not differentiate nil vs empty slice
|
|
|
|
if !Equal(t, len(expected), len(actual)) {
|
|
return false
|
|
}
|
|
|
|
for i := 0; i < len(expected); i++ {
|
|
if !Equal(t, expected[i], actual[i]) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func equalMap(t *testing.T, expected, actual objects.Object) bool {
|
|
expectedT := expected.(*objects.Map).Value
|
|
actualT := actual.(*objects.Map).Value
|
|
|
|
if !Equal(t, len(expectedT), len(actualT)) {
|
|
return false
|
|
}
|
|
|
|
for key, expectedVal := range expectedT {
|
|
actualVal := actualT[key]
|
|
|
|
if !Equal(t, expectedVal, actualVal) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func equalCompiledFunction(t *testing.T, expected, actual objects.Object) bool {
|
|
expectedT := expected.(*objects.CompiledFunction)
|
|
actualT := actual.(*objects.CompiledFunction)
|
|
|
|
return Equal(t, expectedT.Instructions, actualT.Instructions)
|
|
}
|
|
|
|
func equalClosure(t *testing.T, expected, actual objects.Object) bool {
|
|
expectedT := expected.(*objects.Closure)
|
|
actualT := actual.(*objects.Closure)
|
|
|
|
if !Equal(t, expectedT.Fn, actualT.Fn) {
|
|
return false
|
|
}
|
|
|
|
if !Equal(t, len(expectedT.Free), len(actualT.Free)) {
|
|
return false
|
|
}
|
|
|
|
for i := 0; i < len(expectedT.Free); i++ {
|
|
if !Equal(t, *expectedT.Free[i], *actualT.Free[i]) {
|
|
return false
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|