Files
gf/test/gtest/gtest_util.go

372 lines
10 KiB
Go
Raw Permalink Normal View History

2021-01-17 21:46:25 +08:00
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
2020-03-19 22:56:12 +08:00
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
package gtest
import (
"fmt"
2021-09-15 21:17:45 +08:00
"io/ioutil"
2020-03-19 22:56:12 +08:00
"os"
2021-09-15 21:17:45 +08:00
"path/filepath"
2020-03-19 22:56:12 +08:00
"reflect"
"testing"
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/debug/gdebug"
2021-11-13 23:34:16 +08:00
"github.com/gogf/gf/v2/internal/empty"
2021-10-11 21:41:56 +08:00
"github.com/gogf/gf/v2/util/gconv"
2020-03-19 22:56:12 +08:00
)
const (
pathFilterKey = "/test/gtest/gtest"
2020-03-19 22:56:12 +08:00
)
2021-09-15 21:17:45 +08:00
// C creates a unit testing case.
// The parameter `t` is the pointer to testing.T of stdlib (*testing.T).
// The parameter `f` is the closure function for unit testing case.
2020-03-19 22:56:12 +08:00
func C(t *testing.T, f func(t *T)) {
defer func() {
if err := recover(); err != nil {
2022-02-22 10:43:31 +08:00
_, _ = fmt.Fprintf(os.Stderr, "%v\n%s", err, gdebug.StackWithFilter([]string{pathFilterKey}))
2020-03-19 22:56:12 +08:00
t.Fail()
}
}()
f(&T{t})
}
// Assert checks `value` and `expect` EQUAL.
2020-03-19 22:56:12 +08:00
func Assert(value, expect interface{}) {
rvExpect := reflect.ValueOf(expect)
if empty.IsNil(value) {
2020-03-19 22:56:12 +08:00
value = nil
}
if rvExpect.Kind() == reflect.Map {
if err := compareMap(value, expect); err != nil {
panic(err)
}
return
}
var (
strValue = gconv.String(value)
strExpect = gconv.String(expect)
)
2020-03-19 22:56:12 +08:00
if strValue != strExpect {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v == %v`, strValue, strExpect))
}
}
// AssertEQ checks `value` and `expect` EQUAL, including their TYPES.
2020-03-19 22:56:12 +08:00
func AssertEQ(value, expect interface{}) {
// Value assert.
rvExpect := reflect.ValueOf(expect)
if empty.IsNil(value) {
2020-03-19 22:56:12 +08:00
value = nil
}
if rvExpect.Kind() == reflect.Map {
if err := compareMap(value, expect); err != nil {
panic(err)
}
return
}
strValue := gconv.String(value)
strExpect := gconv.String(expect)
if strValue != strExpect {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v == %v`, strValue, strExpect))
}
// Type assert.
t1 := reflect.TypeOf(value)
t2 := reflect.TypeOf(expect)
if t1 != t2 {
panic(fmt.Sprintf(`[ASSERT] EXPECT TYPE %v[%v] == %v[%v]`, strValue, t1, strExpect, t2))
}
}
// AssertNE checks `value` and `expect` NOT EQUAL.
2020-03-19 22:56:12 +08:00
func AssertNE(value, expect interface{}) {
rvExpect := reflect.ValueOf(expect)
if empty.IsNil(value) {
2020-03-19 22:56:12 +08:00
value = nil
}
if rvExpect.Kind() == reflect.Map {
if err := compareMap(value, expect); err == nil {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v != %v`, value, expect))
}
return
}
var (
strValue = gconv.String(value)
strExpect = gconv.String(expect)
)
2020-03-19 22:56:12 +08:00
if strValue == strExpect {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v != %v`, strValue, strExpect))
}
}
// AssertNQ checks `value` and `expect` NOT EQUAL, including their TYPES.
func AssertNQ(value, expect interface{}) {
// Type assert.
t1 := reflect.TypeOf(value)
t2 := reflect.TypeOf(expect)
if t1 == t2 {
panic(
fmt.Sprintf(
`[ASSERT] EXPECT TYPE %v[%v] != %v[%v]`,
gconv.String(value), t1, gconv.String(expect), t2,
),
)
}
// Value assert.
AssertNE(value, expect)
}
// AssertGT checks `value` is GREATER THAN `expect`.
2020-03-19 22:56:12 +08:00
// Notice that, only string, integer and float types can be compared by AssertGT,
// others are invalid.
func AssertGT(value, expect interface{}) {
passed := false
switch reflect.ValueOf(expect).Kind() {
case reflect.String:
passed = gconv.String(value) > gconv.String(expect)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
passed = gconv.Int(value) > gconv.Int(expect)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
passed = gconv.Uint(value) > gconv.Uint(expect)
case reflect.Float32, reflect.Float64:
passed = gconv.Float64(value) > gconv.Float64(expect)
}
if !passed {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v > %v`, value, expect))
}
}
// AssertGE checks `value` is GREATER OR EQUAL THAN `expect`.
2020-03-19 22:56:12 +08:00
// Notice that, only string, integer and float types can be compared by AssertGTE,
// others are invalid.
func AssertGE(value, expect interface{}) {
passed := false
switch reflect.ValueOf(expect).Kind() {
case reflect.String:
passed = gconv.String(value) >= gconv.String(expect)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
passed = gconv.Int64(value) >= gconv.Int64(expect)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
passed = gconv.Uint64(value) >= gconv.Uint64(expect)
case reflect.Float32, reflect.Float64:
passed = gconv.Float64(value) >= gconv.Float64(expect)
}
if !passed {
panic(fmt.Sprintf(
`[ASSERT] EXPECT %v(%v) >= %v(%v)`,
value, reflect.ValueOf(value).Kind(),
expect, reflect.ValueOf(expect).Kind(),
))
}
}
// AssertLT checks `value` is LESS EQUAL THAN `expect`.
2020-03-19 22:56:12 +08:00
// Notice that, only string, integer and float types can be compared by AssertLT,
// others are invalid.
func AssertLT(value, expect interface{}) {
passed := false
switch reflect.ValueOf(expect).Kind() {
case reflect.String:
passed = gconv.String(value) < gconv.String(expect)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
passed = gconv.Int(value) < gconv.Int(expect)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
passed = gconv.Uint(value) < gconv.Uint(expect)
case reflect.Float32, reflect.Float64:
passed = gconv.Float64(value) < gconv.Float64(expect)
}
if !passed {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v < %v`, value, expect))
}
}
// AssertLE checks `value` is LESS OR EQUAL THAN `expect`.
2020-03-19 22:56:12 +08:00
// Notice that, only string, integer and float types can be compared by AssertLTE,
// others are invalid.
func AssertLE(value, expect interface{}) {
passed := false
switch reflect.ValueOf(expect).Kind() {
case reflect.String:
passed = gconv.String(value) <= gconv.String(expect)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
passed = gconv.Int(value) <= gconv.Int(expect)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
passed = gconv.Uint(value) <= gconv.Uint(expect)
case reflect.Float32, reflect.Float64:
passed = gconv.Float64(value) <= gconv.Float64(expect)
}
if !passed {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v <= %v`, value, expect))
}
}
// AssertIN checks `value` is IN `expect`.
// The `expect` should be a slice,
// but the `value` can be a slice or a basic type variable.
2020-03-19 22:56:12 +08:00
// TODO map support.
func AssertIN(value, expect interface{}) {
var (
passed = true
expectKind = reflect.ValueOf(expect).Kind()
)
2020-03-19 22:56:12 +08:00
switch expectKind {
case reflect.Slice, reflect.Array:
expectSlice := gconv.Strings(expect)
for _, v1 := range gconv.Strings(value) {
result := false
for _, v2 := range expectSlice {
if v1 == v2 {
result = true
break
}
}
if !result {
passed = false
break
}
}
default:
panic(fmt.Sprintf(`[ASSERT] INVALID EXPECT VALUE TYPE: %v`, expectKind))
}
if !passed {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v IN %v`, value, expect))
}
}
// AssertNI checks `value` is NOT IN `expect`.
// The `expect` should be a slice,
// but the `value` can be a slice or a basic type variable.
2020-03-19 22:56:12 +08:00
// TODO map support.
func AssertNI(value, expect interface{}) {
var (
passed = true
expectKind = reflect.ValueOf(expect).Kind()
)
2020-03-19 22:56:12 +08:00
switch expectKind {
case reflect.Slice, reflect.Array:
for _, v1 := range gconv.Strings(value) {
result := true
for _, v2 := range gconv.Strings(expect) {
if v1 == v2 {
result = false
break
}
}
if !result {
passed = false
break
}
}
default:
panic(fmt.Sprintf(`[ASSERT] INVALID EXPECT VALUE TYPE: %v`, expectKind))
}
if !passed {
panic(fmt.Sprintf(`[ASSERT] EXPECT %v NOT IN %v`, value, expect))
}
}
// Error panics with given `message`.
2020-03-19 22:56:12 +08:00
func Error(message ...interface{}) {
panic(fmt.Sprintf("[ERROR] %s", fmt.Sprint(message...)))
}
// Fatal prints `message` to stderr and exit the process.
2020-03-19 22:56:12 +08:00
func Fatal(message ...interface{}) {
fmt.Fprintf(os.Stderr, "[FATAL] %s\n%s", fmt.Sprint(message...), gdebug.StackWithFilter([]string{pathFilterKey}))
2020-03-19 22:56:12 +08:00
os.Exit(1)
}
// compareMap compares two maps, returns nil if they are equal, or else returns error.
func compareMap(value, expect interface{}) error {
var (
rvValue = reflect.ValueOf(value)
rvExpect = reflect.ValueOf(expect)
)
if empty.IsNil(value) {
2020-03-19 22:56:12 +08:00
value = nil
}
if rvExpect.Kind() == reflect.Map {
if rvValue.Kind() == reflect.Map {
if rvExpect.Len() == rvValue.Len() {
// Turn two interface maps to the same type for comparison.
// Direct use of rvValue.MapIndex(key).Interface() will panic
// when the key types are inconsistent.
mValue := make(map[string]string)
mExpect := make(map[string]string)
ksValue := rvValue.MapKeys()
ksExpect := rvExpect.MapKeys()
for _, key := range ksValue {
mValue[gconv.String(key.Interface())] = gconv.String(rvValue.MapIndex(key).Interface())
}
for _, key := range ksExpect {
mExpect[gconv.String(key.Interface())] = gconv.String(rvExpect.MapIndex(key).Interface())
}
for k, v := range mExpect {
if v != mValue[k] {
return fmt.Errorf(`[ASSERT] EXPECT VALUE map["%v"]:%v == map["%v"]:%v`+
"\nGIVEN : %v\nEXPECT: %v", k, mValue[k], k, v, mValue, mExpect)
}
}
} else {
return fmt.Errorf(`[ASSERT] EXPECT MAP LENGTH %d == %d`, rvValue.Len(), rvExpect.Len())
}
} else {
return fmt.Errorf(`[ASSERT] EXPECT VALUE TO BE A MAP, BUT GIVEN "%s"`, rvValue.Kind())
2020-03-19 22:56:12 +08:00
}
}
return nil
}
// AssertNil asserts `value` is nil.
func AssertNil(value interface{}) {
if empty.IsNil(value) {
return
}
if err, ok := value.(error); ok {
panic(fmt.Sprintf(`%+v`, err))
2020-03-19 22:56:12 +08:00
}
Assert(value, nil)
2020-03-19 22:56:12 +08:00
}
2021-09-15 21:17:45 +08:00
// DataPath retrieves and returns the testdata path of current package,
2021-09-15 21:17:45 +08:00
// which is used for unit testing cases only.
// The optional parameter `names` specifies the sub-folders/sub-files,
// which will be joined with current system separator and returned with the path.
func DataPath(names ...string) string {
_, path, _ := gdebug.CallerWithFilter([]string{pathFilterKey})
2021-09-15 21:17:45 +08:00
path = filepath.Dir(path) + string(filepath.Separator) + "testdata"
for _, name := range names {
path += string(filepath.Separator) + name
}
return path
}
// DataContent retrieves and returns the file content for specified testdata path of current package
func DataContent(names ...string) string {
path := DataPath(names...)
2021-09-15 21:17:45 +08:00
if path != "" {
data, err := ioutil.ReadFile(path)
if err == nil {
return string(data)
}
}
return ""
}