Description
Background
#16221 added a func (*T) Context() context.Context
to the testing package, which was backed out due to concerns from @dsnet, @neild, @niemeyer, me, and perhaps others as well (#18199).
CL 134395 (errgroup
propagation of Goexit
, among other things) mitigates those concerns somewhat: it gives test authors a straightforward way to start a collection of goroutines and wait for them to finish, and cancels them if any of the goroutines calls t.FailNow
or t.SkipNow
(#15758 notwithstanding).
That leaves one use-case unaddressed: as @neild noted in the golang-dev thread and @dsnet noted in #18199 (comment), if a test times out (e.g. because of a deadlock in the package under test), it is often useful to shut it down cleanly and emit useful logs rather than wait for the testing
package to dump all of the running goroutines.
@bradfitz suggested that test authors can add their own derived context, but a test in general has no idea how much time it has left: if we want to check the -test.timeout
flag, we need to be able to subtract off the amount of time spent so far, and that implies the use of TestMain
and the boilerplate that goes with it.
Proposal
I propose that we add back the (*testing.T).Context
method, with the same signature as before but the following semantics:
- When a test binary times out:
- First, all tests that are still running are marked as having failed due to timing out.
- Then, the contexts provided to tests are marked as
Done
, withErr
returningcontext.DeadlineExceeded
. - Finally, the test binary waits for some “reasonable” (but unspecified) grace period elapses, or until all tests that called their
Context
method (and their subtests) have returned, whichever occurs first. - If any tests are still running at that point, the test binary writes out the logs for any completed tests and then panics as before.
- When a test is marked as
Failed
orSkipped
:- The context provided to that test (if any) is marked as
Done
, withErr
returningcontext.Canceled
.
- The context provided to that test (if any) is marked as
- When a test function exits normally:
- The context provided to that test is not marked as
Done
. That discourages the use of(*T).Context
for normal cleanup of “background” goroutines, which should be accomplished by some other means (sync.WaitGroup
,errgroup
,tomb
, or whatever alternative the test author prefers.)
- The context provided to that test is not marked as