chore: add a deflake utility (#11111)
This commit is contained in:
parent
e34716ab0e
commit
4a2a37b825
@ -20,11 +20,16 @@ import type {ProtocolMapping} from 'devtools-protocol/types/protocol-mapping.js'
|
|||||||
|
|
||||||
import type {CDPEvents, CDPSession} from '../api/CDPSession.js';
|
import type {CDPEvents, CDPSession} from '../api/CDPSession.js';
|
||||||
import type {Connection as CdpConnection} from '../cdp/Connection.js';
|
import type {Connection as CdpConnection} from '../cdp/Connection.js';
|
||||||
|
import {debug} from '../common/Debug.js';
|
||||||
import {TargetCloseError} from '../common/Errors.js';
|
import {TargetCloseError} from '../common/Errors.js';
|
||||||
import type {Handler} from '../common/EventEmitter.js';
|
import type {Handler} from '../common/EventEmitter.js';
|
||||||
|
|
||||||
import {BidiConnection} from './Connection.js';
|
import {BidiConnection} from './Connection.js';
|
||||||
|
|
||||||
|
const bidiServerLogger = (prefix: string, ...args: unknown[]): void => {
|
||||||
|
debug(`bidi:${prefix}`)(args);
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
@ -54,7 +59,9 @@ export async function connectBidiOverCdp(
|
|||||||
const bidiServer = await BidiMapper.BidiServer.createAndStart(
|
const bidiServer = await BidiMapper.BidiServer.createAndStart(
|
||||||
transportBiDi,
|
transportBiDi,
|
||||||
cdpConnectionAdapter,
|
cdpConnectionAdapter,
|
||||||
''
|
'',
|
||||||
|
undefined,
|
||||||
|
bidiServerLogger
|
||||||
);
|
);
|
||||||
return pptrBiDiConnection;
|
return pptrBiDiConnection;
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,11 @@ module.exports = {
|
|||||||
selector:
|
selector:
|
||||||
'MemberExpression[object.name="puppeteer"][property.name="launch"]',
|
'MemberExpression[object.name="puppeteer"][property.name="launch"]',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
message: 'Unexpected debugging mocha test.',
|
||||||
|
selector:
|
||||||
|
'CallExpression[callee.object.name="it"] > MemberExpression > Identifier[name="deflake"], CallExpression[callee.object.name="it"] > MemberExpression > Identifier[name="deflakeOnly"]',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"puppeteer-core": "file:../packages/puppeteer-core",
|
||||||
"puppeteer": "file:../packages/puppeteer"
|
"puppeteer": "file:../packages/puppeteer"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,12 +26,7 @@ import type {Page} from 'puppeteer-core/internal/api/Page.js';
|
|||||||
import {rmSync} from 'puppeteer-core/internal/node/util/fs.js';
|
import {rmSync} from 'puppeteer-core/internal/node/util/fs.js';
|
||||||
import sinon from 'sinon';
|
import sinon from 'sinon';
|
||||||
|
|
||||||
import {
|
import {getTestState, isHeadless, launch} from './mocha-utils.js';
|
||||||
getTestState,
|
|
||||||
isHeadless,
|
|
||||||
itOnlyRegularInstall,
|
|
||||||
launch,
|
|
||||||
} from './mocha-utils.js';
|
|
||||||
import {dumpFrames, waitEvent} from './utils.js';
|
import {dumpFrames, waitEvent} from './utils.js';
|
||||||
|
|
||||||
const TMP_FOLDER = path.join(os.tmpdir(), 'pptr_tmp_folder-');
|
const TMP_FOLDER = path.join(os.tmpdir(), 'pptr_tmp_folder-');
|
||||||
@ -601,7 +596,7 @@ describe('Launcher specs', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
describe('Puppeteer.launch', function () {
|
describe('Puppeteer.launch', function () {
|
||||||
itOnlyRegularInstall('should be able to launch Chrome', async () => {
|
it('should be able to launch Chrome', async () => {
|
||||||
const {browser, close} = await launch({product: 'chrome'});
|
const {browser, close} = await launch({product: 'chrome'});
|
||||||
try {
|
try {
|
||||||
const userAgent = await browser.userAgent();
|
const userAgent = await browser.userAgent();
|
||||||
@ -875,7 +870,7 @@ describe('Launcher specs', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
describe('Puppeteer.executablePath', function () {
|
describe('Puppeteer.executablePath', function () {
|
||||||
itOnlyRegularInstall('should work', async () => {
|
it('should work', async () => {
|
||||||
const {puppeteer} = await getTestState({
|
const {puppeteer} = await getTestState({
|
||||||
skipLaunch: true,
|
skipLaunch: true,
|
||||||
});
|
});
|
||||||
|
@ -20,15 +20,11 @@ import path from 'path';
|
|||||||
import {TestServer} from '@pptr/testserver';
|
import {TestServer} from '@pptr/testserver';
|
||||||
import type {Protocol} from 'devtools-protocol';
|
import type {Protocol} from 'devtools-protocol';
|
||||||
import expect from 'expect';
|
import expect from 'expect';
|
||||||
import type * as Mocha from 'mocha';
|
import type * as MochaBase from 'mocha';
|
||||||
import puppeteer from 'puppeteer/lib/cjs/puppeteer/puppeteer.js';
|
import puppeteer from 'puppeteer/lib/cjs/puppeteer/puppeteer.js';
|
||||||
import type {Browser} from 'puppeteer-core/internal/api/Browser.js';
|
import type {Browser} from 'puppeteer-core/internal/api/Browser.js';
|
||||||
import type {BrowserContext} from 'puppeteer-core/internal/api/BrowserContext.js';
|
import type {BrowserContext} from 'puppeteer-core/internal/api/BrowserContext.js';
|
||||||
import type {Page} from 'puppeteer-core/internal/api/Page.js';
|
import type {Page} from 'puppeteer-core/internal/api/Page.js';
|
||||||
import {
|
|
||||||
setLogCapture,
|
|
||||||
getCapturedLogs,
|
|
||||||
} from 'puppeteer-core/internal/common/Debug.js';
|
|
||||||
import type {
|
import type {
|
||||||
PuppeteerLaunchOptions,
|
PuppeteerLaunchOptions,
|
||||||
PuppeteerNode,
|
PuppeteerNode,
|
||||||
@ -40,11 +36,45 @@ import sinon from 'sinon';
|
|||||||
|
|
||||||
import {extendExpectWithToBeGolden} from './utils.js';
|
import {extendExpectWithToBeGolden} from './utils.js';
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-namespace
|
||||||
|
namespace Mocha {
|
||||||
|
export interface SuiteFunction {
|
||||||
|
/**
|
||||||
|
* Use it if you want to capture debug logs for a specitic test suite in CI.
|
||||||
|
* This describe function enables capturing of debug logs and would print them
|
||||||
|
* only if a test fails to reduce the amount of output.
|
||||||
|
*/
|
||||||
|
withDebugLogs: (
|
||||||
|
description: string,
|
||||||
|
body: (this: MochaBase.Suite) => void
|
||||||
|
) => void;
|
||||||
|
}
|
||||||
|
export interface TestFunction {
|
||||||
|
/*
|
||||||
|
* Use to rerun the test and capture logs for the failed attempts
|
||||||
|
* that way we don't push all the logs making it easier to read.
|
||||||
|
*/
|
||||||
|
deflake: (
|
||||||
|
repeats: number,
|
||||||
|
title: string,
|
||||||
|
fn: MochaBase.AsyncFunc
|
||||||
|
) => void;
|
||||||
|
/*
|
||||||
|
* Use to rerun a single test and capture logs for the failed attempts
|
||||||
|
*/
|
||||||
|
deflakeOnly: (
|
||||||
|
repeats: number,
|
||||||
|
title: string,
|
||||||
|
fn: MochaBase.AsyncFunc
|
||||||
|
) => void;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const product =
|
const product =
|
||||||
process.env['PRODUCT'] || process.env['PUPPETEER_PRODUCT'] || 'chrome';
|
process.env['PRODUCT'] || process.env['PUPPETEER_PRODUCT'] || 'chrome';
|
||||||
|
|
||||||
const alternativeInstall = process.env['PUPPETEER_ALT_INSTALL'] || false;
|
|
||||||
|
|
||||||
const headless = (process.env['HEADLESS'] || 'true').trim().toLowerCase() as
|
const headless = (process.env['HEADLESS'] || 'true').trim().toLowerCase() as
|
||||||
| 'true'
|
| 'true'
|
||||||
| 'false'
|
| 'false'
|
||||||
@ -95,7 +125,6 @@ if (defaultBrowserOptions.executablePath) {
|
|||||||
|
|
||||||
const processVariables: {
|
const processVariables: {
|
||||||
product: string;
|
product: string;
|
||||||
alternativeInstall: string | boolean;
|
|
||||||
headless: 'true' | 'false' | 'new';
|
headless: 'true' | 'false' | 'new';
|
||||||
isHeadless: boolean;
|
isHeadless: boolean;
|
||||||
isFirefox: boolean;
|
isFirefox: boolean;
|
||||||
@ -104,7 +133,6 @@ const processVariables: {
|
|||||||
defaultBrowserOptions: PuppeteerLaunchOptions;
|
defaultBrowserOptions: PuppeteerLaunchOptions;
|
||||||
} = {
|
} = {
|
||||||
product,
|
product,
|
||||||
alternativeInstall,
|
|
||||||
headless,
|
headless,
|
||||||
isHeadless,
|
isHeadless,
|
||||||
isFirefox,
|
isFirefox,
|
||||||
@ -217,17 +245,6 @@ export interface PuppeteerTestState {
|
|||||||
}
|
}
|
||||||
const state: Partial<PuppeteerTestState> = {};
|
const state: Partial<PuppeteerTestState> = {};
|
||||||
|
|
||||||
export const itOnlyRegularInstall = (
|
|
||||||
description: string,
|
|
||||||
body: Mocha.AsyncFunc
|
|
||||||
): Mocha.Test => {
|
|
||||||
if (processVariables.alternativeInstall || process.env['BINARY']) {
|
|
||||||
return it.skip(description, body);
|
|
||||||
} else {
|
|
||||||
return it(description, body);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
process.env['MOCHA_WORKER_ID'] === undefined ||
|
process.env['MOCHA_WORKER_ID'] === undefined ||
|
||||||
process.env['MOCHA_WORKER_ID'] === '0'
|
process.env['MOCHA_WORKER_ID'] === '0'
|
||||||
@ -376,34 +393,6 @@ export const expectCookieEquals = async (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Use it if you want to capture debug logs for a specitic test suite in CI.
|
|
||||||
* This describe function enables capturing of debug logs and would print them
|
|
||||||
* only if a test fails to reduce the amount of output.
|
|
||||||
*/
|
|
||||||
export const describeWithDebugLogs = (
|
|
||||||
description: string,
|
|
||||||
body: (this: Mocha.Suite) => void
|
|
||||||
): Mocha.Suite | void => {
|
|
||||||
describe(description + '-debug', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
setLogCapture(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(function () {
|
|
||||||
if (this.currentTest?.state === 'failed') {
|
|
||||||
console.log(
|
|
||||||
`\n"${this.currentTest.fullTitle()}" failed. Here is a debug log:`
|
|
||||||
);
|
|
||||||
console.log(getCapturedLogs().join('\n') + '\n');
|
|
||||||
}
|
|
||||||
setLogCapture(false);
|
|
||||||
});
|
|
||||||
|
|
||||||
describe(description, body);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
export const shortWaitForArrayToHaveAtLeastNElements = async (
|
export const shortWaitForArrayToHaveAtLeastNElements = async (
|
||||||
data: unknown[],
|
data: unknown[],
|
||||||
minLength: number,
|
minLength: number,
|
||||||
|
@ -18,10 +18,10 @@ import expect from 'expect';
|
|||||||
import type {BrowserContext} from 'puppeteer-core/internal/api/BrowserContext.js';
|
import type {BrowserContext} from 'puppeteer-core/internal/api/BrowserContext.js';
|
||||||
import type {CdpTarget} from 'puppeteer-core/internal/cdp/Target.js';
|
import type {CdpTarget} from 'puppeteer-core/internal/cdp/Target.js';
|
||||||
|
|
||||||
import {describeWithDebugLogs, getTestState, launch} from './mocha-utils.js';
|
import {getTestState, launch} from './mocha-utils.js';
|
||||||
import {attachFrame, detachFrame, navigateFrame} from './utils.js';
|
import {attachFrame, detachFrame, navigateFrame} from './utils.js';
|
||||||
|
|
||||||
describeWithDebugLogs('OOPIF', function () {
|
describe('OOPIF', function () {
|
||||||
/* We use a special browser for this test as we need the --site-per-process flag */
|
/* We use a special browser for this test as we need the --site-per-process flag */
|
||||||
let state: Awaited<ReturnType<typeof launch>>;
|
let state: Awaited<ReturnType<typeof launch>>;
|
||||||
|
|
||||||
|
@ -246,6 +246,7 @@ describe('Target', function () {
|
|||||||
expect(targetChanged).toBe(false);
|
expect(targetChanged).toBe(false);
|
||||||
context.off('targetchanged', listener);
|
context.off('targetchanged', listener);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should not crash while redirecting if original request was missed', async () => {
|
it('should not crash while redirecting if original request was missed', async () => {
|
||||||
const {page, server, context} = await getTestState();
|
const {page, server, context} = await getTestState();
|
||||||
|
|
||||||
@ -268,11 +269,12 @@ describe('Target', function () {
|
|||||||
{timeout: 3000}
|
{timeout: 3000}
|
||||||
);
|
);
|
||||||
const newPage = (await target.page())!;
|
const newPage = (await target.page())!;
|
||||||
|
const loadEvent = waitEvent(newPage, 'load');
|
||||||
// Issue a redirect.
|
// Issue a redirect.
|
||||||
serverResponse.writeHead(302, {location: '/injectedstyle.css'});
|
serverResponse.writeHead(302, {location: '/injectedstyle.css'});
|
||||||
serverResponse.end();
|
serverResponse.end();
|
||||||
// Wait for the new page to load.
|
// Wait for the new page to load.
|
||||||
await waitEvent(newPage, 'load');
|
await loadEvent;
|
||||||
// Cleanup.
|
// Cleanup.
|
||||||
await newPage.close();
|
await newPage.close();
|
||||||
});
|
});
|
||||||
|
@ -71,3 +71,33 @@ Examples:
|
|||||||
Currently, expectations are updated manually. The test runner outputs the
|
Currently, expectations are updated manually. The test runner outputs the
|
||||||
suggested changes to the expectation file if the test run does not match
|
suggested changes to the expectation file if the test run does not match
|
||||||
expectations.
|
expectations.
|
||||||
|
|
||||||
|
## Debugging flaky test
|
||||||
|
|
||||||
|
### Utility functions:
|
||||||
|
|
||||||
|
| Utility | Params | Description |
|
||||||
|
| ------------------------ | ------------------------------- | --------------------------------------------------------------------------------- |
|
||||||
|
| `describe.withDebugLogs` | `(title, <DescribeBody>)` | Capture and print debug logs for each test that failed |
|
||||||
|
| `it.deflake` | `(repeat, title, <itFunction>)` | Reruns the test N number of times and print the debug logs if for the failed runs |
|
||||||
|
| `it.deflakeOnly` | `(repeat, title, <itFunction>)` | Same as `it.deflake` but runs only this specific test |
|
||||||
|
|
||||||
|
### With Environment variable
|
||||||
|
|
||||||
|
Run the test with the following environment variable to wrap it around `describe.withDebugLogs`. Example:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
PUPPETEER_DEFLAKE_TESTS="[navigation.spec] navigation Page.goto should navigate to empty page with networkidle0" npm run test:chrome:headless
|
||||||
|
```
|
||||||
|
|
||||||
|
It also works with [patterns](#1--this-is-my-header) just like `TestExpectations.json`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
PUPPETEER_DEFLAKE_TESTS="[navigation.spec] *" npm run test:chrome:headless
|
||||||
|
```
|
||||||
|
|
||||||
|
By default the test is rerun 100 times, but you can control this as well:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
PUPPETEER_DEFLAKE_RETRIES=1000 PUPPETEER_DEFLAKE_TESTS="[navigation.spec] *" npm run test:chrome:headless
|
||||||
|
```
|
||||||
|
@ -29,5 +29,8 @@
|
|||||||
"build"
|
"build"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"puppeteer-core": "file:../../packages/puppeteer-core"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,10 @@
|
|||||||
|
|
||||||
import Mocha from 'mocha';
|
import Mocha from 'mocha';
|
||||||
import commonInterface from 'mocha/lib/interfaces/common';
|
import commonInterface from 'mocha/lib/interfaces/common';
|
||||||
|
import {
|
||||||
|
setLogCapture,
|
||||||
|
getCapturedLogs,
|
||||||
|
} from 'puppeteer-core/internal/common/Debug.js';
|
||||||
|
|
||||||
import {testIdMatchesExpectationPattern} from './utils.js';
|
import {testIdMatchesExpectationPattern} from './utils.js';
|
||||||
|
|
||||||
@ -28,6 +32,10 @@ const skippedTests: Array<{testIdPattern: string; skip: true}> = process.env[
|
|||||||
? JSON.parse(process.env['PUPPETEER_SKIPPED_TEST_CONFIG'])
|
? JSON.parse(process.env['PUPPETEER_SKIPPED_TEST_CONFIG'])
|
||||||
: [];
|
: [];
|
||||||
|
|
||||||
|
const deflakeRetries = Number(process.env['PUPPETEER_DEFLAKE_RETRIES'] ?? 100);
|
||||||
|
const deflakeTestPattern: string | undefined =
|
||||||
|
process.env['PUPPETEER_DEFLAKE_TESTS'];
|
||||||
|
|
||||||
function shouldSkipTest(test: Mocha.Test): boolean {
|
function shouldSkipTest(test: Mocha.Test): boolean {
|
||||||
// TODO: more efficient lookup.
|
// TODO: more efficient lookup.
|
||||||
const definition = skippedTests.find(skippedTest => {
|
const definition = skippedTests.find(skippedTest => {
|
||||||
@ -39,8 +47,26 @@ function shouldSkipTest(test: Mocha.Test): boolean {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function shouldDeflakeTest(test: Mocha.Test): boolean {
|
||||||
|
if (deflakeTestPattern) {
|
||||||
|
// TODO: cache if we have seen it already
|
||||||
|
return testIdMatchesExpectationPattern(test, deflakeTestPattern);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function dumpLogsIfFail(this: Mocha.Context) {
|
||||||
|
if (this.currentTest?.state === 'failed') {
|
||||||
|
console.log(
|
||||||
|
`\n"${this.currentTest.fullTitle()}" failed. Here is a debug log:`
|
||||||
|
);
|
||||||
|
console.log(getCapturedLogs().join('\n') + '\n');
|
||||||
|
}
|
||||||
|
setLogCapture(false);
|
||||||
|
}
|
||||||
|
|
||||||
function customBDDInterface(suite: Mocha.Suite) {
|
function customBDDInterface(suite: Mocha.Suite) {
|
||||||
const suites = [suite];
|
const suites: [Mocha.Suite] = [suite];
|
||||||
|
|
||||||
suite.on(
|
suite.on(
|
||||||
Mocha.Suite.constants.EVENT_FILE_PRE_REQUIRE,
|
Mocha.Suite.constants.EVENT_FILE_PRE_REQUIRE,
|
||||||
@ -78,12 +104,25 @@ function customBDDInterface(suite: Mocha.Suite) {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
describe.withDebugLogs = function (
|
||||||
|
description: string,
|
||||||
|
body: (this: Mocha.Suite) => void
|
||||||
|
): void {
|
||||||
|
context['describe']('with Debug Logs', () => {
|
||||||
|
context['beforeEach'](() => {
|
||||||
|
setLogCapture(true);
|
||||||
|
});
|
||||||
|
context['afterEach'](dumpLogsIfFail);
|
||||||
|
context['describe'](description, body);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||||
// @ts-expect-error
|
// @ts-expect-error
|
||||||
context['describe'] = describe;
|
context['describe'] = describe;
|
||||||
|
|
||||||
function it(title: string, fn: Mocha.TestFunction, itOnly = false) {
|
function it(title: string, fn: Mocha.TestFunction, itOnly = false) {
|
||||||
const suite = suites[0]!;
|
const suite = suites[0]! as Mocha.Suite;
|
||||||
const test = new Mocha.Test(title, suite.isPending() ? undefined : fn);
|
const test = new Mocha.Test(title, suite.isPending() ? undefined : fn);
|
||||||
test.file = file;
|
test.file = file;
|
||||||
test.parent = suite;
|
test.parent = suite;
|
||||||
@ -95,8 +134,22 @@ function customBDDInterface(suite: Mocha.Suite) {
|
|||||||
return child === suite;
|
return child === suite;
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
if (shouldDeflakeTest(test)) {
|
||||||
|
const deflakeSuit = Mocha.Suite.create(suite, 'with Debug Logs');
|
||||||
|
test.parent = deflakeSuit;
|
||||||
|
deflakeSuit.file = file;
|
||||||
|
deflakeSuit.parent = suite;
|
||||||
|
|
||||||
if (shouldSkipTest(test) && !(itOnly || describeOnly)) {
|
deflakeSuit.beforeEach(function () {
|
||||||
|
setLogCapture(true);
|
||||||
|
});
|
||||||
|
deflakeSuit.afterEach(dumpLogsIfFail);
|
||||||
|
for (let i = 0; i < deflakeRetries; i++) {
|
||||||
|
deflakeSuit.addTest(test.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
return test;
|
||||||
|
} else if (!(itOnly || describeOnly) && shouldSkipTest(test)) {
|
||||||
const test = new Mocha.Test(title);
|
const test = new Mocha.Test(title);
|
||||||
test.file = file;
|
test.file = file;
|
||||||
suite.addTest(test);
|
suite.addTest(test);
|
||||||
@ -110,7 +163,7 @@ function customBDDInterface(suite: Mocha.Suite) {
|
|||||||
it.only = function (title: string, fn: Mocha.TestFunction) {
|
it.only = function (title: string, fn: Mocha.TestFunction) {
|
||||||
return common.test.only(
|
return common.test.only(
|
||||||
mocha,
|
mocha,
|
||||||
(context['it'] as typeof it)(title, fn, true)
|
(context['it'] as unknown as typeof it)(title, fn, true)
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -118,6 +171,24 @@ function customBDDInterface(suite: Mocha.Suite) {
|
|||||||
return context['it'](title);
|
return context['it'](title);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function wrapDeflake(
|
||||||
|
func: Function
|
||||||
|
): (repeats: number, title: string, fn: Mocha.AsyncFunc) => void {
|
||||||
|
return (repeats: number, title: string, fn: Mocha.AsyncFunc): void => {
|
||||||
|
(context['describe'] as unknown as typeof describe).withDebugLogs(
|
||||||
|
'with Debug Logs',
|
||||||
|
() => {
|
||||||
|
for (let i = 1; i <= repeats; i++) {
|
||||||
|
func(`${i}/${title}`, fn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
it.deflake = wrapDeflake(it);
|
||||||
|
it.deflakeOnly = wrapDeflake(it.only);
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||||
// @ts-expect-error
|
// @ts-expect-error
|
||||||
context.it = it;
|
context.it = it;
|
||||||
|
Loading…
Reference in New Issue
Block a user