diff --git a/api/v1/setup_teardown_routes_test.go b/api/v1/setup_teardown_routes_test.go index fa61aeecb14..8829755c901 100644 --- a/api/v1/setup_teardown_routes_test.go +++ b/api/v1/setup_teardown_routes_test.go @@ -142,7 +142,7 @@ func TestSetupData(t *testing.T) { require.NoError(t, err) require.NoError(t, engine.OutputManager.StartOutputs()) - defer engine.OutputManager.StopOutputs() + defer engine.OutputManager.StopOutputs(nil) globalCtx, globalCancel := context.WithCancel(context.Background()) runCtx, runCancel := context.WithCancel(globalCtx) diff --git a/api/v1/status_routes_test.go b/api/v1/status_routes_test.go index 685a2c25737..03ce1f2d4f5 100644 --- a/api/v1/status_routes_test.go +++ b/api/v1/status_routes_test.go @@ -116,7 +116,7 @@ func TestPatchStatus(t *testing.T) { require.NoError(t, err) require.NoError(t, engine.OutputManager.StartOutputs()) - defer engine.OutputManager.StopOutputs() + defer engine.OutputManager.StopOutputs(nil) ctx, cancel := context.WithTimeout(context.Background(), 2*time.Minute) run, wait, err := engine.Init(ctx, ctx) diff --git a/cloudapi/api.go b/cloudapi/api.go index 76d9c86a6f5..e40b82001b7 100644 --- a/cloudapi/api.go +++ b/cloudapi/api.go @@ -43,10 +43,10 @@ type CreateTestRunResponse struct { } type TestProgressResponse struct { - RunStatusText string `json:"run_status_text"` - RunStatus lib.RunStatus `json:"run_status"` - ResultStatus ResultStatus `json:"result_status"` - Progress float64 `json:"progress"` + RunStatusText string `json:"run_status_text"` + RunStatus RunStatus `json:"run_status"` + ResultStatus ResultStatus `json:"result_status"` + Progress float64 `json:"progress"` } type LoginResponse struct { @@ -133,7 +133,9 @@ func (c *Client) StartCloudTestRun(name string, projectID int64, arc *lib.Archiv return &ctrr, nil } -func (c *Client) TestFinished(referenceID string, thresholds ThresholdResult, tained bool, runStatus lib.RunStatus) error { +// TestFinished sends the result and run status values to the cloud, along with +// information for the test thresholds, and marks the test run as finished. +func (c *Client) TestFinished(referenceID string, thresholds ThresholdResult, tained bool, runStatus RunStatus) error { url := fmt.Sprintf("%s/tests/%s", c.baseURL, referenceID) resultStatus := ResultStatusPassed @@ -143,7 +145,7 @@ func (c *Client) TestFinished(referenceID string, thresholds ThresholdResult, ta data := struct { ResultStatus ResultStatus `json:"result_status"` - RunStatus lib.RunStatus `json:"run_status"` + RunStatus RunStatus `json:"run_status"` Thresholds ThresholdResult `json:"thresholds"` }{ resultStatus, diff --git a/lib/run_status.go b/cloudapi/run_status.go similarity index 71% rename from lib/run_status.go rename to cloudapi/run_status.go index b331599ebf7..98500a63532 100644 --- a/lib/run_status.go +++ b/cloudapi/run_status.go @@ -1,10 +1,7 @@ -package lib +package cloudapi -// TODO: move to some other package - types? models? - -// RunStatus values can be used by k6 to denote how a script run ends -// and by the cloud executor and collector so that k6 knows the current -// status of a particular script run. +// RunStatus values are used to tell the cloud output how a local test run +// ended, and to get that information from the cloud for cloud tests. type RunStatus int // Possible run status values; iota isn't used intentionally diff --git a/cmd/cloud.go b/cmd/cloud.go index e11cf6d01ab..0ffcbee4858 100644 --- a/cmd/cloud.go +++ b/cmd/cloud.go @@ -63,6 +63,7 @@ func (c *cmdCloud) preRun(cmd *cobra.Command, args []string) error { } // TODO: split apart some more +// //nolint:funlen,gocognit,cyclop func (c *cmdCloud) run(cmd *cobra.Command, args []string) error { printBanner(c.gs) @@ -235,9 +236,9 @@ func (c *cmdCloud) run(cmd *cobra.Command, args []string) error { statusText := testProgress.RunStatusText - if testProgress.RunStatus == lib.RunStatusFinished { + if testProgress.RunStatus == cloudapi.RunStatusFinished { testProgress.Progress = 1 - } else if testProgress.RunStatus == lib.RunStatusRunning { + } else if testProgress.RunStatus == cloudapi.RunStatusRunning { if startTime.IsZero() { startTime = time.Now() } @@ -274,8 +275,8 @@ func (c *cmdCloud) run(cmd *cobra.Command, args []string) error { testProgress = newTestProgress testProgressLock.Unlock() - if (newTestProgress.RunStatus > lib.RunStatusRunning) || - (c.exitOnRunning && newTestProgress.RunStatus == lib.RunStatusRunning) { + if (newTestProgress.RunStatus > cloudapi.RunStatusRunning) || + (c.exitOnRunning && newTestProgress.RunStatus == cloudapi.RunStatusRunning) { globalCancel() break } diff --git a/cmd/run.go b/cmd/run.go index fac3c66e17a..c2060c86527 100644 --- a/cmd/run.go +++ b/cmd/run.go @@ -37,7 +37,7 @@ type cmdRun struct { // TODO: split apart some more // //nolint:funlen,gocognit,gocyclo,cyclop -func (c *cmdRun) run(cmd *cobra.Command, args []string) error { +func (c *cmdRun) run(cmd *cobra.Command, args []string) (err error) { printBanner(c.gs) test, err := loadAndConfigureTest(c.gs, cmd, args, getConfig) @@ -159,7 +159,9 @@ func (c *cmdRun) run(cmd *cobra.Command, args []string) error { if err != nil { return err } - defer engine.OutputManager.StopOutputs() + defer func() { + engine.OutputManager.StopOutputs(err) + }() printExecutionDescription( c.gs, "local", args[0], "", conf, execScheduler.GetState().ExecutionTuple, executionPlan, outputs, @@ -274,7 +276,9 @@ func (c *cmdRun) run(cmd *cobra.Command, args []string) error { } else { logger.Error("some thresholds have failed") // log this, even if there was already a previous error } - err = errext.WithExitCodeIfNone(err, exitcodes.ThresholdsHaveFailed) + err = errext.WithAbortReasonIfNone( + errext.WithExitCodeIfNone(err, exitcodes.ThresholdsHaveFailed), errext.AbortedByThresholdsAfterTestEnd, + ) } return err } diff --git a/cmd/tests/cmd_cloud_test.go b/cmd/tests/cmd_cloud_test.go index 1917d658fcb..4aa8800e47d 100644 --- a/cmd/tests/cmd_cloud_test.go +++ b/cmd/tests/cmd_cloud_test.go @@ -13,7 +13,6 @@ import ( "github.com/stretchr/testify/require" "go.k6.io/k6/cloudapi" "go.k6.io/k6/cmd" - "go.k6.io/k6/lib" ) func cloudTestStartSimple(t *testing.T, testRunID int) http.Handler { @@ -34,7 +33,7 @@ func getMockCloud( testProgressURL := fmt.Sprintf("GET ^/v1/test-progress/%d$", testRunID) defaultProgress := cloudapi.TestProgressResponse{ RunStatusText: "Finished", - RunStatus: lib.RunStatusFinished, + RunStatus: cloudapi.RunStatusFinished, ResultStatus: cloudapi.ResultStatusPassed, Progress: 1, } @@ -130,7 +129,7 @@ func TestCloudExitOnRunning(t *testing.T) { cs := func() cloudapi.TestProgressResponse { return cloudapi.TestProgressResponse{ RunStatusText: "Running", - RunStatus: lib.RunStatusRunning, + RunStatus: cloudapi.RunStatusRunning, } } diff --git a/cmd/tests/cmd_run_test.go b/cmd/tests/cmd_run_test.go index e3e612d7a10..98c6cd58932 100644 --- a/cmd/tests/cmd_run_test.go +++ b/cmd/tests/cmd_run_test.go @@ -24,7 +24,6 @@ import ( "go.k6.io/k6/cloudapi" "go.k6.io/k6/cmd" "go.k6.io/k6/errext/exitcodes" - "go.k6.io/k6/lib" "go.k6.io/k6/lib/consts" "go.k6.io/k6/lib/testutils" "go.k6.io/k6/lib/testutils/httpmultibin" @@ -426,7 +425,7 @@ func getTestServer(t *testing.T, routes map[string]http.Handler) *httptest.Serve func getCloudTestEndChecker( t *testing.T, testRunID int, - testStart http.Handler, expRunStatus lib.RunStatus, expResultStatus cloudapi.ResultStatus, + testStart http.Handler, expRunStatus cloudapi.RunStatus, expResultStatus cloudapi.ResultStatus, ) *httptest.Server { testFinished := false @@ -449,7 +448,7 @@ func getCloudTestEndChecker( runStatus := gjson.GetBytes(body, "run_status") require.True(t, runStatus.Exists()) // important to check, since run_status can be 0 assert.Equalf( - t, expRunStatus, lib.RunStatus(runStatus.Int()), + t, expRunStatus, cloudapi.RunStatus(runStatus.Int()), "received wrong run_status value", ) @@ -473,7 +472,7 @@ func getCloudTestEndChecker( func getSimpleCloudOutputTestState( t *testing.T, script string, cliFlags []string, - expRunStatus lib.RunStatus, expResultStatus cloudapi.ResultStatus, expExitCode exitcodes.ExitCode, + expRunStatus cloudapi.RunStatus, expResultStatus cloudapi.ResultStatus, expExitCode exitcodes.ExitCode, ) *GlobalTestState { if cliFlags == nil { cliFlags = []string{"-v", "--log-output=stdout"} @@ -520,7 +519,7 @@ func TestSetupTeardownThresholds(t *testing.T) { }; `) - ts := getSimpleCloudOutputTestState(t, script, nil, lib.RunStatusFinished, cloudapi.ResultStatusPassed, 0) + ts := getSimpleCloudOutputTestState(t, script, nil, cloudapi.RunStatusFinished, cloudapi.ResultStatusPassed, 0) cmd.ExecuteWithGlobalState(ts.GlobalState) stdOut := ts.Stdout.String() @@ -566,7 +565,7 @@ func TestThresholdsFailed(t *testing.T) { // Since these thresholds don't have an abortOnFail property, the run_status // in the cloud will still be Finished, even if the test itself failed. ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusFinished, cloudapi.ResultStatusFailed, exitcodes.ThresholdsHaveFailed, + t, script, nil, cloudapi.RunStatusFinished, cloudapi.ResultStatusFailed, exitcodes.ThresholdsHaveFailed, ) cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -607,7 +606,7 @@ func TestAbortedByThreshold(t *testing.T) { ` ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusAbortedThreshold, cloudapi.ResultStatusFailed, exitcodes.ThresholdsHaveFailed, + t, script, nil, cloudapi.RunStatusAbortedThreshold, cloudapi.ResultStatusFailed, exitcodes.ThresholdsHaveFailed, ) cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -654,7 +653,7 @@ func TestAbortedByUserWithGoodThresholds(t *testing.T) { }; ` - ts := getSimpleCloudOutputTestState(t, script, nil, lib.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ExternalAbort) + ts := getSimpleCloudOutputTestState(t, script, nil, cloudapi.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ExternalAbort) asyncWaitForStdoutAndStopTestWithInterruptSignal(t, ts, 15, time.Second, "simple iter 2") @@ -782,7 +781,7 @@ func TestAbortedByUserWithRestAPI(t *testing.T) { ts := getSimpleCloudOutputTestState( t, script, []string{"-v", "--log-output=stdout", "--iterations", "20"}, - lib.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ScriptStoppedFromRESTAPI, + cloudapi.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ScriptStoppedFromRESTAPI, ) asyncWaitForStdoutAndStopTestFromRESTAPI(t, ts, 15, time.Second, "a simple iteration") @@ -822,7 +821,7 @@ func TestAbortedByScriptSetupErrorWithDependency(t *testing.T) { export { handleSummary } from "./bar.js"; ` - srv := getCloudTestEndChecker(t, 123, nil, lib.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed) + srv := getCloudTestEndChecker(t, 123, nil, cloudapi.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed) ts := NewGlobalTestState(t) require.NoError(t, afero.WriteFile(ts.FS, filepath.Join(ts.Cwd, "test.js"), []byte(mainScript), 0o644)) @@ -937,7 +936,7 @@ func TestAbortedByScriptTeardownError(t *testing.T) { func testAbortedByScriptError(t *testing.T, script string, runTest func(*testing.T, *GlobalTestState)) *GlobalTestState { ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed, exitcodes.ScriptException, + t, script, nil, cloudapi.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed, exitcodes.ScriptException, ) runTest(t, ts) @@ -1082,7 +1081,7 @@ func testAbortedByScriptTestAbort( t *testing.T, shouldHaveMetrics bool, script string, runTest func(*testing.T, *GlobalTestState), ) *GlobalTestState { //nolint:unparam ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ScriptAborted, + t, script, nil, cloudapi.RunStatusAbortedUser, cloudapi.ResultStatusPassed, exitcodes.ScriptAborted, ) runTest(t, ts) @@ -1125,7 +1124,7 @@ func TestAbortedByInterruptDuringVUInit(t *testing.T) { // This is testing the current behavior, which is expected, but it's not // actually the desired one! See https://github.com/grafana/k6/issues/2804 ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusAbortedSystem, cloudapi.ResultStatusPassed, exitcodes.GenericEngine, + t, script, nil, cloudapi.RunStatusAbortedSystem, cloudapi.ResultStatusPassed, exitcodes.GenericEngine, ) asyncWaitForStdoutAndStopTestWithInterruptSignal(t, ts, 15, time.Second, "VU init sleeping for a while") cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -1157,7 +1156,7 @@ func TestAbortedByScriptInitError(t *testing.T) { ` ts := getSimpleCloudOutputTestState( - t, script, nil, lib.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed, exitcodes.ScriptException, + t, script, nil, cloudapi.RunStatusAbortedScriptError, cloudapi.ResultStatusPassed, exitcodes.ScriptException, ) cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -1257,7 +1256,7 @@ func TestMetricTagAndSetupDataIsolation(t *testing.T) { ts := getSimpleCloudOutputTestState( t, script, []string{"--quiet", "--log-output", "stdout"}, - lib.RunStatusFinished, cloudapi.ResultStatusPassed, 0, + cloudapi.RunStatusFinished, cloudapi.ResultStatusPassed, 0, ) cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -1414,7 +1413,7 @@ func TestMinIterationDuration(t *testing.T) { import { Counter } from 'k6/metrics'; export let options = { - minIterationDuration: '5s', + minIterationDuration: '7s', setupTimeout: '2s', teardownTimeout: '2s', thresholds: { @@ -1429,14 +1428,14 @@ func TestMinIterationDuration(t *testing.T) { export function teardown() { c.add(1); }; ` - ts := getSimpleCloudOutputTestState(t, script, nil, lib.RunStatusFinished, cloudapi.ResultStatusPassed, 0) + ts := getSimpleCloudOutputTestState(t, script, nil, cloudapi.RunStatusFinished, cloudapi.ResultStatusPassed, 0) start := time.Now() cmd.ExecuteWithGlobalState(ts.GlobalState) elapsed := time.Since(start) - assert.Greater(t, elapsed, 5*time.Second, "expected more time to have passed because of minIterationDuration") + assert.Greater(t, elapsed, 7*time.Second, "expected more time to have passed because of minIterationDuration") assert.Less( - t, elapsed, 10*time.Second, + t, elapsed, 14*time.Second, "expected less time to have passed because minIterationDuration should not affect setup() and teardown() ", ) @@ -1566,7 +1565,7 @@ func TestRunWithCloudOutputOverrides(t *testing.T) { _, err := fmt.Fprint(resp, `{"reference_id": "132", "config": {"webAppURL": "https://bogus.url"}}`) assert.NoError(t, err) }) - srv := getCloudTestEndChecker(t, 132, configOverride, lib.RunStatusFinished, cloudapi.ResultStatusPassed) + srv := getCloudTestEndChecker(t, 132, configOverride, cloudapi.RunStatusFinished, cloudapi.ResultStatusPassed) ts.Env["K6_CLOUD_HOST"] = srv.URL cmd.ExecuteWithGlobalState(ts.GlobalState) @@ -1601,7 +1600,7 @@ func TestRunWithCloudOutputMoreOverrides(t *testing.T) { }`) assert.NoError(t, err) }) - srv := getCloudTestEndChecker(t, 1337, configOverride, lib.RunStatusFinished, cloudapi.ResultStatusPassed) + srv := getCloudTestEndChecker(t, 1337, configOverride, cloudapi.RunStatusFinished, cloudapi.ResultStatusPassed) ts.Env["K6_CLOUD_HOST"] = srv.URL cmd.ExecuteWithGlobalState(ts.GlobalState) diff --git a/core/engine.go b/core/engine.go index ed98d744cbf..5e31a567a4f 100644 --- a/core/engine.go +++ b/core/engine.go @@ -107,7 +107,6 @@ func (e *Engine) Init(globalCtx, runCtx context.Context) (run func() error, wait // TODO: if we ever need metrics processing in the init context, we can move // this below the other components... or even start them concurrently? if err := e.ExecutionScheduler.Init(runCtx, e.Samples); err != nil { - e.setRunStatusFromError(err) return nil, nil, err } @@ -147,18 +146,6 @@ func (e *Engine) Init(globalCtx, runCtx context.Context) (run func() error, wait return runFn, waitFn, nil } -func (e *Engine) setRunStatusFromError(err error) { - var serr errext.Exception - switch { - case errors.As(err, &serr): - e.OutputManager.SetRunStatus(lib.RunStatusAbortedScriptError) - case errext.IsInterruptError(err): - e.OutputManager.SetRunStatus(lib.RunStatusAbortedUser) - default: - e.OutputManager.SetRunStatus(lib.RunStatusAbortedSystem) - } -} - // This starts a bunch of goroutines to process metrics, thresholds, and set the // test run status when it ends. It returns a function that can be used after // the provided context is called, to wait for the complete winding down of all @@ -195,27 +182,31 @@ func (e *Engine) startBackgroundProcesses( case err = <-execRunResult: if err != nil { e.logger.WithError(err).Debug("run: execution scheduler returned an error") - e.setRunStatusFromError(err) } else { e.logger.Debug("run: execution scheduler finished nominally") - e.OutputManager.SetRunStatus(lib.RunStatusFinished) } // do nothing, return the same err value we got from the Run() // ExecutionScheduler result, we just set the run_status based on it case <-runCtx.Done(): e.logger.Debug("run: context expired; exiting...") - e.OutputManager.SetRunStatus(lib.RunStatusAbortedUser) - err = errext.WithExitCodeIfNone(errors.New("test run aborted by signal"), exitcodes.ExternalAbort) + err = errext.WithAbortReasonIfNone( + errext.WithExitCodeIfNone(errors.New("test run aborted by signal"), exitcodes.ExternalAbort), + errext.AbortedByUser, + ) case <-e.stopChan: runSubCancel() e.logger.Debug("run: stopped by user via REST API; exiting...") - e.OutputManager.SetRunStatus(lib.RunStatusAbortedUser) - err = errext.WithExitCodeIfNone(errors.New("test run stopped from the REST API"), exitcodes.ScriptStoppedFromRESTAPI) + err = errext.WithAbortReasonIfNone( + errext.WithExitCodeIfNone(errors.New("test run stopped from the REST API"), exitcodes.ScriptStoppedFromRESTAPI), + errext.AbortedByUser, + ) case <-thresholdAbortChan: e.logger.Debug("run: stopped by thresholds; exiting...") runSubCancel() - e.OutputManager.SetRunStatus(lib.RunStatusAbortedThreshold) - err = errext.WithExitCodeIfNone(errors.New("test run aborted by failed thresholds"), exitcodes.ThresholdsHaveFailed) + err = errext.WithAbortReasonIfNone( + errext.WithExitCodeIfNone(errors.New("test run aborted by failed thresholds"), exitcodes.ThresholdsHaveFailed), + errext.AbortedByThreshold, + ) } }() diff --git a/core/engine_test.go b/core/engine_test.go index fb22d048df3..d06962dd242 100644 --- a/core/engine_test.go +++ b/core/engine_test.go @@ -109,7 +109,7 @@ func newTestEngineWithTestPreInitState( //nolint:golint test.runCancel() globalCancel() waitFn() - engine.OutputManager.StopOutputs() + engine.OutputManager.StopOutputs(nil) }, piState: piState, } @@ -1333,7 +1333,7 @@ func TestActiveVUsCount(t *testing.T) { require.NoError(t, err) cancel() waitFn() - engine.OutputManager.StopOutputs() + engine.OutputManager.StopOutputs(nil) require.False(t, engine.IsTainted()) } diff --git a/errext/abort_reason.go b/errext/abort_reason.go new file mode 100644 index 00000000000..a9452f92be7 --- /dev/null +++ b/errext/abort_reason.go @@ -0,0 +1,54 @@ +package errext + +import "errors" + +// AbortReason is used to signal to outputs what type of error caused the test +// run to be stopped prematurely. +type AbortReason uint8 + +// These are the various reasons why a test might have been aborted prematurely. +const ( + AbortedByUser AbortReason = iota + 1 + AbortedByThreshold + AbortedByThresholdsAfterTestEnd // TODO: rename? + AbortedByScriptError + AbortedByScriptAbort + AbortedByTimeout +) + +// HasAbortReason is a wrapper around an error with an attached abort reason. +type HasAbortReason interface { + error + AbortReason() AbortReason +} + +// WithAbortReasonIfNone can attach an abort reason to the given error, if it +// doesn't have one already. It won't do anything if the error already had an +// abort reason attached. Similarly, if there is no error (i.e. the given error +// is nil), it also won't do anything and will return nil. +func WithAbortReasonIfNone(err error, abortReason AbortReason) error { + if err == nil { + return nil // No error, do nothing + } + var arerr HasAbortReason + if errors.As(err, &arerr) { + // The given error already has an abort reason, do nothing + return err + } + return withAbortReason{err, abortReason} +} + +type withAbortReason struct { + error + abortReason AbortReason +} + +func (ar withAbortReason) Unwrap() error { + return ar.error +} + +func (ar withAbortReason) AbortReason() AbortReason { + return ar.abortReason +} + +var _ HasAbortReason = withAbortReason{} diff --git a/errext/exception.go b/errext/exception.go index 77d33f4afeb..cce7260de04 100644 --- a/errext/exception.go +++ b/errext/exception.go @@ -5,5 +5,6 @@ package errext // a stack trace that lead to them. type Exception interface { error + HasAbortReason StackTrace() string } diff --git a/errext/interrupt_error.go b/errext/interrupt_error.go index 637622daed2..77a6e649163 100644 --- a/errext/interrupt_error.go +++ b/errext/interrupt_error.go @@ -11,7 +11,10 @@ type InterruptError struct { Reason string } -var _ HasExitCode = &InterruptError{} +var _ interface { + HasExitCode + HasAbortReason +} = &InterruptError{} // Error returns the reason of the interruption. func (i *InterruptError) Error() string { @@ -23,6 +26,12 @@ func (i *InterruptError) ExitCode() exitcodes.ExitCode { return exitcodes.ScriptAborted } +// AbortReason is used to signal that an InterruptError is caused by the +// test.abort() functin in k6/execution. +func (i *InterruptError) AbortReason() AbortReason { + return AbortedByScriptAbort +} + // AbortTest is the reason emitted when a test script calls test.abort() const AbortTest = "test aborted" diff --git a/js/runner.go b/js/runner.go index a8bd4a0a9a9..edb1d43c648 100644 --- a/js/runner.go +++ b/js/runner.go @@ -845,11 +845,12 @@ type scriptException struct { inner *goja.Exception } -var ( - _ errext.Exception = &scriptException{} - _ errext.HasExitCode = &scriptException{} - _ errext.HasHint = &scriptException{} -) +var _ interface { + errext.Exception + errext.HasExitCode + errext.HasHint + errext.HasAbortReason +} = &scriptException{} func (s *scriptException) Error() string { // this calls String instead of error so that by default if it's printed to print the stacktrace @@ -868,6 +869,10 @@ func (s *scriptException) Hint() string { return "script exception" } +func (s *scriptException) AbortReason() errext.AbortReason { + return errext.AbortedByScriptError +} + func (s *scriptException) ExitCode() exitcodes.ExitCode { return exitcodes.ScriptException } diff --git a/js/runner_test.go b/js/runner_test.go index 00abf47ca65..c794ac3a1c0 100644 --- a/js/runner_test.go +++ b/js/runner_test.go @@ -391,7 +391,7 @@ func TestDataIsolation(t *testing.T) { engine, err := core.NewEngine(testRunState, execScheduler, []output.Output{mockOutput}) require.NoError(t, err) require.NoError(t, engine.OutputManager.StartOutputs()) - defer engine.OutputManager.StopOutputs() + defer engine.OutputManager.StopOutputs(nil) ctx, cancel := context.WithCancel(context.Background()) run, wait, err := engine.Init(ctx, ctx) diff --git a/js/timeout_error.go b/js/timeout_error.go index 72cc1d7aa79..e887b0922b2 100644 --- a/js/timeout_error.go +++ b/js/timeout_error.go @@ -15,10 +15,11 @@ type timeoutError struct { d time.Duration } -var ( - _ errext.HasExitCode = timeoutError{} - _ errext.HasHint = timeoutError{} -) +var _ interface { + errext.HasExitCode + errext.HasHint + errext.HasAbortReason +} = timeoutError{} // newTimeoutError returns a new timeout error, reporting that a timeout has // happened at the given place and given duration. @@ -44,6 +45,10 @@ func (t timeoutError) Hint() string { return hint } +func (t timeoutError) AbortReason() errext.AbortReason { + return errext.AbortedByTimeout +} + // ExitCode returns the coresponding exit code value to the place. func (t timeoutError) ExitCode() exitcodes.ExitCode { // TODO: add handleSummary() diff --git a/lib/testutils/mockoutput/mockoutput.go b/lib/testutils/mockoutput/mockoutput.go index f99b128d080..6d73123eb63 100644 --- a/lib/testutils/mockoutput/mockoutput.go +++ b/lib/testutils/mockoutput/mockoutput.go @@ -1,9 +1,7 @@ package mockoutput import ( - "go.k6.io/k6/lib" "go.k6.io/k6/metrics" - "go.k6.io/k6/output" ) // New exists so that the usage from tests avoids repetition, i.e. is @@ -16,15 +14,12 @@ func New() *MockOutput { type MockOutput struct { SampleContainers []metrics.SampleContainer Samples []metrics.Sample - RunStatus lib.RunStatus DescFn func() string StartFn func() error StopFn func() error } -var _ output.WithRunStatusUpdates = &MockOutput{} - // AddMetricSamples just saves the results in memory. func (mo *MockOutput) AddMetricSamples(scs []metrics.SampleContainer) { mo.SampleContainers = append(mo.SampleContainers, scs...) @@ -33,11 +28,6 @@ func (mo *MockOutput) AddMetricSamples(scs []metrics.SampleContainer) { } } -// SetRunStatus updates the RunStatus property. -func (mo *MockOutput) SetRunStatus(latestStatus lib.RunStatus) { - mo.RunStatus = latestStatus -} - // Description calls the supplied DescFn callback, if available. func (mo *MockOutput) Description() string { if mo.DescFn != nil { diff --git a/output/cloud/output.go b/output/cloud/output.go index ee98b193b4d..f05ad1d530d 100644 --- a/output/cloud/output.go +++ b/output/cloud/output.go @@ -14,6 +14,7 @@ import ( "gopkg.in/guregu/null.v3" "go.k6.io/k6/cloudapi" + "go.k6.io/k6/errext" "go.k6.io/k6/output" "go.k6.io/k6/lib" @@ -36,8 +37,6 @@ type Output struct { thresholds map[string][]*metrics.Threshold client *MetricsClient - runStatus lib.RunStatus - bufferMutex sync.Mutex bufferHTTPTrails []*httpext.Trail bufferSamples []*Sample @@ -65,7 +64,7 @@ type Output struct { // Verify that Output implements the wanted interfaces var _ interface { - output.WithRunStatusUpdates + output.WithStopWithTestError output.WithThresholds output.WithTestRunStop } = &Output{} @@ -259,9 +258,18 @@ func (out *Output) startBackgroundProcesses() { }() } -// Stop gracefully stops all metric emission from the output and when all metric -// samples are emitted, it sends an API to the cloud to finish the test run. +// Stop gracefully stops all metric emission from the output: when all metric +// samples are emitted, it makes a cloud API call to finish the test run. +// +// Deprecated: use StopWithTestError() instead. func (out *Output) Stop() error { + return out.StopWithTestError(nil) +} + +// StopWithTestError gracefully stops all metric emission from the output: when +// all metric samples are emitted, it makes a cloud API call to finish the test +// run. If testErr was specified, it extracts the RunStatus from it. +func (out *Output) StopWithTestError(testErr error) error { out.logger.Debug("Stopping the cloud output...") close(out.stopAggregation) out.aggregationDone.Wait() // could be a no-op, if we have never started the aggregation @@ -269,7 +277,7 @@ func (out *Output) Stop() error { close(out.stopOutput) out.outputDone.Wait() out.logger.Debug("Metric emission stopped, calling cloud API...") - err := out.testFinished() + err := out.testFinished(testErr) if err != nil { out.logger.WithFields(logrus.Fields{"error": err}).Warn("Failed to send test finished to the cloud") } else { @@ -283,9 +291,51 @@ func (out *Output) Description() string { return fmt.Sprintf("cloud (%s)", cloudapi.URLForResults(out.referenceID, out.config)) } -// SetRunStatus receives the latest run status from the Engine. -func (out *Output) SetRunStatus(status lib.RunStatus) { - out.runStatus = status +// getRunStatus determines the run status of the test based on the error. +func (out *Output) getRunStatus(testErr error) cloudapi.RunStatus { + if testErr == nil { + return cloudapi.RunStatusFinished + } + + var err errext.HasAbortReason + if errors.As(testErr, &err) { + abortReason := err.AbortReason() + switch abortReason { + case errext.AbortedByUser: + return cloudapi.RunStatusAbortedUser + case errext.AbortedByThreshold: + return cloudapi.RunStatusAbortedThreshold + case errext.AbortedByScriptError: + return cloudapi.RunStatusAbortedScriptError + case errext.AbortedByScriptAbort: + return cloudapi.RunStatusAbortedUser // TODO: have a better value than this? + case errext.AbortedByTimeout: + return cloudapi.RunStatusAbortedLimit + case errext.AbortedByThresholdsAfterTestEnd: + // The test run finished normally, it wasn't prematurely aborted by + // anything while running, but the thresholds failed at the end and + // k6 will return an error and a non-zero exit code to the user. + // + // However, failures are tracked somewhat differently by the k6 + // cloud compared to k6 OSS. It doesn't have a single pass/fail + // variable with multiple failure states, like k6's exit codes. + // Instead, it has two variables, result_status and run_status. + // + // The status of the thresholds is tracked by the binary + // result_status variable, which signifies whether the thresholds + // passed or failed (failure also called "tainted" in some places of + // the API here). The run_status signifies whether the test run + // finished normally and has a few fixed failures values. + // + // So, this specific k6 error will be communicated to the cloud only + // via result_status, while the run_status will appear normal. + return cloudapi.RunStatusFinished + } + } + + // By default, the catch-all error is "aborted by system", but let's log that + out.logger.WithError(testErr).Debug("unknown test error classified as 'aborted by system'") + return cloudapi.RunStatusAbortedSystem } // SetThresholds receives the thresholds before the output is Start()-ed. @@ -617,7 +667,7 @@ func (out *Output) pushMetrics() { }).Debug("Pushing metrics to cloud finished") } -func (out *Output) testFinished() error { +func (out *Output) testFinished(testErr error) error { if out.referenceID == "" || out.config.PushRefID.Valid { return nil } @@ -634,11 +684,7 @@ func (out *Output) testFinished() error { } } - runStatus := lib.RunStatusFinished - if out.runStatus != lib.RunStatusQueued { - runStatus = out.runStatus - } - + runStatus := out.getRunStatus(testErr) out.logger.WithFields(logrus.Fields{ "ref": out.referenceID, "tainted": testTainted, diff --git a/output/cloud/output_test.go b/output/cloud/output_test.go index 86a0778b181..7b16e03a1f7 100644 --- a/output/cloud/output_test.go +++ b/output/cloud/output_test.go @@ -506,7 +506,6 @@ func testCloudOutputStopSendingMetric(t *testing.T, stopOnError bool) { require.NoError(t, out.Stop()) - require.Equal(t, lib.RunStatusQueued, out.runStatus) select { case <-out.stopSendingMetrics: // all is fine @@ -598,7 +597,6 @@ func TestCloudOutputAggregationPeriodZeroNoBlock(t *testing.T) { tb.Mux.HandleFunc(fmt.Sprintf("/v1/metrics/%s", out.referenceID), getSampleChecker(t, expSamples)) require.NoError(t, out.Stop()) - require.Equal(t, lib.RunStatusQueued, out.runStatus) } func TestCloudOutputPushRefID(t *testing.T) { diff --git a/output/manager.go b/output/manager.go index 38d72e53b78..4d041dd346c 100644 --- a/output/manager.go +++ b/output/manager.go @@ -2,7 +2,6 @@ package output import ( "github.com/sirupsen/logrus" - "go.k6.io/k6/lib" "go.k6.io/k6/metrics" ) @@ -35,7 +34,7 @@ func (om *Manager) StartOutputs() error { } if err := out.Start(); err != nil { - om.stopOutputs(i) + om.stopOutputs(err, i) return err } } @@ -43,25 +42,23 @@ func (om *Manager) StartOutputs() error { } // StopOutputs stops all configured outputs. -func (om *Manager) StopOutputs() { - om.stopOutputs(len(om.outputs)) +func (om *Manager) StopOutputs(testErr error) { + om.stopOutputs(testErr, len(om.outputs)) } -func (om *Manager) stopOutputs(upToID int) { +func (om *Manager) stopOutputs(testErr error, upToID int) { om.logger.Debugf("Stopping %d outputs...", upToID) for i := 0; i < upToID; i++ { - if err := om.outputs[i].Stop(); err != nil { - om.logger.WithError(err).Errorf("Stopping output %d failed", i) + out := om.outputs[i] + var err error + if sout, ok := out.(WithStopWithTestError); ok { + err = sout.StopWithTestError(testErr) + } else { + err = out.Stop() } - } -} -// SetRunStatus checks which outputs implement the WithRunStatusUpdates -// interface and sets the provided RunStatus to them. -func (om *Manager) SetRunStatus(status lib.RunStatus) { - for _, out := range om.outputs { - if statUpdOut, ok := out.(WithRunStatusUpdates); ok { - statUpdOut.SetRunStatus(status) + if err != nil { + om.logger.WithError(err).Errorf("Stopping output %d failed", i) } } } diff --git a/output/types.go b/output/types.go index 09885d03566..646bacc8589 100644 --- a/output/types.go +++ b/output/types.go @@ -75,10 +75,17 @@ type WithTestRunStop interface { SetTestRunStopCallback(func(error)) } -// WithRunStatusUpdates means the output can receive test run status updates. -type WithRunStatusUpdates interface { +// WithStopWithTestError allows output to receive the error value that the test +// finished with. It could be nil, if the test finished nominally. +// +// If this interface is implemented by the output, StopWithError() will be +// called instead of Stop(). +// +// TODO: refactor the main interface to use this method instead of Stop()? Or +// something else along the lines of https://github.com/grafana/k6/issues/2430 ? +type WithStopWithTestError interface { Output - SetRunStatus(latestStatus lib.RunStatus) + StopWithTestError(testRunErr error) error // nil testRunErr means error-free test run } // WithBuiltinMetrics means the output can receive the builtin metrics.