-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathTableTest.go
85 lines (82 loc) · 2.08 KB
/
TableTest.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
package testcase
import (
"fmt"
"sort"
)
// TableTest allows you to make table tests, without the need to use a boilerplate.
// It optionally allows to use a Spec instead of a testing.TB,
// and then the table tests will inherit the Spec context.
// It guards against mistakes such as using for+t.Run+t.Parallel without variable shadowing.
// TableTest allows a variety of use, please check examples for further information on that.
func TableTest[TBS anyTBOrSpec, TC func(s *Spec) | func(t *T) | any, Act func(t *T) | func(s *Spec) | func(*T, TC)](
tbs TBS,
tcs map[ /* description */ string]TC,
act Act,
) {
s := ToSpec(tbs)
var tests []tableTestTestCase[TC]
for desc, tc := range tcs {
tests = append(tests, tableTestTestCase[TC]{
Desc: desc,
TC: tc,
})
}
sort.Slice(tests, func(i, j int) bool {
return tests[i].Desc < tests[j].Desc
})
runT := func(s *Spec, test tableTestTestCase[TC], act func(t *T, tc TC)) {
switch tc := any(test.TC).(type) {
case func(s *Spec):
s.Context(test.Desc, func(s *Spec) {
tc(s)
s.Test("", func(t *T) {
act(t, test.TC)
})
})
case func(t *T):
s.Context(test.Desc, func(s *Spec) {
s.Before(tc)
s.Test("", func(t *T) {
act(t, test.TC)
})
})
default:
s.Test(test.Desc, func(t *T) {
act(t, test.TC)
})
}
}
runS := func(s *Spec, test tableTestTestCase[TC], act func(s *Spec)) {
switch tc := any(test.TC).(type) {
case func(s *Spec):
s.Context(test.Desc, func(s *Spec) {
tc(s)
act(s)
})
case func(t *T):
s.Context(test.Desc, func(s *Spec) {
s.Before(tc)
act(s)
})
default:
panic(fmt.Sprintf("unsuported TableTest setup: TC<%T> <-> Act<%T>", test.TC, act))
}
}
s.Context("", func(s *Spec) {
for _, test := range tests {
test := test // pass by value copy to avoid funny concurrency issues
switch act := any(act).(type) {
case func(s *Spec):
runS(s, test, act)
case func(t *T):
runT(s, test, func(t *T, tc TC) { act(t) })
case func(*T, TC):
runT(s, test, act)
}
}
})
}
type tableTestTestCase[TC any] struct {
Desc string
TC TC
}