testing: reject calls to Run within Cleanup callbacks

Calling t.Run inside t.Cleanup can mess up the execution order of
registered Cleanup callbacks. Reject calls to Run within Cleanup
callbacks.

Fixes #48515

Change-Id: I61e4cb35253db1a8bbe3351d59055433030aa289
Reviewed-on: https://go-review.googlesource.com/c/go/+/352349
TryBot-Result: Gopher Robot <gobot@golang.org>
Reviewed-by: Joedian Reid <joedian@golang.org>
Run-TryBot: Changkun Ou <mail@changkun.de>
Auto-Submit: Bryan Mills <bcmills@google.com>
Reviewed-by: Bryan Mills <bcmills@google.com>
This commit is contained in:
Changkun Ou 2021-09-27 11:37:56 +02:00 committed by Gopher Robot
parent 7c8c209b45
commit 978ce7e252
2 changed files with 31 additions and 6 deletions

View File

@ -224,6 +224,11 @@ func TestMorePanic(t *testing.T) {
want: `panic: die want: `panic: die
panic: test executed panic(nil) or runtime.Goexit`, panic: test executed panic(nil) or runtime.Goexit`,
}, },
{
desc: "Issue 48515: call t.Run in t.Cleanup should trigger panic",
flags: []string{"-test.run=TestCallRunInCleanupHelper"},
want: `panic: testing: t.Run is called during t.Cleanup`,
},
} }
for _, tc := range testCases { for _, tc := range testCases {
@ -239,6 +244,18 @@ func TestMorePanic(t *testing.T) {
} }
} }
func TestCallRunInCleanupHelper(t *testing.T) {
if os.Getenv("GO_WANT_HELPER_PROCESS") != "1" {
return
}
t.Cleanup(func() {
t.Run("in-cleanup", func(t *testing.T) {
t.Log("must not be executed")
})
})
}
func TestGoexitInCleanupAfterPanicHelper(t *testing.T) { func TestGoexitInCleanupAfterPanicHelper(t *testing.T) {
if os.Getenv("GO_WANT_HELPER_PROCESS") != "1" { if os.Getenv("GO_WANT_HELPER_PROCESS") != "1" {
return return

View File

@ -604,12 +604,13 @@ type common struct {
finished bool // Test function has completed. finished bool // Test function has completed.
inFuzzFn bool // Whether the fuzz target, if this is one, is running. inFuzzFn bool // Whether the fuzz target, if this is one, is running.
chatty *chattyPrinter // A copy of chattyPrinter, if the chatty flag is set. chatty *chattyPrinter // A copy of chattyPrinter, if the chatty flag is set.
bench bool // Whether the current test is a benchmark. bench bool // Whether the current test is a benchmark.
hasSub atomic.Bool // whether there are sub-benchmarks. hasSub atomic.Bool // whether there are sub-benchmarks.
raceErrors int // Number of races detected during test. cleanupStarted atomic.Bool // Registered cleanup callbacks have started to execute
runner string // Function name of tRunner running the test. raceErrors int // Number of races detected during test.
isParallel bool // Whether the test is parallel. runner string // Function name of tRunner running the test.
isParallel bool // Whether the test is parallel.
parent *common parent *common
level int // Nesting depth of test or benchmark. level int // Nesting depth of test or benchmark.
@ -1291,6 +1292,9 @@ const (
// If catchPanic is true, this will catch panics, and return the recovered // If catchPanic is true, this will catch panics, and return the recovered
// value if any. // value if any.
func (c *common) runCleanup(ph panicHandling) (panicVal any) { func (c *common) runCleanup(ph panicHandling) (panicVal any) {
c.cleanupStarted.Store(true)
defer c.cleanupStarted.Store(false)
if ph == recoverAndReturnPanic { if ph == recoverAndReturnPanic {
defer func() { defer func() {
panicVal = recover() panicVal = recover()
@ -1583,6 +1587,10 @@ func tRunner(t *T, fn func(t *T)) {
// Run may be called simultaneously from multiple goroutines, but all such calls // Run may be called simultaneously from multiple goroutines, but all such calls
// must return before the outer test function for t returns. // must return before the outer test function for t returns.
func (t *T) Run(name string, f func(t *T)) bool { func (t *T) Run(name string, f func(t *T)) bool {
if t.cleanupStarted.Load() {
panic("testing: t.Run is called during t.Cleanup")
}
t.hasSub.Store(true) t.hasSub.Store(true)
testName, ok, _ := t.context.match.fullName(&t.common, name) testName, ok, _ := t.context.match.fullName(&t.common, name)
if !ok || shouldFailFast() { if !ok || shouldFailFast() {