-
Notifications
You must be signed in to change notification settings - Fork 18.4k
Description
Background
The handling of panics and calls to runtime.Goexit in x/sync/errgroup has come up several times in its history:
- x/sync/errgroup: why not recover the fn's err in errgroup #40484 asked why we don't recover panics in goroutines.
- proposal: x/sync: add panicgroup struct to deal with panic in the sub-goroutine #49802 proposed a separate
panicgroupAPI to propagate or handle panics - CL 131815 pointed out that calls to
t.Fataland/ort.Skipwithin aGroupin a test will generally result in either a hard-to-diagnose deadlock or an awkward half-aborted test, instead of skipping or failing the test immediately as expected. - In my GopherCon 2018 talk, “Rethinking Classical Concurrency Patterns”, I recommended that API authors “[m]ake concurrency an internal detail.” In multiple discussions after the talk, folks asked me how to handle panics in goroutines, and I realized that making concurrency an internal detail requires that we propagate panics (and
runtime.Goexitcalls) back to the caller's goroutine. (Otherwise, a concurrent call that panics would terminate the program, while a sequential call that panics would be recoverable!)
Proposal
I propose that:
-
The
(*Group).Waitmethod should continue to wait for all goroutines in the group to exit, However, once that condition is met, if any of the goroutines in the group terminated with an unrecoveredpanic,Waitshould panic with a value wrapping the first panic-value recovered from a goroutine in the group. Otherwise, if any of the goroutines exited viaruntime.GoexitWaitshould invokeruntime.Goexiton its own goroutine.- Because the runtime does not support saving and restoring the stack trace of a recovered panic, the value passed to
panicbyWaitshould include a best-effort stack dump for the goroutine that initiated the panic. - Because some packages may use
recoverfor error-handling (despite our advice to the contrary), if the recovered value implements theerrorinterface, the value passed topanicbyWaitshould also implement theerrorinterface, and should wrap the recovered error (so that it can be retrieved byerrors.Unwrap).
- Because the runtime does not support saving and restoring the stack trace of a recovered panic, the value passed to
-
The
Contextvalue returned byerrgroup.WithContextshould be canceled as soon as any function call in the group returns a non-nil error, panics, or exits viaruntime.Goexit.- All of these conditions indicate that
Waithas an abnormal status to report, and thus should shut down all work associated with theGroupso that the abnormal status can be reported quickly.
- All of these conditions indicate that
Specifically, if Wait panics, the panic-value would have either type PanicValue or type PanicError, defined as follows:
// A PanicError wraps an error recovered from an unhandled panic
// when calling a function passed to Go or TryGo.
type PanicError struct {
Recovered error
Stack []byte
}
func (p PanicError) Error() string {
// A Go Error method conventionally does not include a stack dump, so omit it
// here. (Callers who care can extract it from the Stack field.)
return fmt.Sprintf("recovered from errgroup.Group: %v", p.Recovered)
}
func (p PanicError) Unwrap() error { return p.Recovered }
// A PanicValue wraps a value that does not implement the error interface,
// recovered from an unhandled panic when calling a function passed to Go or
// TryGo.
type PanicValue struct {
Recovered interface{}
Stack []byte
}
func (p PanicValue) String() string {
if len(p.Stack) > 0 {
return fmt.Sprintf("recovered from errgroup.Group: %v\n%s", p.Recovered, p.Stack)
}
return fmt.Sprintf("recovered from errgroup.Group: %v", p.Recovered)
}Compatibility
Any program that today initiates an unrecovered panic within a Go or TryGo callback terminates due to that unrecovered panic, So recovering and propagating such a panic can only change broken programs into non-broken ones; it cannot break any program that was not already broken.
A valid program could in theory call runtime.Goexit from within a Go callback today. However, the vast majority of calls to runtime.Goexit are via testing.T methods, and according to the documentation for those methods today they “must be called from the goroutine running the test or benchmark function, not from other goroutines created during the test.” Moreover, it would be possible to implement the documented errgroup.Group API today in a way that would cause Wait to always deadlock if runtime.Goexit were called, so any caller relying on the existing runtime.Goexit behavior is assuming an implementation detail that is not guaranteed.
In light of the above, I believe that the proposed changes are backward-compatible.
Metadata
Metadata
Assignees
Labels
Type
Projects
Status