diff --git a/cmd/root.go b/cmd/root.go index d629a11a1ab3..496e9314db84 100644 --- a/cmd/root.go +++ b/cmd/root.go @@ -24,7 +24,10 @@ import ( "go.k6.io/k6/log" ) -const waitLoggerCloseTimeout = time.Second * 5 +const ( + waitLoggerCloseTimeout = time.Second * 5 + waitForTracerProviderStopTimeout = time.Second * 5 +) // This is to keep all fields needed for the main/root k6 command type rootCommand struct { @@ -93,7 +96,21 @@ func (c *rootCommand) execute() { exitCode := -1 defer func() { cancel() - c.stopLoggers() + + // Wait for loggers and tracer provider to stop. + // As both methods already use a timeout, just wait. + var wg sync.WaitGroup + ff := [...]func(){c.stopLoggers, c.stopTracerProvider} + wg.Add(len(ff)) + for _, f := range ff { + f := f + go func() { + f() + wg.Done() + }() + } + wg.Wait() + c.globalState.OSExit(exitCode) }() @@ -154,6 +171,17 @@ func (c *rootCommand) stopLoggers() { } } +func (c *rootCommand) stopTracerProvider() { + ctx, cancel := context.WithTimeout(context.Background(), waitForTracerProviderStopTimeout) + defer cancel() + + if err := c.globalState.TracerProvider.Shutdown(ctx); err != nil { + c.globalState.FallbackLogger.Errorf( + "The tracer provider didn't stop gracefully: %v", err, + ) + } +} + func rootCmdPersistentFlagSet(gs *state.GlobalState) *pflag.FlagSet { flags := pflag.NewFlagSet("", pflag.ContinueOnError) // TODO: refactor this config, the default value management with pflag is