mirror of
https://github.com/microsoft/playwright.git
synced 2025-01-07 11:46:42 +03:00
chore(test runner): remove fake skipped test results (#27762)
Fixes #27455.
This commit is contained in:
parent
c33b41df77
commit
210168e36d
@ -239,6 +239,9 @@ export class TestCase extends Base implements reporterTypes.TestCase {
|
|||||||
_poolDigest = '';
|
_poolDigest = '';
|
||||||
_workerHash = '';
|
_workerHash = '';
|
||||||
_projectId = '';
|
_projectId = '';
|
||||||
|
// This is different from |results.length| because sometimes we do not run the test, but consume
|
||||||
|
// an attempt, for example when skipping tests in a serial suite after a failure.
|
||||||
|
_runAttempts = 0;
|
||||||
// Annotations known statically before running the test, e.g. `test.skip()` or `test.describe.skip()`.
|
// Annotations known statically before running the test, e.g. `test.skip()` or `test.describe.skip()`.
|
||||||
_staticAnnotations: Annotation[] = [];
|
_staticAnnotations: Annotation[] = [];
|
||||||
|
|
||||||
@ -256,16 +259,10 @@ export class TestCase extends Base implements reporterTypes.TestCase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
outcome(): 'skipped' | 'expected' | 'unexpected' | 'flaky' {
|
outcome(): 'skipped' | 'expected' | 'unexpected' | 'flaky' {
|
||||||
// Ignore initial skips that may be a result of "skipped because previous test in serial mode failed".
|
const results = this.results.filter(result => result.status !== 'interrupted');
|
||||||
const results = [...this.results];
|
if (results.every(result => result.status === 'skipped'))
|
||||||
while (results[0]?.status === 'skipped' || results[0]?.status === 'interrupted')
|
|
||||||
results.shift();
|
|
||||||
|
|
||||||
// All runs were skipped.
|
|
||||||
if (!results.length)
|
|
||||||
return 'skipped';
|
return 'skipped';
|
||||||
|
const failures = results.filter(result => result.status !== this.expectedStatus);
|
||||||
const failures = results.filter(result => result.status !== 'skipped' && result.status !== 'interrupted' && result.status !== this.expectedStatus);
|
|
||||||
if (!failures.length) // all passed
|
if (!failures.length) // all passed
|
||||||
return 'expected';
|
return 'expected';
|
||||||
if (failures.length === results.length) // all failed
|
if (failures.length === results.length) // all failed
|
||||||
|
@ -493,16 +493,10 @@ export class TeleTestCase implements reporterTypes.TestCase {
|
|||||||
}
|
}
|
||||||
|
|
||||||
outcome(): 'skipped' | 'expected' | 'unexpected' | 'flaky' {
|
outcome(): 'skipped' | 'expected' | 'unexpected' | 'flaky' {
|
||||||
// Ignore initial skips that may be a result of "skipped because previous test in serial mode failed".
|
const results = this.results.filter(result => result.status !== 'interrupted');
|
||||||
const results = [...this.results];
|
if (results.every(result => result.status === 'skipped'))
|
||||||
while (results[0]?.status === 'skipped' || results[0]?.status === 'interrupted')
|
|
||||||
results.shift();
|
|
||||||
|
|
||||||
// All runs were skipped.
|
|
||||||
if (!results.length)
|
|
||||||
return 'skipped';
|
return 'skipped';
|
||||||
|
const failures = results.filter(result => result.status !== this.expectedStatus);
|
||||||
const failures = results.filter(result => result.status !== 'skipped' && result.status !== 'interrupted' && result.status !== this.expectedStatus);
|
|
||||||
if (!failures.length) // all passed
|
if (!failures.length) // all passed
|
||||||
return 'expected';
|
return 'expected';
|
||||||
if (failures.length === results.length) // all failed
|
if (failures.length === results.length) // all failed
|
||||||
|
@ -290,7 +290,7 @@ class JobDispatcher {
|
|||||||
test.expectedStatus = params.expectedStatus;
|
test.expectedStatus = params.expectedStatus;
|
||||||
test.annotations = params.annotations;
|
test.annotations = params.annotations;
|
||||||
test.timeout = params.timeout;
|
test.timeout = params.timeout;
|
||||||
const isFailure = result.status !== 'skipped' && result.status !== test.expectedStatus;
|
const isFailure = result.status !== test.expectedStatus;
|
||||||
if (isFailure)
|
if (isFailure)
|
||||||
this._failedTests.add(test);
|
this._failedTests.add(test);
|
||||||
this._reportTestEnd(test, result);
|
this._reportTestEnd(test, result);
|
||||||
@ -369,22 +369,17 @@ class JobDispatcher {
|
|||||||
}
|
}
|
||||||
result.errors = [...errors];
|
result.errors = [...errors];
|
||||||
result.error = result.errors[0];
|
result.error = result.errors[0];
|
||||||
result.status = errors.length ? 'failed' : 'skipped';
|
result.status = 'failed';
|
||||||
this._reportTestEnd(test, result);
|
this._reportTestEnd(test, result);
|
||||||
this._failedTests.add(test);
|
this._failedTests.add(test);
|
||||||
}
|
}
|
||||||
|
|
||||||
private _massSkipTestsFromRemaining(testIds: Set<string>, errors: TestError[]) {
|
private _handleFatalErrors(errors: TestError[]) {
|
||||||
for (const test of this._remainingByTestId.values()) {
|
const test = this._remainingByTestId.values().next().value as TestCase | undefined;
|
||||||
if (!testIds.has(test.id))
|
if (test) {
|
||||||
continue;
|
this._failTestWithErrors(test, errors);
|
||||||
if (!this._failureTracker.hasReachedMaxFailures()) {
|
|
||||||
this._failTestWithErrors(test, errors);
|
|
||||||
errors = []; // Only report errors for the first test.
|
|
||||||
}
|
|
||||||
this._remainingByTestId.delete(test.id);
|
this._remainingByTestId.delete(test.id);
|
||||||
}
|
} else if (errors.length) {
|
||||||
if (errors.length) {
|
|
||||||
// We had fatal errors after all tests have passed - most likely in some teardown.
|
// We had fatal errors after all tests have passed - most likely in some teardown.
|
||||||
// Let's just fail the test run.
|
// Let's just fail the test run.
|
||||||
this._failureTracker.onWorkerError();
|
this._failureTracker.onWorkerError();
|
||||||
@ -412,23 +407,28 @@ class JobDispatcher {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (params.fatalErrors.length) {
|
if (params.fatalErrors.length) {
|
||||||
// In case of fatal errors, report first remaining test as failing with these errors,
|
// In case of fatal errors, report the first remaining test as failing with these errors,
|
||||||
// and all others as skipped.
|
// and "skip" all other tests to avoid running into the same issue over and over.
|
||||||
this._massSkipTestsFromRemaining(new Set(this._remainingByTestId.keys()), params.fatalErrors);
|
this._handleFatalErrors(params.fatalErrors);
|
||||||
|
this._remainingByTestId.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle tests that should be skipped because of the setup failure.
|
// Handle tests that should be skipped because of the setup failure.
|
||||||
this._massSkipTestsFromRemaining(new Set(params.skipTestsDueToSetupFailure), []);
|
for (const testId of params.skipTestsDueToSetupFailure)
|
||||||
|
this._remainingByTestId.delete(testId);
|
||||||
|
|
||||||
if (params.unexpectedExitError) {
|
if (params.unexpectedExitError) {
|
||||||
// When worker exits during a test, we blame the test itself.
|
if (this._currentlyRunning) {
|
||||||
//
|
// When worker exits during a test, we blame the test itself.
|
||||||
// The most common situation when worker exits while not running a test is:
|
this._failTestWithErrors(this._currentlyRunning.test, [params.unexpectedExitError]);
|
||||||
// worker failed to require the test file (at the start) because of an exception in one of imports.
|
this._remainingByTestId.delete(this._currentlyRunning.test.id);
|
||||||
// In this case, "skip" all remaining tests, to avoid running into the same exception over and over.
|
} else {
|
||||||
if (this._currentlyRunning)
|
// The most common situation when worker exits while not running a test is:
|
||||||
this._massSkipTestsFromRemaining(new Set([this._currentlyRunning.test.id]), [params.unexpectedExitError]);
|
// worker failed to require the test file (at the start) because of an exception in one of imports.
|
||||||
else
|
// In this case, "skip" all remaining tests, to avoid running into the same exception over and over.
|
||||||
this._massSkipTestsFromRemaining(new Set(this._remainingByTestId.keys()), [params.unexpectedExitError]);
|
this._handleFatalErrors([params.unexpectedExitError]);
|
||||||
|
this._remainingByTestId.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const retryCandidates = new Set<TestCase>();
|
const retryCandidates = new Set<TestCase>();
|
||||||
@ -446,26 +446,22 @@ class JobDispatcher {
|
|||||||
serialSuitesWithFailures.add(outermostSerialSuite);
|
serialSuitesWithFailures.add(outermostSerialSuite);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we have failed tests that belong to a serial suite,
|
|
||||||
// we should skip all future tests from the same serial suite.
|
|
||||||
const testsBelongingToSomeSerialSuiteWithFailures = [...this._remainingByTestId.values()].filter(test => {
|
|
||||||
let parent: Suite | undefined = test.parent;
|
|
||||||
while (parent && !serialSuitesWithFailures.has(parent))
|
|
||||||
parent = parent.parent;
|
|
||||||
return !!parent;
|
|
||||||
});
|
|
||||||
this._massSkipTestsFromRemaining(new Set(testsBelongingToSomeSerialSuiteWithFailures.map(test => test.id)), []);
|
|
||||||
|
|
||||||
for (const serialSuite of serialSuitesWithFailures) {
|
for (const serialSuite of serialSuitesWithFailures) {
|
||||||
// Add all tests from failed serial suites for possible retry.
|
serialSuite.allTests().forEach(test => {
|
||||||
// These will only be retried together, because they have the same
|
// Skip remaining tests from serial suites with failures.
|
||||||
// "retries" setting and the same number of previous runs.
|
this._remainingByTestId.delete(test.id);
|
||||||
serialSuite.allTests().forEach(test => retryCandidates.add(test));
|
// Schedule them for the retry all together.
|
||||||
|
retryCandidates.add(test);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (const test of this._job.tests) {
|
||||||
|
if (!this._remainingByTestId.has(test.id))
|
||||||
|
++test._runAttempts;
|
||||||
|
}
|
||||||
const remaining = [...this._remainingByTestId.values()];
|
const remaining = [...this._remainingByTestId.values()];
|
||||||
for (const test of retryCandidates) {
|
for (const test of retryCandidates) {
|
||||||
if (test.results.length < test.retries + 1)
|
if (test._runAttempts < test.retries + 1)
|
||||||
remaining.push(test);
|
remaining.push(test);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@ export class FailureTracker {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onTestEnd(test: TestCase, result: TestResult) {
|
onTestEnd(test: TestCase, result: TestResult) {
|
||||||
if (result.status !== 'skipped' && result.status !== test.expectedStatus)
|
if (result.status !== test.expectedStatus)
|
||||||
++this._failureCount;
|
++this._failureCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -399,7 +399,7 @@ test('beforeAll failure should prevent the test, but not afterAll', async ({ run
|
|||||||
});
|
});
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'beforeAll',
|
'beforeAll',
|
||||||
'afterAll',
|
'afterAll',
|
||||||
@ -499,7 +499,7 @@ test('beforeAll timeout should be reported and prevent more tests', async ({ run
|
|||||||
}, { timeout: 1000 });
|
}, { timeout: 1000 });
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'beforeAll',
|
'beforeAll',
|
||||||
'afterAll',
|
'afterAll',
|
||||||
@ -688,7 +688,7 @@ test('unhandled rejection during beforeAll should be reported and prevent more t
|
|||||||
});
|
});
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'beforeAll',
|
'beforeAll',
|
||||||
'afterAll',
|
'afterAll',
|
||||||
@ -801,7 +801,7 @@ test('beforeAll failure should only prevent tests that are affected', async ({ r
|
|||||||
});
|
});
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.passed).toBe(1);
|
expect(result.passed).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'beforeAll',
|
'beforeAll',
|
||||||
|
@ -216,8 +216,8 @@ test('should retry beforeAll failure', async ({ runInlineTest }) => {
|
|||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(0);
|
expect(result.passed).toBe(0);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.output.split('\n')[2]).toBe('×°×°F°');
|
expect(result.output.split('\n')[2]).toBe('××F');
|
||||||
expect(result.output).toContain('BeforeAll is bugged!');
|
expect(result.output).toContain('BeforeAll is bugged!');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -111,7 +111,7 @@ test('should report subprocess creation error', async ({ runInlineTest }, testIn
|
|||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(0);
|
expect(result.passed).toBe(0);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.output).toContain('Error: worker process exited unexpectedly (code=42, signal=null)');
|
expect(result.output).toContain('Error: worker process exited unexpectedly (code=42, signal=null)');
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -634,9 +634,9 @@ test('should not hang on worker error in test file', async ({ runInlineTest }) =
|
|||||||
`,
|
`,
|
||||||
}, { 'timeout': 3000 });
|
}, { 'timeout': 3000 });
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
|
expect(result.results).toHaveLength(1);
|
||||||
expect(result.results[0].status).toBe('failed');
|
expect(result.results[0].status).toBe('failed');
|
||||||
expect(result.results[0].error.message).toContain('Error: worker process exited unexpectedly');
|
expect(result.results[0].error.message).toContain('Error: worker process exited unexpectedly');
|
||||||
expect(result.results[1].status).toBe('skipped');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('fast double SIGINT should be ignored', async ({ interactWithTestRunner }) => {
|
test('fast double SIGINT should be ignored', async ({ interactWithTestRunner }) => {
|
||||||
|
@ -640,7 +640,7 @@ test('static modifiers should be added in serial mode', async ({ runInlineTest }
|
|||||||
});
|
});
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(0);
|
expect(result.passed).toBe(0);
|
||||||
expect(result.skipped).toBe(3);
|
expect(result.didNotRun).toBe(3);
|
||||||
expect(result.report.suites[0].specs[0].tests[0].annotations).toEqual([{ type: 'slow' }]);
|
expect(result.report.suites[0].specs[0].tests[0].annotations).toEqual([{ type: 'slow' }]);
|
||||||
expect(result.report.suites[0].specs[1].tests[0].annotations).toEqual([{ type: 'fixme' }]);
|
expect(result.report.suites[0].specs[1].tests[0].annotations).toEqual([{ type: 'fixme' }]);
|
||||||
expect(result.report.suites[0].specs[2].tests[0].annotations).toEqual([{ type: 'skip' }]);
|
expect(result.report.suites[0].specs[2].tests[0].annotations).toEqual([{ type: 'skip' }]);
|
||||||
|
@ -47,7 +47,7 @@ test('test.describe.serial should work', async ({ runInlineTest }) => {
|
|||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(2);
|
expect(result.passed).toBe(2);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(2);
|
expect(result.didNotRun).toBe(2);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'test1',
|
'test1',
|
||||||
'test2',
|
'test2',
|
||||||
@ -87,7 +87,7 @@ test('test.describe.serial should work in describe', async ({ runInlineTest }) =
|
|||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(2);
|
expect(result.passed).toBe(2);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(2);
|
expect(result.didNotRun).toBe(2);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'test1',
|
'test1',
|
||||||
'test2',
|
'test2',
|
||||||
@ -128,7 +128,7 @@ test('test.describe.serial should work with retry', async ({ runInlineTest }) =>
|
|||||||
expect(result.passed).toBe(2);
|
expect(result.passed).toBe(2);
|
||||||
expect(result.flaky).toBe(1);
|
expect(result.flaky).toBe(1);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'test1',
|
'test1',
|
||||||
'test2',
|
'test2',
|
||||||
@ -272,7 +272,7 @@ test('test.describe.serial should work with test.fail', async ({ runInlineTest }
|
|||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
expect(result.passed).toBe(2);
|
expect(result.passed).toBe(2);
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.skipped).toBe(1);
|
expect(result.didNotRun).toBe(1);
|
||||||
expect(result.outputLines).toEqual([
|
expect(result.outputLines).toEqual([
|
||||||
'zero',
|
'zero',
|
||||||
'one',
|
'one',
|
||||||
@ -394,3 +394,55 @@ test('test.describe.serial should work with fullyParallel', async ({ runInlineTe
|
|||||||
'two',
|
'two',
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('serial fail + skip is failed', async ({ runInlineTest }) => {
|
||||||
|
const result = await runInlineTest({
|
||||||
|
'a.test.ts': `
|
||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
test.describe.configure({ mode: 'serial', retries: 1 });
|
||||||
|
test.describe.serial('serial suite', () => {
|
||||||
|
test('one', async () => {
|
||||||
|
expect(test.info().retry).toBe(0);
|
||||||
|
});
|
||||||
|
test('two', async () => {
|
||||||
|
expect(1).toBe(2);
|
||||||
|
});
|
||||||
|
test('three', async () => {
|
||||||
|
});
|
||||||
|
});
|
||||||
|
`,
|
||||||
|
}, { workers: 1 });
|
||||||
|
expect(result.exitCode).toBe(1);
|
||||||
|
expect(result.passed).toBe(0);
|
||||||
|
expect(result.skipped).toBe(0);
|
||||||
|
expect(result.flaky).toBe(1);
|
||||||
|
expect(result.failed).toBe(1);
|
||||||
|
expect(result.interrupted).toBe(0);
|
||||||
|
expect(result.didNotRun).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('serial skip + fail is failed', async ({ runInlineTest }) => {
|
||||||
|
const result = await runInlineTest({
|
||||||
|
'a.test.ts': `
|
||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
test.describe.configure({ mode: 'serial', retries: 1 });
|
||||||
|
test.describe.serial('serial suite', () => {
|
||||||
|
test('one', async () => {
|
||||||
|
expect(test.info().retry).toBe(1);
|
||||||
|
});
|
||||||
|
test('two', async () => {
|
||||||
|
expect(1).toBe(2);
|
||||||
|
});
|
||||||
|
test('three', async () => {
|
||||||
|
});
|
||||||
|
});
|
||||||
|
`,
|
||||||
|
}, { workers: 1 });
|
||||||
|
expect(result.exitCode).toBe(1);
|
||||||
|
expect(result.passed).toBe(0);
|
||||||
|
expect(result.skipped).toBe(0);
|
||||||
|
expect(result.flaky).toBe(1);
|
||||||
|
expect(result.failed).toBe(1);
|
||||||
|
expect(result.interrupted).toBe(0);
|
||||||
|
expect(result.didNotRun).toBe(1);
|
||||||
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user