2017-10-05 11:52:42 +00:00
|
|
|
// The MIT License (MIT)
|
|
|
|
|
|
|
|
// Copyright (c) 2014 Ben Johnson
|
|
|
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
// SOFTWARE.
|
|
|
|
|
|
|
|
package testutil
|
|
|
|
|
|
|
|
import (
|
2019-08-09 01:36:42 +00:00
|
|
|
"fmt"
|
2017-10-05 11:52:42 +00:00
|
|
|
"reflect"
|
2020-07-24 09:10:20 +00:00
|
|
|
"testing"
|
testutil: Enriched Equals with diff on error. (#7053)
## Example:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e160), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e180), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1a0), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1c0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e280), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e2a0), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e220), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e240), MinTime:6, MaxTime:6}}
```
Now:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e740), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e760), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e780), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e7a0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e800), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e820), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e860), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e880), MinTime:6, MaxTime:6}}
Diff:
--- Expected
+++ Actual
@@ -50,3 +50,3 @@
}),
- MinTime: (int64) 24,
+ MinTime: (int64) 6,
MaxTime: (int64) 6
```
Signed-off-by: Bartlomiej Plotka <bwplotka@gmail.com>
2020-03-26 09:48:23 +00:00
|
|
|
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
|
|
"github.com/pmezard/go-difflib/difflib"
|
2020-07-24 09:10:20 +00:00
|
|
|
"go.uber.org/goleak"
|
2017-10-05 11:52:42 +00:00
|
|
|
)
|
|
|
|
|
2018-07-25 12:10:09 +00:00
|
|
|
// This package is imported by non-test code and therefore cannot import the
|
|
|
|
// testing package, which has side effects such as adding flags. Hence we use an
|
|
|
|
// interface to testing.{T,B}.
|
|
|
|
type TB interface {
|
|
|
|
Helper()
|
|
|
|
Fatalf(string, ...interface{})
|
|
|
|
}
|
|
|
|
|
2017-10-05 11:52:42 +00:00
|
|
|
// Assert fails the test if the condition is false.
|
2018-07-25 12:10:09 +00:00
|
|
|
func Assert(tb TB, condition bool, format string, a ...interface{}) {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Helper()
|
2017-10-05 11:52:42 +00:00
|
|
|
if !condition {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Fatalf("\033[31m"+format+"\033[39m\n", a...)
|
2017-10-05 11:52:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ok fails the test if an err is not nil.
|
2018-07-25 12:10:09 +00:00
|
|
|
func Ok(tb TB, err error) {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Helper()
|
2017-10-05 11:52:42 +00:00
|
|
|
if err != nil {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Fatalf("\033[31munexpected error: %v\033[39m\n", err)
|
2017-10-05 11:52:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-12 11:14:58 +00:00
|
|
|
// NotOk fails the test if an err is nil.
|
2019-08-14 09:07:02 +00:00
|
|
|
func NotOk(tb TB, err error, a ...interface{}) {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Helper()
|
2017-10-12 11:14:58 +00:00
|
|
|
if err == nil {
|
2017-10-16 15:27:52 +00:00
|
|
|
if len(a) != 0 {
|
2019-08-14 09:07:02 +00:00
|
|
|
format := a[0].(string)
|
2020-01-20 13:17:11 +00:00
|
|
|
tb.Fatalf("\033[31m"+format+": expected error, got none\033[39m", a[1:]...)
|
2017-10-16 15:27:52 +00:00
|
|
|
}
|
|
|
|
tb.Fatalf("\033[31mexpected error, got none\033[39m")
|
2017-10-12 11:14:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-05 11:52:42 +00:00
|
|
|
// Equals fails the test if exp is not equal to act.
|
2019-08-09 01:36:42 +00:00
|
|
|
func Equals(tb TB, exp, act interface{}, msgAndArgs ...interface{}) {
|
2017-10-16 15:27:52 +00:00
|
|
|
tb.Helper()
|
2017-10-05 11:52:42 +00:00
|
|
|
if !reflect.DeepEqual(exp, act) {
|
testutil: Enriched Equals with diff on error. (#7053)
## Example:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e160), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e180), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1a0), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1c0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e280), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e2a0), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e220), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e240), MinTime:6, MaxTime:6}}
```
Now:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e740), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e760), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e780), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e7a0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e800), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e820), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e860), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e880), MinTime:6, MaxTime:6}}
Diff:
--- Expected
+++ Actual
@@ -50,3 +50,3 @@
}),
- MinTime: (int64) 24,
+ MinTime: (int64) 6,
MaxTime: (int64) 6
```
Signed-off-by: Bartlomiej Plotka <bwplotka@gmail.com>
2020-03-26 09:48:23 +00:00
|
|
|
tb.Fatalf("\033[31m%s\n\nexp: %#v\n\ngot: %#v%s\033[39m\n", formatMessage(msgAndArgs), exp, act, diff(exp, act))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func typeAndKind(v interface{}) (reflect.Type, reflect.Kind) {
|
|
|
|
t := reflect.TypeOf(v)
|
|
|
|
k := t.Kind()
|
|
|
|
|
|
|
|
if k == reflect.Ptr {
|
|
|
|
t = t.Elem()
|
|
|
|
k = t.Kind()
|
2017-10-05 11:52:42 +00:00
|
|
|
}
|
testutil: Enriched Equals with diff on error. (#7053)
## Example:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e160), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e180), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1a0), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e1c0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e280), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e2a0), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e220), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e240), MinTime:6, MaxTime:6}}
```
Now:
```
exp: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e740), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e760), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e780), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e7a0), MinTime:24, MaxTime:6}}
got: []chunks.Meta{chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e800), MinTime:1, MaxTime:2}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e820), MinTime:3, MaxTime:3}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e860), MinTime:5, MaxTime:5}, chunks.Meta{Ref:0x0, Chunk:(*chunkenc.XORChunk)(0xc00000e880), MinTime:6, MaxTime:6}}
Diff:
--- Expected
+++ Actual
@@ -50,3 +50,3 @@
}),
- MinTime: (int64) 24,
+ MinTime: (int64) 6,
MaxTime: (int64) 6
```
Signed-off-by: Bartlomiej Plotka <bwplotka@gmail.com>
2020-03-26 09:48:23 +00:00
|
|
|
return t, k
|
|
|
|
}
|
|
|
|
|
|
|
|
// diff returns a diff of both values as long as both are of the same type and
|
|
|
|
// are a struct, map, slice, array or string. Otherwise it returns an empty string.
|
|
|
|
func diff(expected interface{}, actual interface{}) string {
|
|
|
|
if expected == nil || actual == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
et, ek := typeAndKind(expected)
|
|
|
|
at, _ := typeAndKind(actual)
|
|
|
|
if et != at {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if ek != reflect.Struct && ek != reflect.Map && ek != reflect.Slice && ek != reflect.Array && ek != reflect.String {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
var e, a string
|
|
|
|
c := spew.ConfigState{
|
|
|
|
Indent: " ",
|
|
|
|
DisablePointerAddresses: true,
|
|
|
|
DisableCapacities: true,
|
|
|
|
SortKeys: true,
|
|
|
|
}
|
|
|
|
if et != reflect.TypeOf("") {
|
|
|
|
e = c.Sdump(expected)
|
|
|
|
a = c.Sdump(actual)
|
|
|
|
} else {
|
|
|
|
e = reflect.ValueOf(expected).String()
|
|
|
|
a = reflect.ValueOf(actual).String()
|
|
|
|
}
|
|
|
|
|
|
|
|
diff, _ := difflib.GetUnifiedDiffString(difflib.UnifiedDiff{
|
|
|
|
A: difflib.SplitLines(e),
|
|
|
|
B: difflib.SplitLines(a),
|
|
|
|
FromFile: "Expected",
|
|
|
|
FromDate: "",
|
|
|
|
ToFile: "Actual",
|
|
|
|
ToDate: "",
|
|
|
|
Context: 1,
|
|
|
|
})
|
|
|
|
return "\n\nDiff:\n" + diff
|
2017-10-05 11:52:42 +00:00
|
|
|
}
|
2019-08-09 01:36:42 +00:00
|
|
|
|
2019-12-17 21:11:33 +00:00
|
|
|
// ErrorEqual compares Go errors for equality.
|
|
|
|
func ErrorEqual(tb TB, left, right error, msgAndArgs ...interface{}) {
|
|
|
|
tb.Helper()
|
|
|
|
if left == right {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if left != nil && right != nil {
|
|
|
|
Equals(tb, left.Error(), right.Error(), msgAndArgs...)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tb.Fatalf("\033[31m%s\n\nexp: %#v\n\ngot: %#v\033[39m\n", formatMessage(msgAndArgs), left, right)
|
|
|
|
}
|
|
|
|
|
2019-08-09 01:36:42 +00:00
|
|
|
func formatMessage(msgAndArgs []interface{}) string {
|
|
|
|
if len(msgAndArgs) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if msg, ok := msgAndArgs[0].(string); ok {
|
2019-08-14 09:07:02 +00:00
|
|
|
return fmt.Sprintf("\n\nmsg: "+msg, msgAndArgs[1:]...)
|
2019-08-09 01:36:42 +00:00
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2020-07-24 09:10:20 +00:00
|
|
|
|
|
|
|
// TolerantVerifyLeak verifies go leaks but excludes the go routines that are
|
|
|
|
// launched as side effects of some of our dependencies.
|
|
|
|
func TolerantVerifyLeak(m *testing.M) {
|
|
|
|
goleak.VerifyTestMain(m,
|
|
|
|
// https://github.com/census-instrumentation/opencensus-go/blob/d7677d6af5953e0506ac4c08f349c62b917a443a/stats/view/worker.go#L34
|
|
|
|
goleak.IgnoreTopFunction("go.opencensus.io/stats/view.(*worker).start"),
|
|
|
|
// https://github.com/kubernetes/klog/blob/c85d02d1c76a9ebafa81eb6d35c980734f2c4727/klog.go#L417
|
|
|
|
goleak.IgnoreTopFunction("k8s.io/klog/v2.(*loggingT).flushDaemon"),
|
|
|
|
)
|
|
|
|
}
|