aboutsummaryrefslogtreecommitdiff
path: root/src/testing/sub_test.go
diff options
context:
space:
mode:
authorBryan C. Mills <bcmills@google.com>2023-06-28 09:37:47 -0400
committerGopher Robot <gobot@golang.org>2023-11-21 22:58:46 +0000
commite6b76bfc4675eae8d1e4c7e8dc28897339b13824 (patch)
treedc62e2a94a7162476d9d3417853142f7c3925453 /src/testing/sub_test.go
parentf5bf9fb278c473104b0b987fc1dd165566cbec71 (diff)
downloadgo-e6b76bfc4675eae8d1e4c7e8dc28897339b13824.tar.xz
testing: simplify concurrency and cleanup logic
While investigating #60083, I found a couple of bugs (notably #61034) that had slipped through code review in part because the concurrency patterns used in the testing package were too complex for me to fully reason about. This change adjusts those patterns to be more in line with current idioms, and to reduce the number of special cases that depend on details that should be orthogonal. (For example: the details of how we invoke the Cleanup functions should not depend on whether the test happened to run any parallel subtests.) In the process, this change fixes a handful of bugs: - Concurrent calls to Run (explicitly allowed by TestParallelSub) could previously drive the testcontext.running count negative, causing the number of running parallel tests to exceed the -parallel flag. - The -failfast flag now takes effect immediately on failure. It no longer delays until the test finishes, and no longer misses failures during cleanup (fixing #61034). - If a Cleanup function calls runtime.Goexit (typically via t.FailNow) during a panic, Cleanup functions from its parent tests are no longer skipped and buffered logs from its parent tests are now flushed. - The time reported for a test with subtests now includes the time spent running those subtests, regardless of whether they are parallel. (Previously, non-parallel subtests were included but parallel subtests were not.) - Calls to (*B).Run in iterations after the first are now diagnosed with a panic. (This diagnoses badly-behaved benchmarks: if Run is called during the first iteration, no subsequent iterations are supposed to occur.) Fixes #61034. Change-Id: I3797f6ef5210a3d2d5d6c2710d3f35c0219b02ea Cq-Include-Trybots: luci.golang.try:gotip-linux-amd64-longtest,gotip-linux-amd64-longtest-race,gotip-windows-amd64-longtest Reviewed-on: https://go-review.googlesource.com/c/go/+/506755 Auto-Submit: Bryan Mills <bcmills@google.com> Reviewed-by: Alan Donovan <adonovan@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Diffstat (limited to 'src/testing/sub_test.go')
-rw-r--r--src/testing/sub_test.go131
1 files changed, 19 insertions, 112 deletions
diff --git a/src/testing/sub_test.go b/src/testing/sub_test.go
index 55b14c3795..7cbbd2e1cf 100644
--- a/src/testing/sub_test.go
+++ b/src/testing/sub_test.go
@@ -21,102 +21,6 @@ func init() {
benchTime.d = 100 * time.Millisecond
}
-func TestTestContext(t *T) {
- const (
- add1 = 0
- done = 1
- )
- // After each of the calls are applied to the context, the
- type call struct {
- typ int // run or done
- // result from applying the call
- running int
- waiting int
- started bool
- }
- testCases := []struct {
- max int
- run []call
- }{{
- max: 1,
- run: []call{
- {typ: add1, running: 1, waiting: 0, started: true},
- {typ: done, running: 0, waiting: 0, started: false},
- },
- }, {
- max: 1,
- run: []call{
- {typ: add1, running: 1, waiting: 0, started: true},
- {typ: add1, running: 1, waiting: 1, started: false},
- {typ: done, running: 1, waiting: 0, started: true},
- {typ: done, running: 0, waiting: 0, started: false},
- {typ: add1, running: 1, waiting: 0, started: true},
- },
- }, {
- max: 3,
- run: []call{
- {typ: add1, running: 1, waiting: 0, started: true},
- {typ: add1, running: 2, waiting: 0, started: true},
- {typ: add1, running: 3, waiting: 0, started: true},
- {typ: add1, running: 3, waiting: 1, started: false},
- {typ: add1, running: 3, waiting: 2, started: false},
- {typ: add1, running: 3, waiting: 3, started: false},
- {typ: done, running: 3, waiting: 2, started: true},
- {typ: add1, running: 3, waiting: 3, started: false},
- {typ: done, running: 3, waiting: 2, started: true},
- {typ: done, running: 3, waiting: 1, started: true},
- {typ: done, running: 3, waiting: 0, started: true},
- {typ: done, running: 2, waiting: 0, started: false},
- {typ: done, running: 1, waiting: 0, started: false},
- {typ: done, running: 0, waiting: 0, started: false},
- },
- }}
- for i, tc := range testCases {
- ctx := &testContext{
- startParallel: make(chan bool),
- maxParallel: tc.max,
- }
- for j, call := range tc.run {
- doCall := func(f func()) chan bool {
- done := make(chan bool)
- go func() {
- f()
- done <- true
- }()
- return done
- }
- started := false
- switch call.typ {
- case add1:
- signal := doCall(ctx.waitParallel)
- select {
- case <-signal:
- started = true
- case ctx.startParallel <- true:
- <-signal
- }
- case done:
- signal := doCall(ctx.release)
- select {
- case <-signal:
- case <-ctx.startParallel:
- started = true
- <-signal
- }
- }
- if started != call.started {
- t.Errorf("%d:%d:started: got %v; want %v", i, j, started, call.started)
- }
- if ctx.running != call.running {
- t.Errorf("%d:%d:running: got %v; want %v", i, j, ctx.running, call.running)
- }
- if ctx.numWaiting != call.waiting {
- t.Errorf("%d:%d:waiting: got %v; want %v", i, j, ctx.numWaiting, call.waiting)
- }
- }
- }
-}
-
func TestTRun(t *T) {
realTest := t
testCases := []struct {
@@ -511,10 +415,10 @@ func TestTRun(t *T) {
buf := &strings.Builder{}
root := &T{
common: common{
- signal: make(chan bool),
- barrier: make(chan bool),
- name: "",
- w: buf,
+ doneOrParallel: make(chan struct{}),
+ runParallel: make(chan struct{}),
+ name: "",
+ w: buf,
},
context: ctx,
}
@@ -523,7 +427,6 @@ func TestTRun(t *T) {
root.chatty.json = tc.json
}
ok := root.Run(tc.desc, tc.f)
- ctx.release()
if ok != tc.ok {
t.Errorf("%s:ok: got %v; want %v", tc.desc, ok, tc.ok)
@@ -531,8 +434,8 @@ func TestTRun(t *T) {
if ok != !root.Failed() {
t.Errorf("%s:root failed: got %v; want %v", tc.desc, !ok, root.Failed())
}
- if ctx.running != 0 || ctx.numWaiting != 0 {
- t.Errorf("%s:running and waiting non-zero: got %d and %d", tc.desc, ctx.running, ctx.numWaiting)
+ if n := ctx.running(); n != 0 {
+ t.Errorf("%s:running non-zero: got %d", tc.desc, n)
}
got := strings.TrimSpace(buf.String())
want := strings.TrimSpace(tc.output)
@@ -701,9 +604,9 @@ func TestBRun(t *T) {
// the benchtime and catch the failure result of the subbenchmark.
root := &B{
common: common{
- signal: make(chan bool),
- name: "root",
- w: buf,
+ doneOrParallel: make(chan struct{}),
+ name: "root",
+ w: buf,
},
benchFunc: func(b *B) { ok = b.Run("test", tc.f) }, // Use Run to catch failure.
benchTime: durationOrCountFlag{d: 1 * time.Microsecond},
@@ -837,8 +740,8 @@ func TestLogAfterComplete(t *T) {
common: common{
// Use a buffered channel so that tRunner can write
// to it although nothing is reading from it.
- signal: make(chan bool, 1),
- w: &buf,
+ doneOrParallel: make(chan struct{}),
+ w: &buf,
},
context: ctx,
}
@@ -910,18 +813,22 @@ func TestCleanup(t *T) {
func TestConcurrentCleanup(t *T) {
cleanups := 0
t.Run("test", func(t *T) {
- done := make(chan struct{})
+ var wg sync.WaitGroup
+ wg.Add(2)
for i := 0; i < 2; i++ {
i := i
go func() {
t.Cleanup(func() {
+ // Although the calls to Cleanup are concurrent, the functions passed
+ // to Cleanup should be called sequentially, in some nondeterministic
+ // order based on when the Cleanup calls happened to be scheduled.
+ // So these assignments to the cleanups variable should not race.
cleanups |= 1 << i
})
- done <- struct{}{}
+ wg.Done()
}()
}
- <-done
- <-done
+ wg.Wait()
})
if cleanups != 1|2 {
t.Errorf("unexpected cleanup; got %d want 3", cleanups)