From b042b435a99bf5b69a987cc8b922d6af3022b916 Mon Sep 17 00:00:00 2001 From: uid11 Date: Wed, 23 Oct 2024 16:37:10 +0300 Subject: [PATCH 1/6] FI-1432 feat: support waitings before retries fix: remove references on TestCafe fix: set actual timeout of tests for Playwright engine fix: turn off unnecessary rule `@typescript-eslint/consistent-return` feat: add `documentUrl` argument to `assertPage` method of pages fix: restore `fullMocks` fixture for internal test of "full mocks" --- .eslintrc.yaml | 2 +- README.md | 3 + autotests/configurator/index.ts | 1 + .../configurator/mapLogPayloadInConsole.ts | 1 - autotests/configurator/types/index.ts | 1 + autotests/configurator/types/packSpecific.ts | 1 + autotests/fixtures/fullMocks/jKDXNUZ75U.json | 1 - autotests/fixtures/fullMocks/mr-iHTD7Lp.json | 1 + autotests/packs/allTests.ts | 3 +- package.json | 4 +- src/Page.ts | 3 +- src/actions/pages/assertPage.ts | 2 +- src/actions/pages/navigateToPage.ts | 2 +- src/constants/testRun.ts | 4 +- src/generators/createRunId.ts | 18 +++- src/types/clientFunction.ts | 21 ----- src/types/config/config.ts | 21 ++--- src/types/config/ownE2edConfig.ts | 15 ++++ src/types/internal.ts | 2 - src/types/userland/createPackSpecificTypes.ts | 1 + src/utils/fs/index.ts | 2 + src/utils/fs/readEventFromFile.ts | 23 +++++ src/utils/fs/readEventsFromFiles.ts | 14 +-- src/utils/selectors/Selector.ts | 1 - src/utils/selectors/createCustomMethods.ts | 2 +- src/utils/test/beforeTest.ts | 17 +++- src/utils/test/getRunTest.ts | 9 +- src/utils/test/runTestFn.ts | 9 +- src/utils/test/waitBeforeRetry.ts | 85 +++++++++++++++++++ 29 files changed, 206 insertions(+), 63 deletions(-) delete mode 100644 autotests/fixtures/fullMocks/jKDXNUZ75U.json create mode 100644 autotests/fixtures/fullMocks/mr-iHTD7Lp.json create mode 100644 src/utils/fs/readEventFromFile.ts create mode 100644 src/utils/test/waitBeforeRetry.ts diff --git a/.eslintrc.yaml b/.eslintrc.yaml index cd1a5283..e4a94bbf 100644 --- a/.eslintrc.yaml +++ b/.eslintrc.yaml @@ -163,7 +163,7 @@ rules: '@typescript-eslint/class-literal-property-style': error '@typescript-eslint/consistent-generic-constructors': error '@typescript-eslint/consistent-indexed-object-style': error - '@typescript-eslint/consistent-return': error + '@typescript-eslint/consistent-return': off '@typescript-eslint/consistent-type-assertions': [error, {assertionStyle: as, objectLiteralTypeAssertions: never}] '@typescript-eslint/consistent-type-definitions': [error, type] diff --git a/README.md b/README.md index 2261834c..b3fbf68b 100644 --- a/README.md +++ b/README.md @@ -369,6 +369,9 @@ This parameter can be overridden in the test-specific options. `viewportWidth: number`: width of viewport of page in pixels. +`waitBeforeRetry: (options: Options) => number`: returns how many milliseconds `e2ed` +should wait before running test (for retries). + `waitForAllRequestsComplete.maxIntervalBetweenRequestsInMs: number`: default maximum interval (in milliseconds) between requests for `waitForAllRequestsComplete` function. If there are no new requests for more than this interval, then the promise diff --git a/autotests/configurator/index.ts b/autotests/configurator/index.ts index dbdbcc52..376e5756 100644 --- a/autotests/configurator/index.ts +++ b/autotests/configurator/index.ts @@ -27,4 +27,5 @@ export type { SkipTests, TestFunction, TestMeta, + WaitBeforeRetry, } from './types'; diff --git a/autotests/configurator/mapLogPayloadInConsole.ts b/autotests/configurator/mapLogPayloadInConsole.ts index b9bbce55..fce15437 100644 --- a/autotests/configurator/mapLogPayloadInConsole.ts +++ b/autotests/configurator/mapLogPayloadInConsole.ts @@ -19,7 +19,6 @@ export const mapLogPayloadInConsole: MapLogPayloadInConsole = (message, payload) if ( message.startsWith('Caught an error when running tests in retry') || - message.startsWith('Warning from TestCafe:') || message.startsWith('Usage:') ) { return payload; diff --git a/autotests/configurator/types/index.ts b/autotests/configurator/types/index.ts index 29aecba9..2d0dbdc3 100644 --- a/autotests/configurator/types/index.ts +++ b/autotests/configurator/types/index.ts @@ -16,6 +16,7 @@ export type { MapLogPayloadInReport, Pack, TestFunction, + WaitBeforeRetry, } from './packSpecific'; export type {SkipTests} from './skipTests'; export type {TestMeta} from './testMeta'; diff --git a/autotests/configurator/types/packSpecific.ts b/autotests/configurator/types/packSpecific.ts index c4ccc859..4ed96353 100644 --- a/autotests/configurator/types/packSpecific.ts +++ b/autotests/configurator/types/packSpecific.ts @@ -23,4 +23,5 @@ export type MapLogPayloadInConsole = PackSpecificTypes['MapLogPayloadInConsole'] export type MapLogPayloadInLogFile = PackSpecificTypes['MapLogPayloadInLogFile']; export type MapLogPayloadInReport = PackSpecificTypes['MapLogPayloadInReport']; export type TestFunction = PackSpecificTypes['TestFunction']; +export type WaitBeforeRetry = PackSpecificTypes['WaitBeforeRetry']; export type {Pack}; diff --git a/autotests/fixtures/fullMocks/jKDXNUZ75U.json b/autotests/fixtures/fullMocks/jKDXNUZ75U.json deleted file mode 100644 index 184ef5ee..00000000 --- a/autotests/fixtures/fullMocks/jKDXNUZ75U.json +++ /dev/null @@ -1 +0,0 @@ -{"/api/product/135865":[{"completionTimeInMs":1721837017221,"duration":"4ms","request":{"method":"POST","query":{"size":"13"},"requestBody":{"cookies":[],"input":17,"model":"samsung","version":"12"},"requestHeaders":{"accept":"*/*","accept-language":"en-US","content-type":"application/json; charset=UTF-8","origin":"https://joomcode.github.io","referer":"https://joomcode.github.io/","user-agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.6533.17 Safari/537.36","sec-ch-ua":"\"Not)A;Brand\";v=\"99\", \"HeadlessChrome\";v=\"127\", \"Chromium\";v=\"127\"","sec-ch-ua-mobile":"?0","sec-ch-ua-platform":"\"Windows\""},"url":"https://reqres.in/api/product/135865?size=13","utcTimeInMs":1721837017217},"responseBody":{"id":135865,"method":"POST","output":"17","payload":{"id":"135865","cookies":[],"input":17,"model":"samsung","version":"12"},"query":{"size":"13"},"url":"https://reqres.in/api/product/135865?size=13"},"responseHeaders":{"access-control-allow-origin":"https://joomcode.github.io","access-control-allow-credentials":"true","content-length":"201","vary":"Origin","content-type":"application/json; charset=UTF-8"},"statusCode":200},{"completionTimeInMs":1721837017604,"duration":"2ms","request":{"method":"POST","query":{"size":"13"},"requestBody":{"cookies":[],"input":17,"model":"samsung","version":"12"},"requestHeaders":{"accept":"*/*","accept-language":"en-US","content-type":"application/json; charset=UTF-8","origin":"https://joomcode.github.io","referer":"https://joomcode.github.io/","user-agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.6533.17 Safari/537.36","sec-ch-ua":"\"Not)A;Brand\";v=\"99\", \"HeadlessChrome\";v=\"127\", \"Chromium\";v=\"127\"","sec-ch-ua-mobile":"?0","sec-ch-ua-platform":"\"Windows\""},"url":"https://reqres.in/api/product/135865?size=13","utcTimeInMs":1721837017602},"responseBody":{"cookies":[],"input":17,"model":"samsung","version":"12","id":"246","createdAt":"2024-07-24T16:03:37.444Z"},"responseHeaders":{"date":"Wed, 24 Jul 2024 16:03:37 GMT","via":"1.1 vegur","cf-cache-status":"DYNAMIC","nel":"{\"report_to\":\"heroku-nel\",\"max_age\":3600,\"success_fraction\":0.005,\"failure_fraction\":0.05,\"response_headers\":[\"Via\"]}","server":"cloudflare","x-powered-by":"Express","etag":"W/\"6c-u6X0/XoH52vavslyTQkfjMGy8UI\"","report-to":"{\"group\":\"heroku-nel\",\"max_age\":3600,\"endpoints\":[{\"url\":\"https://nel.heroku.com/reports?ts=1721837017&sid=c4c9725f-1ab0-44d8-820f-430df2718e11&s=QtldDuafqeEmbMdmRCPQibdEu5m%2BGzC8BP8Dd%2F%2FFv0A%3D\"}]}","content-type":"application/json; charset=utf-8","access-control-allow-origin":"*","cf-ray":"8a85242e4d4c0d52-ARN","content-length":"108","reporting-endpoints":"heroku-nel=https://nel.heroku.com/reports?ts=1721837017&sid=c4c9725f-1ab0-44d8-820f-430df2718e11&s=QtldDuafqeEmbMdmRCPQibdEu5m%2BGzC8BP8Dd%2F%2FFv0A%3D"},"statusCode":201}]} \ No newline at end of file diff --git a/autotests/fixtures/fullMocks/mr-iHTD7Lp.json b/autotests/fixtures/fullMocks/mr-iHTD7Lp.json new file mode 100644 index 00000000..a197eb8a --- /dev/null +++ b/autotests/fixtures/fullMocks/mr-iHTD7Lp.json @@ -0,0 +1 @@ +{"/api/product/135865":[{"completionTimeInMs":1729850053899,"duration":"4ms","request":{"method":"POST","query":{"size":"13"},"requestBody":{"cookies":[],"input":17,"model":"samsung","version":"12"},"requestHeaders":{"accept":"*/*","accept-language":"en-US","content-type":"application/json; charset=UTF-8","referer":"https://joomcode.github.io/","user-agent":"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.35 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.35","sec-ch-ua":"\"Chromium\";v=\"130\", \"HeadlessChrome\";v=\"130\", \"Not?A_Brand\";v=\"99\"","sec-ch-ua-mobile":"?0","sec-ch-ua-platform":"\"Linux\""},"url":"https://reqres.in/api/product/135865?size=13","utcTimeInMs":1729850053895},"responseBody":{"id":135865,"method":"POST","output":"17","payload":{"id":"135865","cookies":[],"input":17,"model":"samsung","version":"12"},"query":{"size":"13"},"url":"https://reqres.in/api/product/135865?size=13"},"responseHeaders":{"content-type":"application/json; charset=UTF-8","content-length":"201"},"statusCode":200},{"completionTimeInMs":1729850054453,"duration":"4ms","request":{"method":"POST","query":{"size":"13"},"requestBody":{"cookies":[],"input":17,"model":"samsung","version":"12"},"requestHeaders":{"accept":"*/*","accept-language":"en-US","content-type":"application/json; charset=UTF-8","referer":"https://joomcode.github.io/","user-agent":"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.35 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.35","sec-ch-ua":"\"Chromium\";v=\"130\", \"HeadlessChrome\";v=\"130\", \"Not?A_Brand\";v=\"99\"","sec-ch-ua-mobile":"?0","sec-ch-ua-platform":"\"Linux\""},"url":"https://reqres.in/api/product/135865?size=13","utcTimeInMs":1729850054449},"responseBody":{"cookies":[],"input":17,"model":"samsung","version":"12","id":"816","createdAt":"2024-10-25T09:54:14.388Z"},"responseHeaders":{"reporting-endpoints":"heroku-nel=https://nel.heroku.com/reports?ts=1729850054&sid=c4c9725f-1ab0-44d8-820f-430df2718e11&s=LZBdsiG2rgFml4T6awFYmhftTlXLyvdfcMob39wiK2I%3D","nel":"{\"report_to\":\"heroku-nel\",\"max_age\":3600,\"success_fraction\":0.005,\"failure_fraction\":0.05,\"response_headers\":[\"Via\"]}","cf-cache-status":"DYNAMIC","etag":"W/\"6c-uS8VtSQALUKVvQbVlIe2ZwBwLRE\"","report-to":"{\"group\":\"heroku-nel\",\"max_age\":3600,\"endpoints\":[{\"url\":\"https://nel.heroku.com/reports?ts=1729850054&sid=c4c9725f-1ab0-44d8-820f-430df2718e11&s=LZBdsiG2rgFml4T6awFYmhftTlXLyvdfcMob39wiK2I%3D\"}]}","via":"1.1 vegur","cf-ray":"8d8152f7a8d43cff-CDG","access-control-allow-origin":"*","content-length":"108","date":"Fri, 25 Oct 2024 09:54:14 GMT","content-type":"application/json; charset=utf-8","x-powered-by":"Express","server":"cloudflare"},"statusCode":201}]} \ No newline at end of file diff --git a/autotests/packs/allTests.ts b/autotests/packs/allTests.ts index b926c23b..eefbc47e 100644 --- a/autotests/packs/allTests.ts +++ b/autotests/packs/allTests.ts @@ -78,10 +78,11 @@ export const pack: Pack = { takeViewportScreenshotOnError: true, testFileGlobs: ['**/autotests/tests/**/*.ts'], testIdleTimeout: 8_000, - testTimeout: 60_000, + testTimeout: 15_000, userAgent, viewportHeight: 1080, viewportWidth: 1920, + waitBeforeRetry: () => 0, waitForAllRequestsComplete: { maxIntervalBetweenRequestsInMs: 500, timeout: 30_000, diff --git a/package.json b/package.json index 7a182a66..103d11de 100644 --- a/package.json +++ b/package.json @@ -1,10 +1,10 @@ { "name": "e2ed", "version": "0.18.15", - "description": "E2E testing framework over TestCafe", + "description": "E2E testing framework over Playwright", "keywords": [ "E2E", - "TestCafe", + "Playwright", "testing" ], "author": "uid11", diff --git a/src/Page.ts b/src/Page.ts index 9c072029..3c60f64a 100644 --- a/src/Page.ts +++ b/src/Page.ts @@ -80,8 +80,9 @@ export abstract class Page { * Asserts that we are on the expected page by `isMatch` flage. * `isMatch` equals `true`, if url matches the page with given parameters, and `false` otherwise. */ - assertPage(isMatch: boolean): AsyncVoid { + assertPage(isMatch: boolean, documentUrl: Url): AsyncVoid { assertValueIsTrue(isMatch, `the document url matches the page "${this.constructor.name}"`, { + documentUrl, page: this, }); } diff --git a/src/actions/pages/assertPage.ts b/src/actions/pages/assertPage.ts index db9903b8..9e1999c6 100644 --- a/src/actions/pages/assertPage.ts +++ b/src/actions/pages/assertPage.ts @@ -34,7 +34,7 @@ export const assertPage = async ( LogEventType.InternalAction, ); - await page.assertPage(isMatch); + await page.assertPage(isMatch, documentUrl); await page.afterAssertPage?.(); diff --git a/src/actions/pages/navigateToPage.ts b/src/actions/pages/navigateToPage.ts index 02f02b7c..06aa1159 100644 --- a/src/actions/pages/navigateToPage.ts +++ b/src/actions/pages/navigateToPage.ts @@ -46,7 +46,7 @@ export const navigateToPage = async ( const documentUrl = await getDocumentUrl(); const isMatch = route.isMatchUrl(documentUrl); - await page.assertPage(isMatch); + await page.assertPage(isMatch, documentUrl); await page.afterAssertPage?.(); diff --git a/src/constants/testRun.ts b/src/constants/testRun.ts index 4f867d47..25ba3cbf 100644 --- a/src/constants/testRun.ts +++ b/src/constants/testRun.ts @@ -13,8 +13,8 @@ declare type TestRunTypesChecks = [ /** * Main status of test run. - * Failed if it have run error and passed if not. - * Broken if the test failed and TestCafe restarted it themself. + * `Failed` if it have run error and passed if not. + * Probably should never be `Broken`. */ export const enum TestRunStatus { Broken = 'broken', diff --git a/src/generators/createRunId.ts b/src/generators/createRunId.ts index c8b4df76..d879fe90 100644 --- a/src/generators/createRunId.ts +++ b/src/generators/createRunId.ts @@ -1,9 +1,21 @@ -import {getRandomId} from './getRandomId'; +import {createHash} from 'node:crypto'; -import type {RunId} from '../types/internal'; +import type {RunId, Test} from '../types/internal'; + +const runIdBaseLength = 10; /** * Creates new RunId for TestRun. * @internal */ -export const createRunId = (): RunId => getRandomId().replace(/:/g, '-') as RunId; +export const createRunId = (test: Test, retryIndex: number): RunId => { + const data = {...test, testFn: test.testFn.toString()}; + const text = JSON.stringify(data); + const hash = createHash('sha1'); + + hash.update(text); + + const base = hash.digest('base64url').slice(0, runIdBaseLength); + + return `${base}-${retryIndex}` as RunId; +}; diff --git a/src/types/clientFunction.ts b/src/types/clientFunction.ts index 48570d3c..2dbb9d66 100644 --- a/src/types/clientFunction.ts +++ b/src/types/clientFunction.ts @@ -7,24 +7,3 @@ export type ClientFunction this: void, ...args: Args ) => Promise; - -/** - * Result of the internal client function wrapper (object with error as string or with value). - * @internal - */ -export type ClientFunctionWrapperResult = Readonly< - | { - errorMessage: string; - result: undefined; - } - | { - errorMessage: undefined; - result: Result; - } ->; - -/** - * Internal TestCafe error object or undefined. - * @internal - */ -export type MaybeTestCafeError = {code?: string} | undefined; diff --git a/src/types/config/config.ts b/src/types/config/config.ts index 2b425375..10592056 100644 --- a/src/types/config/config.ts +++ b/src/types/config/config.ts @@ -10,18 +10,6 @@ import type { import type {WithDoBeforePack} from './doBeforePack'; import type {OwnE2edConfig} from './ownE2edConfig'; -/** - * Userland part of TestCafe config. - */ -type UserlandTestCafeConfig = Readonly<{ - assertionTimeout: number; - concurrency: number; - pageRequestTimeout: number; - port1: number; - port2: number; - selectorTimeout: number; -}>; - /** * Supported browsers. */ @@ -69,5 +57,12 @@ export type UserlandPackWithoutDoBeforePack< CustomReportProperties = CustomReportPropertiesPlaceholder, SkipTests = SkipTestsPlaceholder, TestMeta = TestMetaPlaceholder, -> = UserlandTestCafeConfig & +> = Readonly<{ + assertionTimeout: number; + concurrency: number; + pageRequestTimeout: number; + port1: number; + port2: number; + selectorTimeout: number; +}> & OwnE2edConfig; diff --git a/src/types/config/ownE2edConfig.ts b/src/types/config/ownE2edConfig.ts index ab9e7dfc..98e6786d 100644 --- a/src/types/config/ownE2edConfig.ts +++ b/src/types/config/ownE2edConfig.ts @@ -1,5 +1,7 @@ import type {PlaywrightTestConfig} from '@playwright/test'; +import type {TestRunStatus} from '../../constants/internal'; + import type {FullMocksConfig} from '../fullMocks'; import type {MapBackendResponseToLog, MapLogPayload, MapLogPayloadInReport} from '../log'; import type {MaybePromise} from '../promise'; @@ -255,6 +257,19 @@ export type OwnE2edConfig< */ viewportWidth: number; + /** + * Returns how many milliseconds `e2ed` should wait before running test (for retries). + */ + waitBeforeRetry: ( + this: void, + options: Readonly<{ + previousError: string | undefined; + retryIndex: number; + status: TestRunStatus; + testStaticOptions: TestStaticOptions; + }>, + ) => MaybePromise; + /** * Group of settings for the `waitForAllRequestsComplete` function. */ diff --git a/src/types/internal.ts b/src/types/internal.ts index 9bc167c9..844c7805 100644 --- a/src/types/internal.ts +++ b/src/types/internal.ts @@ -5,8 +5,6 @@ export type {Brand, IsBrand} from './brand'; export type {Expect, IsEqual, IsReadonlyKey} from './checks'; export type {Class} from './class'; export type {ClientFunction} from './clientFunction'; -/** @internal */ -export type {ClientFunctionWrapperResult, MaybeTestCafeError} from './clientFunction'; export type { AnyPack, BrowserName, diff --git a/src/types/userland/createPackSpecificTypes.ts b/src/types/userland/createPackSpecificTypes.ts index 3fc8383a..f9f8a231 100644 --- a/src/types/userland/createPackSpecificTypes.ts +++ b/src/types/userland/createPackSpecificTypes.ts @@ -36,4 +36,5 @@ export type CreatePackSpecificTypes< MapLogPayloadInLogFile: MapLogPayload; MapLogPayloadInReport: MapLogPayloadInReport; TestFunction: TestFunction; + WaitBeforeRetry: FullPackConfigByPack['waitBeforeRetry']; }>; diff --git a/src/utils/fs/index.ts b/src/utils/fs/index.ts index 56d5ffae..ee1cd2ab 100644 --- a/src/utils/fs/index.ts +++ b/src/utils/fs/index.ts @@ -7,6 +7,8 @@ export {getTestRunEventFileName} from './getTestRunEventFileName'; /** @internal */ export {getLastLogEventTimeInMs, writeLogEventTime} from './logIsoString'; /** @internal */ +export {readEventFromFile} from './readEventFromFile'; +/** @internal */ export {readEventsFromFiles} from './readEventsFromFiles'; /** @internal */ export {readStartInfo} from './readStartInfo'; diff --git a/src/utils/fs/readEventFromFile.ts b/src/utils/fs/readEventFromFile.ts new file mode 100644 index 00000000..618b2006 --- /dev/null +++ b/src/utils/fs/readEventFromFile.ts @@ -0,0 +1,23 @@ +import {readFile} from 'node:fs/promises'; +import {join} from 'node:path'; + +import {EVENTS_DIRECTORY_PATH, READ_FILE_OPTIONS} from '../../constants/internal'; + +import {generalLog} from '../generalLog'; + +/** + * Read event object with test run from temporary directory. + * @internal + */ +export const readEventFromFile = (fileName: string): Promise => { + const filePath = join(EVENTS_DIRECTORY_PATH, fileName); + + return readFile(filePath, READ_FILE_OPTIONS).catch((error: unknown) => { + generalLog(`Caught an error on reading text of test run event from file "${fileName}"`, { + error, + filePath, + }); + + return undefined; + }); +}; diff --git a/src/utils/fs/readEventsFromFiles.ts b/src/utils/fs/readEventsFromFiles.ts index a9b7c50f..3e7c7f53 100644 --- a/src/utils/fs/readEventsFromFiles.ts +++ b/src/utils/fs/readEventsFromFiles.ts @@ -1,18 +1,19 @@ import {execFile} from 'node:child_process'; -import {readdir, readFile} from 'node:fs/promises'; +import {readdir} from 'node:fs/promises'; import {join} from 'node:path'; import { AMOUNT_OF_PARALLEL_OPEN_FILES, EVENTS_DIRECTORY_PATH, INTERNAL_REPORTS_DIRECTORY_PATH, - READ_FILE_OPTIONS, } from '../../constants/internal'; import {assertValueIsDefined, assertValueIsTrue} from '../asserts'; import {generalLog} from '../generalLog'; import {getDurationWithUnits} from '../getDurationWithUnits'; +import {readEventFromFile} from './readEventFromFile'; + import type {FullTestRun, UtcTimeInMs} from '../../types/internal'; /** @@ -43,7 +44,7 @@ export const readEventsFromFiles = async ( fileIndex < newEventFiles.length; fileIndex += AMOUNT_OF_PARALLEL_OPEN_FILES ) { - const readPromises: Promise>[] = []; + const readPromises: Promise | undefined>[] = []; for ( let index = fileIndex; @@ -58,13 +59,14 @@ export const readEventsFromFiles = async ( newEventFilesLength: newEventFiles.length, }); - const filePath = join(EVENTS_DIRECTORY_PATH, fileName); - const promise = readFile(filePath, READ_FILE_OPTIONS).then((text) => ({fileName, text})); + const promise = readEventFromFile(fileName).then((maybeText) => + maybeText === undefined ? undefined : {fileName, text: maybeText}, + ); readPromises.push(promise); } - const filesWithNames = await Promise.all(readPromises); + const filesWithNames = (await Promise.all(readPromises)).filter((value) => value !== undefined); for (const {fileName, text} of filesWithNames) { try { diff --git a/src/utils/selectors/Selector.ts b/src/utils/selectors/Selector.ts index 88326f64..134dac2f 100644 --- a/src/utils/selectors/Selector.ts +++ b/src/utils/selectors/Selector.ts @@ -170,7 +170,6 @@ export class Selector { return result; } - // eslint-disable-next-line @typescript-eslint/consistent-return getPlaywrightLocator(): PlaywrightLocator { const args = this.args!; const selector = this.parentSelector!; diff --git a/src/utils/selectors/createCustomMethods.ts b/src/utils/selectors/createCustomMethods.ts index 88ccae43..cff42316 100644 --- a/src/utils/selectors/createCustomMethods.ts +++ b/src/utils/selectors/createCustomMethods.ts @@ -7,7 +7,7 @@ import type { } from '../../types/internal'; /** - * Creates custom `e2ed` methods of selector (additional to selector's own methods from TestCafe). + * Creates custom `e2ed` methods of selector (additional to selector's own methods). * @internal */ export const createCustomMethods = ( diff --git a/src/utils/test/beforeTest.ts b/src/utils/test/beforeTest.ts index 1c75285a..0ad23c2e 100644 --- a/src/utils/test/beforeTest.ts +++ b/src/utils/test/beforeTest.ts @@ -22,18 +22,29 @@ import type { UtcTimeInMs, } from '../../types/internal'; +import {test} from '@playwright/test'; + type Options = Readonly<{ + beforeRetryTimeout: number | undefined; retryIndex: number; runId: RunId; testFn: TestFn; testStaticOptions: TestStaticOptions; }>; +const additionalDurationToPlaywrightTestTimeoutInMs = 500; + /** * Internal before test hook. * @internal */ -export const beforeTest = ({retryIndex, runId, testFn, testStaticOptions}: Options): void => { +export const beforeTest = ({ + beforeRetryTimeout, + retryIndex, + runId, + testFn, + testStaticOptions, +}: Options): void => { const {options} = testStaticOptions; setMeta(options.meta); @@ -49,6 +60,10 @@ export const beforeTest = ({retryIndex, runId, testFn, testStaticOptions}: Optio const testIdleTimeout = options.testIdleTimeout ?? testIdleTimeoutFromConfig; const testTimeout = options.testTimeout ?? testTimeoutFromConfig; + test.setTimeout( + testTimeout + additionalDurationToPlaywrightTestTimeoutInMs + (beforeRetryTimeout ?? 0), + ); + setTestIdleTimeout(testIdleTimeout); setTestTimeout(testTimeout); diff --git a/src/utils/test/getRunTest.ts b/src/utils/test/getRunTest.ts index 0eca875a..207cd6eb 100644 --- a/src/utils/test/getRunTest.ts +++ b/src/utils/test/getRunTest.ts @@ -10,6 +10,7 @@ import {getShouldRunTest} from './getShouldRunTest'; import {getTestStaticOptions} from './getTestStaticOptions'; import {preparePage} from './preparePage'; import {runTestFn} from './runTestFn'; +import {waitBeforeRetry} from './waitBeforeRetry'; import type {PlaywrightTestArgs, TestInfo} from '@playwright/test'; @@ -26,7 +27,7 @@ export const getRunTest = ({context, page, request}: PlaywrightTestArgs, testInfo: TestInfo): Promise => { const runTest = async (): Promise => { const retryIndex = testInfo.retry + 1; - const runId = createRunId(); + const runId = createRunId(test, retryIndex); let clearPage: (() => Promise) | undefined; let hasRunError = false; @@ -43,13 +44,15 @@ export const getRunTest = return; } + const beforeRetryTimeout = await waitBeforeRetry(runId, testStaticOptions); + clearPage = await preparePage(page); - beforeTest({retryIndex, runId, testFn: test.testFn, testStaticOptions}); + beforeTest({beforeRetryTimeout, retryIndex, runId, testFn: test.testFn, testStaticOptions}); const testController = {context, page, request}; - await runTestFn({retryIndex, runId, testController, testStaticOptions}); + await runTestFn({beforeRetryTimeout, retryIndex, runId, testController, testStaticOptions}); } catch (error) { hasRunError = true; unknownRunError = error; diff --git a/src/utils/test/runTestFn.ts b/src/utils/test/runTestFn.ts index 60539315..e83f8061 100644 --- a/src/utils/test/runTestFn.ts +++ b/src/utils/test/runTestFn.ts @@ -1,10 +1,11 @@ -import {TestRunStatus} from '../../constants/internal'; +import {LogEventType, TestRunStatus} from '../../constants/internal'; import {setTestRunPromise} from '../../context/testRunPromise'; import {getTestTimeout} from '../../context/testTimeout'; import {getFullPackConfig} from '../config'; import {getTestRunEvent} from '../events'; import {enableFullMocks} from '../fullMocks'; +import {log} from '../log'; import {getPromiseWithResolveAndReject} from '../promise'; import type {PlaywrightTestArgs} from '@playwright/test'; @@ -14,6 +15,7 @@ import type {RunId, TestStaticOptions} from '../../types/internal'; const delayForTestRunPromiseResolutionAfterTestTimeoutInMs = 100; type Options = Readonly<{ + beforeRetryTimeout: number | undefined; retryIndex: number; runId: RunId; testController: PlaywrightTestArgs; @@ -25,6 +27,7 @@ type Options = Readonly<{ * @internal */ export const runTestFn = async ({ + beforeRetryTimeout, retryIndex, runId, testController, @@ -40,6 +43,10 @@ export const runTestFn = async ({ setTestRunPromise(testRunPromise); + if (beforeRetryTimeout !== undefined) { + log(`Waited for ${beforeRetryTimeout}ms before running this retry`, LogEventType.InternalUtil); + } + const {fullMocks} = getFullPackConfig(); if (status !== TestRunStatus.Skipped && fullMocks?.filterTests(testStaticOptions)) { diff --git a/src/utils/test/waitBeforeRetry.ts b/src/utils/test/waitBeforeRetry.ts new file mode 100644 index 00000000..ef618ac3 --- /dev/null +++ b/src/utils/test/waitBeforeRetry.ts @@ -0,0 +1,85 @@ +import {assertValueIsTrue} from '../asserts'; +import {getFullPackConfig} from '../config'; +import {getTestRunEventFileName, readEventFromFile} from '../fs'; +import {generalLog} from '../generalLog'; +import {getTimeoutPromise} from '../promise'; + +import type {FullTestRun, RunId, TestStaticOptions} from '../../types/internal'; + +/** + * Waits before running test for some time from pack config (for retries). + * @internal + */ +export const waitBeforeRetry = async ( + runId: RunId, + testStaticOptions: TestStaticOptions, +): Promise => { + const indexOfRetryIndex = runId.lastIndexOf('-'); + + assertValueIsTrue( + indexOfRetryIndex > 0 && indexOfRetryIndex < runId.length - 1, + 'runId has dash', + {runId, testStaticOptions}, + ); + + const retryIndex = Number(runId.slice(indexOfRetryIndex + 1)); + + assertValueIsTrue( + Number.isInteger(retryIndex) && retryIndex > 0, + 'retryIndex from runId is correct', + {runId, testStaticOptions}, + ); + + const previousRetryIndex = retryIndex - 1; + + if (previousRetryIndex < 1) { + return; + } + + const previousRunId = `${runId.slice(0, indexOfRetryIndex)}-${previousRetryIndex}` as RunId; + + const fileName = getTestRunEventFileName(previousRunId); + const fileText = await readEventFromFile(fileName); + + if (fileText === undefined) { + generalLog('Cannot find JSON file of previous test run', { + previousRunId, + runId, + testStaticOptions, + }); + + return; + } + + try { + const fullTestRun = JSON.parse(fileText) as FullTestRun; + + const {runError, status} = fullTestRun; + const {waitBeforeRetry: waitBeforeRetryFromConfig} = getFullPackConfig(); + + const previousError = runError === undefined ? undefined : String(runError); + + const timeoutInMs = await waitBeforeRetryFromConfig({ + previousError, + retryIndex, + status, + testStaticOptions, + }); + + if (timeoutInMs === 0) { + return; + } + + await getTimeoutPromise(timeoutInMs); + + return timeoutInMs; + } catch (error) { + generalLog('Caught an error on getting timeout for "before retry" waiting', { + error, + runId, + testStaticOptions, + }); + + return undefined; + } +}; From 0ae2c510e397f788530d3a49446e51a174550060 Mon Sep 17 00:00:00 2001 From: uid11 Date: Tue, 29 Oct 2024 03:48:02 +0300 Subject: [PATCH 2/6] FI-1498 feat: `pressKey` can receive selector --- src/actions/pressKey.ts | 65 +++++++++++++++++++++++++++++++++++++---- tsconfig.json | 1 + 2 files changed, 60 insertions(+), 6 deletions(-) diff --git a/src/actions/pressKey.ts b/src/actions/pressKey.ts index ef4ea94e..4e9c694c 100644 --- a/src/actions/pressKey.ts +++ b/src/actions/pressKey.ts @@ -1,20 +1,73 @@ import {LogEventType} from '../constants/internal'; import {getPlaywrightPage} from '../useContext'; +import {E2edError} from '../utils/error'; import {log} from '../utils/log'; +import {getDescriptionFromSelector} from '../utils/selectors'; -import type {Keyboard} from '@playwright/test'; +import type {KeyboardPressKey, Selector} from '../types/internal'; -import type {KeyboardPressKey} from '../types/internal'; +type Options = Readonly<{delay?: number; timeout?: number}>; -type Options = Parameters[1]; +type PressKey = (( + this: void, + selector: Selector, + key: KeyboardPressKey, + options?: Options, +) => Promise) & + ((this: void, key: KeyboardPressKey, options?: Options) => Promise); /** * Presses the specified keyboard keys. */ -export const pressKey = async (key: KeyboardPressKey, options: Options = {}): Promise => { - log(`Press keyboard key: "${key}"`, options, LogEventType.InternalAction); +export const pressKey: PressKey = async ( + keyOrSelector: KeyboardPressKey | Selector, + keyOrOptions?: KeyboardPressKey | Options, + maybeOptions?: Options, +): Promise => { + let key: KeyboardPressKey; + let selector: Selector | undefined; + let options: Options; + + if (typeof keyOrSelector === 'string') { + key = keyOrSelector; + + if (typeof keyOrOptions === 'string') { + throw new E2edError('keyOrOptions is string', { + keyOrOptions, + keyOrSelector, + maybeOptions, + }); + } + + options = keyOrOptions ?? {}; + } else { + selector = keyOrSelector; + + if (typeof keyOrOptions !== 'string') { + throw new E2edError('keyOrOptions is not string', { + keyOrOptions, + keyOrSelector, + maybeOptions, + }); + } + + key = keyOrOptions; + + options = maybeOptions ?? {}; + } + + const withDescription = + selector !== undefined + ? ` on element with description ${getDescriptionFromSelector(selector)}` + : ''; + + log(`Press keyboard key${withDescription}: "${key}"`, options, LogEventType.InternalAction); const page = getPlaywrightPage(); - await page.keyboard.press(key, options); + if (selector !== undefined) { + await selector.getPlaywrightLocator().press(key, options); + } else { + await page.keyboard.press(key, options); + } }; diff --git a/tsconfig.json b/tsconfig.json index 9a484543..35d28100 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -14,6 +14,7 @@ "noImplicitReturns": true, "noPropertyAccessFromIndexSignature": true, "noUncheckedIndexedAccess": true, + "noUncheckedSideEffectImports": true, "noUnusedLocals": true, "outDir": "build", "paths": { From 1b32763ee8f13d8ba8d400b1545825d871ed3da2 Mon Sep 17 00:00:00 2001 From: uid11 Date: Sun, 3 Nov 2024 03:14:17 +0300 Subject: [PATCH 3/6] FI-1484 feat: rename `getParamsFromUrl` to `getParamsFromUrlOrThrow` fix: test timeout and interrupt timeout during `waitBeforeRetry` --- autotests/packs/allTests.ts | 6 ++- autotests/routes/apiRoutes/AddUser.ts | 2 +- autotests/routes/apiRoutes/CreateProduct.ts | 2 +- autotests/routes/pageRoutes/Search.ts | 2 +- .../internalTypeTests/mockApiRoute.skip.ts | 4 +- .../internalTypeTests/waitForEvents.skip.ts | 4 +- autotests/tests/main/exists.ts | 4 +- autotests/tests/mockApiRoute.ts | 2 +- scripts/writePrunedPackageJson.ts | 2 +- src/Route.ts | 2 +- src/actions/mock/mockApiRoute.ts | 4 +- src/types/routes.ts | 4 +- src/utils/fullMocks/FullMocksRoute.ts | 2 +- src/utils/getRouteInstanceFromUrl.ts | 8 ++-- src/utils/resourceUsage.ts | 8 ++-- src/utils/retry/runPackWithRetries.ts | 2 +- src/utils/test/beforeTest.ts | 6 +-- src/utils/test/getPreviousRunId.ts | 37 ++++++++++++++++ src/utils/test/waitBeforeRetry.ts | 43 +++++++++---------- 19 files changed, 89 insertions(+), 55 deletions(-) create mode 100644 src/utils/test/getPreviousRunId.ts diff --git a/autotests/packs/allTests.ts b/autotests/packs/allTests.ts index eefbc47e..bac8a1f8 100644 --- a/autotests/packs/allTests.ts +++ b/autotests/packs/allTests.ts @@ -33,8 +33,10 @@ const filterTestsIntoPack: FilterTestsIntoPack = ({options}) => options.meta.tes const userAgent = 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.35 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.35'; -const packTimeoutInMinutes = 5; const msInMinute = 60_000; +const packTimeoutInMinutes = 5; + +const waitBeforeRetryTimeout = 1_000; /** * Pack of tests or tasks (pack configuration object). @@ -82,7 +84,7 @@ export const pack: Pack = { userAgent, viewportHeight: 1080, viewportWidth: 1920, - waitBeforeRetry: () => 0, + waitBeforeRetry: () => waitBeforeRetryTimeout, waitForAllRequestsComplete: { maxIntervalBetweenRequestsInMs: 500, timeout: 30_000, diff --git a/autotests/routes/apiRoutes/AddUser.ts b/autotests/routes/apiRoutes/AddUser.ts index 23bb290d..643b0581 100644 --- a/autotests/routes/apiRoutes/AddUser.ts +++ b/autotests/routes/apiRoutes/AddUser.ts @@ -12,7 +12,7 @@ const pathStart = '/api/users'; * Client API route for adding user-worker. */ export class AddUser extends ApiRoute { - static override getParamsFromUrl(url: Url): Params { + static override getParamsFromUrlOrThrow(url: Url): Params { const urlObject = new URL(url); assertValueIsTrue( diff --git a/autotests/routes/apiRoutes/CreateProduct.ts b/autotests/routes/apiRoutes/CreateProduct.ts index 9dd9889d..b7b0c338 100644 --- a/autotests/routes/apiRoutes/CreateProduct.ts +++ b/autotests/routes/apiRoutes/CreateProduct.ts @@ -18,7 +18,7 @@ export class CreateProduct extends ApiRoute< ApiCreateProductRequest, ApiCreateProductResponse > { - static override getParamsFromUrl(url: Url): Params { + static override getParamsFromUrlOrThrow(url: Url): Params { const urlObject = new URL(url); assertValueIsTrue( diff --git a/autotests/routes/pageRoutes/Search.ts b/autotests/routes/pageRoutes/Search.ts index 27cd282c..f8bae3da 100644 --- a/autotests/routes/pageRoutes/Search.ts +++ b/autotests/routes/pageRoutes/Search.ts @@ -11,7 +11,7 @@ type Params = Readonly<{searchQuery: string}> | undefined; * Route of the Search page. */ export class Search extends PageRoute { - static override getParamsFromUrl(url: Url): Params { + static override getParamsFromUrlOrThrow(url: Url): Params { const {pathname, searchParams} = new URL(url); assertValueIsTrue(pathname === '/search', 'search route matches on url', {url}); diff --git a/autotests/tests/internalTypeTests/mockApiRoute.skip.ts b/autotests/tests/internalTypeTests/mockApiRoute.skip.ts index 7abe5762..6c88eaa0 100644 --- a/autotests/tests/internalTypeTests/mockApiRoute.skip.ts +++ b/autotests/tests/internalTypeTests/mockApiRoute.skip.ts @@ -35,10 +35,10 @@ void mockApiRoute(Main, apiMockFunction); // @ts-expect-error: unmockApiRoute require API route as first argument void unmockApiRoute(Main); -// @ts-expect-error: mockApiRoute require API route with static method getParamsFromUrl +// @ts-expect-error: mockApiRoute require API route with static method getParamsFromUrlOrThrow void mockApiRoute(CreateDevice, apiMockFunction); -// @ts-expect-error: unmockApiRoute require API route with static method getParamsFromUrl +// @ts-expect-error: unmockApiRoute require API route with static method getParamsFromUrlOrThrow void unmockApiRoute(CreateDevice); // ok diff --git a/autotests/tests/internalTypeTests/waitForEvents.skip.ts b/autotests/tests/internalTypeTests/waitForEvents.skip.ts index 03e96e42..c4832401 100644 --- a/autotests/tests/internalTypeTests/waitForEvents.skip.ts +++ b/autotests/tests/internalTypeTests/waitForEvents.skip.ts @@ -63,7 +63,7 @@ void waitForRequestToRoute(AddUser, ({delay}, {requestBody, url}) => { request.requestBody.job === 'foo' && 'delay' in routeParams && routeParams.delay > 0, ); -// @ts-expect-error: waitForRequestToRoute does not accept routes without `getParamsFromUrl` method +// @ts-expect-error: waitForRequestToRoute does not accept routes without `getParamsFromUrlOrThrow` method void waitForRequestToRoute(GetUser); // ok @@ -84,5 +84,5 @@ void waitForResponseToRoute(AddUser, ({delay}, {responseBody, request: {requestB routeParams.delay > 0, ); -// @ts-expect-error: waitForResponseToRoute does not accept routes without `getParamsFromUrl` method +// @ts-expect-error: waitForResponseToRoute does not accept routes without `getParamsFromUrlOrThrow` method void waitForResponseToRoute(GetUser); diff --git a/autotests/tests/main/exists.ts b/autotests/tests/main/exists.ts index 1f1d0d16..43507316 100644 --- a/autotests/tests/main/exists.ts +++ b/autotests/tests/main/exists.ts @@ -77,7 +77,7 @@ test('exists', {meta: {testId: '1'}, testIdleTimeout: 10_000, testTimeout: 15_00 const searchPage = await assertPage(Search, {searchQuery}); /** - * Do not use the following pageParams and url (by getParamsFromUrl) checks in your code. + * Do not use the following pageParams and url (by getParamsFromUrlOrThrow) checks in your code. * These are e2ed internal checks. Use `assertPage` instead. */ await expect(searchPage.pageParams, 'pageParams is correct after assertPage').eql({ @@ -86,7 +86,7 @@ test('exists', {meta: {testId: '1'}, testIdleTimeout: 10_000, testTimeout: 15_00 const url = await getDocumentUrl(); - await expect(SearchRoute.getParamsFromUrl(url), 'page url has expected params').eql({ + await expect(SearchRoute.getParamsFromUrlOrThrow(url), 'page url has expected params').eql({ searchQuery, }); diff --git a/autotests/tests/mockApiRoute.ts b/autotests/tests/mockApiRoute.ts index 7d985e5b..a5c3c149 100644 --- a/autotests/tests/mockApiRoute.ts +++ b/autotests/tests/mockApiRoute.ts @@ -37,7 +37,7 @@ test( const fetchUrl = `https://reqres.in/api/product/${productId}?size=${product.size}` as Url; - const productRouteParams = CreateProductRoute.getParamsFromUrl(fetchUrl); + const productRouteParams = CreateProductRoute.getParamsFromUrlOrThrow(fetchUrl); const productRouteFromUrl = new CreateProductRoute(productRouteParams); diff --git a/scripts/writePrunedPackageJson.ts b/scripts/writePrunedPackageJson.ts index 5f64dc4a..301d956d 100644 --- a/scripts/writePrunedPackageJson.ts +++ b/scripts/writePrunedPackageJson.ts @@ -1,5 +1,5 @@ /** - * @file Generates and writes pruned lightweight package.json for npm package. + * @file Generates and writes pruned lightweight `package.json` for npm package. */ import {writeFileSync} from 'node:fs'; diff --git a/src/Route.ts b/src/Route.ts index e44c4094..63c6c8ec 100644 --- a/src/Route.ts +++ b/src/Route.ts @@ -25,7 +25,7 @@ export abstract class Route { * Returns route params from the passed url. * @throws {Error} If the route does not match on the url. */ - static getParamsFromUrl?(url: Url, method?: Method): unknown; + static getParamsFromUrlOrThrow?(url: Url, method?: Method): unknown; /** * Returns the url of the route. diff --git a/src/actions/mock/mockApiRoute.ts b/src/actions/mock/mockApiRoute.ts index b3300d34..f0f2e12c 100644 --- a/src/actions/mock/mockApiRoute.ts +++ b/src/actions/mock/mockApiRoute.ts @@ -17,9 +17,9 @@ import type { /** * Mock API for some API route. - * Applicable only for routes with the `getParamsFromUrl` method. + * Applicable only for routes with the `getParamsFromUrlOrThrow` method. * The mock is applied to a request that matches the route by url - * (by methods `getParamsFromUrl` and `isMatchUrl`) and by HTTP method (by `getMethod`). + * (by methods `getParamsFromUrlOrThrow` and `isMatchUrl`) and by HTTP method (by `getMethod`). */ export const mockApiRoute = async < RouteParams, diff --git a/src/types/routes.ts b/src/types/routes.ts index 9e39c440..6f7eec10 100644 --- a/src/types/routes.ts +++ b/src/types/routes.ts @@ -16,7 +16,7 @@ export type ApiRouteClassType< }; /** - * API Route class with static method getParamsFromUrl. + * API Route class with static method getParamsFromUrlOrThrow. */ export type ApiRouteClassTypeWithGetParamsFromUrl< RouteParams = Any, @@ -24,5 +24,5 @@ export type ApiRouteClassTypeWithGetParamsFromUrl< SomeResponse extends Response = Response, > = ApiRouteClassType & Readonly<{ - getParamsFromUrl: Exclude<(typeof ApiRoute)['getParamsFromUrl'], undefined>; + getParamsFromUrlOrThrow: Exclude<(typeof ApiRoute)['getParamsFromUrlOrThrow'], undefined>; }>; diff --git a/src/utils/fullMocks/FullMocksRoute.ts b/src/utils/fullMocks/FullMocksRoute.ts index 0cf88b03..25ef975e 100644 --- a/src/utils/fullMocks/FullMocksRoute.ts +++ b/src/utils/fullMocks/FullMocksRoute.ts @@ -14,7 +14,7 @@ import type {FullMocksRouteParams, Url} from '../../types/internal'; * @internal */ export class FullMocksRoute extends ApiRoute { - static override getParamsFromUrl(url: Url): FullMocksRouteParams { + static override getParamsFromUrlOrThrow(url: Url): FullMocksRouteParams { const {fullMocks: fullMocksConfig} = getFullPackConfig(); const fullMocksState = getFullMocksState(); diff --git a/src/utils/getRouteInstanceFromUrl.ts b/src/utils/getRouteInstanceFromUrl.ts index 43bc92ac..8bc0b36c 100644 --- a/src/utils/getRouteInstanceFromUrl.ts +++ b/src/utils/getRouteInstanceFromUrl.ts @@ -9,9 +9,9 @@ type Return = /** * Get route instance and route params from url and method by route class. - * @throws {Error} If `url` accepted without errors by `getParamsFromUrl`, + * @throws {Error} If `url` accepted without errors by `getParamsFromUrlOrThrow`, * but not match by `isMatchUrl` method. - * If `url` not accepted by `getParamsFromUrl`, returns `undefined`. + * If `url` not accepted by `getParamsFromUrlOrThrow`, returns `undefined`. * @internal */ export const getRouteInstanceFromUrl = ( @@ -22,7 +22,7 @@ export const getRouteInstanceFromUrl = ( let routeParams: RouteParams | undefined; try { - routeParams = Route.getParamsFromUrl(url) as RouteParams; + routeParams = Route.getParamsFromUrlOrThrow(url) as RouteParams; route = new Route(routeParams); } catch { return undefined; @@ -30,7 +30,7 @@ export const getRouteInstanceFromUrl = ( if (route.isMatchUrl(url) !== true) { throw new E2edError( - `Inconsistency in "${Route.name}" route: isMatchUrl does not accept url accepted without errors by getParamsFromUrl`, + `Inconsistency in "${Route.name}" route: isMatchUrl does not accept url accepted without errors by getParamsFromUrlOrThrow`, {route, url}, ); } diff --git a/src/utils/resourceUsage.ts b/src/utils/resourceUsage.ts index 505ec509..83885b85 100644 --- a/src/utils/resourceUsage.ts +++ b/src/utils/resourceUsage.ts @@ -10,7 +10,7 @@ const availableCpuCount = availableParallelism(); let previousCores = cpus(); let previousCpuUsage = process.cpuUsage(); let previousTimeInMs = Date.now(); -let timeoutInterval: NodeJS.Timeout; +let timeoutObject: NodeJS.Timeout; const logResourceUsage = (): void => { const cores = cpus(); @@ -65,7 +65,7 @@ const logResourceUsage = (): void => { * @internal */ export const startResourceUsageReading = (resourceUsageReadingInternal: number): void => { - assertValueIsUndefined(timeoutInterval, 'timeoutInterval in not defined', { + assertValueIsUndefined(timeoutObject, 'timeoutObject in not defined', { resourceUsageReadingInternal, }); @@ -73,7 +73,7 @@ export const startResourceUsageReading = (resourceUsageReadingInternal: number): return; } - timeoutInterval = setInterval(logResourceUsage, resourceUsageReadingInternal); + timeoutObject = setInterval(logResourceUsage, resourceUsageReadingInternal); - timeoutInterval.unref(); + timeoutObject.unref(); }; diff --git a/src/utils/retry/runPackWithRetries.ts b/src/utils/retry/runPackWithRetries.ts index 936016b3..b189c685 100644 --- a/src/utils/retry/runPackWithRetries.ts +++ b/src/utils/retry/runPackWithRetries.ts @@ -47,7 +47,7 @@ export const runPackWithRetries = async (): Promise => { endE2ed(EndE2edReason.RetriesCycleEnded); } catch (error) { - generalLog('Caught an error on running testso', { + generalLog('Caught an error on running test', { error, retriesState: truncateRetriesStateForLogs(retriesState), }); diff --git a/src/utils/test/beforeTest.ts b/src/utils/test/beforeTest.ts index 0ad23c2e..29e870ec 100644 --- a/src/utils/test/beforeTest.ts +++ b/src/utils/test/beforeTest.ts @@ -32,7 +32,7 @@ type Options = Readonly<{ testStaticOptions: TestStaticOptions; }>; -const additionalDurationToPlaywrightTestTimeoutInMs = 500; +const additionToPlaywrightTestTimeout = 500; /** * Internal before test hook. @@ -60,9 +60,7 @@ export const beforeTest = ({ const testIdleTimeout = options.testIdleTimeout ?? testIdleTimeoutFromConfig; const testTimeout = options.testTimeout ?? testTimeoutFromConfig; - test.setTimeout( - testTimeout + additionalDurationToPlaywrightTestTimeoutInMs + (beforeRetryTimeout ?? 0), - ); + test.setTimeout(testTimeout + additionToPlaywrightTestTimeout + (beforeRetryTimeout ?? 0)); setTestIdleTimeout(testIdleTimeout); setTestTimeout(testTimeout); diff --git a/src/utils/test/getPreviousRunId.ts b/src/utils/test/getPreviousRunId.ts new file mode 100644 index 00000000..a74d95c1 --- /dev/null +++ b/src/utils/test/getPreviousRunId.ts @@ -0,0 +1,37 @@ +import {assertValueIsTrue} from '../asserts'; + +import type {RunId} from '../../types/internal'; + +type Return = Readonly<{previousRunId: RunId | undefined; retryIndex: number}>; + +/** + * Get previous `runId` if any, by current `runId`. + * @internal + */ +export const getPreviousRunId = (runId: RunId): Return => { + const indexOfRetryIndex = runId.lastIndexOf('-'); + + assertValueIsTrue( + indexOfRetryIndex > 0 && indexOfRetryIndex < runId.length - 1, + 'runId has dash', + {runId}, + ); + + const retryIndex = Number(runId.slice(indexOfRetryIndex + 1)); + + assertValueIsTrue( + Number.isInteger(retryIndex) && retryIndex > 0, + 'retryIndex from runId is correct', + {runId}, + ); + + const previousRetryIndex = retryIndex - 1; + + if (previousRetryIndex < 1) { + return {previousRunId: undefined, retryIndex}; + } + + const previousRunId = `${runId.slice(0, indexOfRetryIndex)}-${previousRetryIndex}` as RunId; + + return {previousRunId, retryIndex}; +}; diff --git a/src/utils/test/waitBeforeRetry.ts b/src/utils/test/waitBeforeRetry.ts index ef618ac3..d33428a0 100644 --- a/src/utils/test/waitBeforeRetry.ts +++ b/src/utils/test/waitBeforeRetry.ts @@ -1,11 +1,16 @@ -import {assertValueIsTrue} from '../asserts'; import {getFullPackConfig} from '../config'; -import {getTestRunEventFileName, readEventFromFile} from '../fs'; +import {getTestRunEventFileName, readEventFromFile, writeLogEventTime} from '../fs'; import {generalLog} from '../generalLog'; import {getTimeoutPromise} from '../promise'; +import {getPreviousRunId} from './getPreviousRunId'; + import type {FullTestRun, RunId, TestStaticOptions} from '../../types/internal'; +import {test} from '@playwright/test'; + +const additionToTimeout = 500; + /** * Waits before running test for some time from pack config (for retries). * @internal @@ -14,30 +19,12 @@ export const waitBeforeRetry = async ( runId: RunId, testStaticOptions: TestStaticOptions, ): Promise => { - const indexOfRetryIndex = runId.lastIndexOf('-'); - - assertValueIsTrue( - indexOfRetryIndex > 0 && indexOfRetryIndex < runId.length - 1, - 'runId has dash', - {runId, testStaticOptions}, - ); - - const retryIndex = Number(runId.slice(indexOfRetryIndex + 1)); - - assertValueIsTrue( - Number.isInteger(retryIndex) && retryIndex > 0, - 'retryIndex from runId is correct', - {runId, testStaticOptions}, - ); + const {previousRunId, retryIndex} = getPreviousRunId(runId); - const previousRetryIndex = retryIndex - 1; - - if (previousRetryIndex < 1) { + if (previousRunId === undefined) { return; } - const previousRunId = `${runId.slice(0, indexOfRetryIndex)}-${previousRetryIndex}` as RunId; - const fileName = getTestRunEventFileName(previousRunId); const fileText = await readEventFromFile(fileName); @@ -55,7 +42,7 @@ export const waitBeforeRetry = async ( const fullTestRun = JSON.parse(fileText) as FullTestRun; const {runError, status} = fullTestRun; - const {waitBeforeRetry: waitBeforeRetryFromConfig} = getFullPackConfig(); + const {testIdleTimeout, waitBeforeRetry: waitBeforeRetryFromConfig} = getFullPackConfig(); const previousError = runError === undefined ? undefined : String(runError); @@ -70,8 +57,18 @@ export const waitBeforeRetry = async ( return; } + test.setTimeout(timeoutInMs + additionToTimeout); + + const timeoutObject = setInterval(() => { + void writeLogEventTime(); + }, testIdleTimeout); + + timeoutObject.unref(); + await getTimeoutPromise(timeoutInMs); + clearInterval(timeoutObject); + return timeoutInMs; } catch (error) { generalLog('Caught an error on getting timeout for "before retry" waiting', { From d15265ca49effbf1d1d895638dc85cbaefd3fe89 Mon Sep 17 00:00:00 2001 From: uid11 Date: Sun, 3 Nov 2024 05:15:27 +0300 Subject: [PATCH 4/6] FI-1485 fix: `waitForRequestToRoute` when `routeParams` = `undefined` --- src/actions/waitFor/waitForRequestToRoute.ts | 13 +++++++++---- src/actions/waitFor/waitForResponseToRoute.ts | 13 +++++++++---- src/utils/test/runTestFn.ts | 5 ++++- 3 files changed, 22 insertions(+), 9 deletions(-) diff --git a/src/actions/waitFor/waitForRequestToRoute.ts b/src/actions/waitFor/waitForRequestToRoute.ts index e5cedb63..76920606 100644 --- a/src/actions/waitFor/waitForRequestToRoute.ts +++ b/src/actions/waitFor/waitForRequestToRoute.ts @@ -1,6 +1,7 @@ import {LogEventType} from '../../constants/internal'; -import {assertValueIsDefined, assertValueIsUndefined} from '../../utils/asserts'; +import {assertValueIsTrue} from '../../utils/asserts'; import {getFullPackConfig} from '../../utils/config'; +import {E2edError} from '../../utils/error'; import {setCustomInspectOnFunction} from '../../utils/fn'; import {getDurationWithUnits} from '../../utils/getDurationWithUnits'; import {getRouteInstanceFromUrl} from '../../utils/getRouteInstanceFromUrl'; @@ -53,7 +54,9 @@ export const waitForRequestToRoute = async < LogEventType.InternalAction, ); - let routeParams: RouteParams | undefined; + const sentinelValue: unique symbol = Symbol('sentinel value'); + + let routeParams: RouteParams | typeof sentinelValue = sentinelValue; const predicateForRequest: RequestPredicate = async (request) => { const maypeRouteWithRouteParams = getRouteInstanceFromUrl(request.url, Route); @@ -70,7 +73,7 @@ export const waitForRequestToRoute = async < return false; } - assertValueIsUndefined(routeParams, 'routeParams is not defined'); + assertValueIsTrue(routeParams === sentinelValue, 'routeParams was not setted'); routeParams = currentRouteParams; @@ -79,7 +82,9 @@ export const waitForRequestToRoute = async < const request = await waitForRequest(predicateForRequest, {skipLogs: true, timeout}); - assertValueIsDefined(routeParams, 'routeParams is defined', {predicate, request}); + if (routeParams === sentinelValue) { + throw new E2edError('routeParams is not setted', {predicate, request}); + } const waitWithUnits = getDurationWithUnits(Date.now() - startTimeInMs); diff --git a/src/actions/waitFor/waitForResponseToRoute.ts b/src/actions/waitFor/waitForResponseToRoute.ts index 0b4b9b4d..2985b5fb 100644 --- a/src/actions/waitFor/waitForResponseToRoute.ts +++ b/src/actions/waitFor/waitForResponseToRoute.ts @@ -1,6 +1,7 @@ import {LogEventType} from '../../constants/internal'; -import {assertValueIsDefined, assertValueIsUndefined} from '../../utils/asserts'; +import {assertValueIsTrue} from '../../utils/asserts'; import {getFullPackConfig} from '../../utils/config'; +import {E2edError} from '../../utils/error'; import {setCustomInspectOnFunction} from '../../utils/fn'; import {getDurationWithUnits} from '../../utils/getDurationWithUnits'; import {getRouteInstanceFromUrl} from '../../utils/getRouteInstanceFromUrl'; @@ -53,7 +54,9 @@ export const waitForResponseToRoute = async < LogEventType.InternalAction, ); - let routeParams: RouteParams | undefined; + const sentinelValue: unique symbol = Symbol('sentinel value'); + + let routeParams: RouteParams | typeof sentinelValue = sentinelValue; const predicateForResponse: ResponsePredicate = async (response) => { const {request} = response; @@ -71,7 +74,7 @@ export const waitForResponseToRoute = async < return false; } - assertValueIsUndefined(routeParams, 'routeParams is not defined'); + assertValueIsTrue(routeParams === sentinelValue, 'routeParams was not setted'); routeParams = currentRouteParams; @@ -80,7 +83,9 @@ export const waitForResponseToRoute = async < const response = await waitForResponse(predicateForResponse, {skipLogs: true, timeout}); - assertValueIsDefined(routeParams, 'routeParams is defined', {predicate, response}); + if (routeParams === sentinelValue) { + throw new E2edError('routeParams is not setted', {predicate, response}); + } const waitWithUnits = getDurationWithUnits(Date.now() - startTimeInMs); diff --git a/src/utils/test/runTestFn.ts b/src/utils/test/runTestFn.ts index e83f8061..52aa55e4 100644 --- a/src/utils/test/runTestFn.ts +++ b/src/utils/test/runTestFn.ts @@ -5,6 +5,7 @@ import {getTestTimeout} from '../../context/testTimeout'; import {getFullPackConfig} from '../config'; import {getTestRunEvent} from '../events'; import {enableFullMocks} from '../fullMocks'; +import {getDurationWithUnits} from '../getDurationWithUnits'; import {log} from '../log'; import {getPromiseWithResolveAndReject} from '../promise'; @@ -44,7 +45,9 @@ export const runTestFn = async ({ setTestRunPromise(testRunPromise); if (beforeRetryTimeout !== undefined) { - log(`Waited for ${beforeRetryTimeout}ms before running this retry`, LogEventType.InternalUtil); + const timeoutWithUnits = getDurationWithUnits(beforeRetryTimeout); + + log(`Waited for ${timeoutWithUnits} before running this retry`, LogEventType.InternalUtil); } const {fullMocks} = getFullPackConfig(); From 058fb2a2b33e672a294e28077eda4c34fd1ec4a4 Mon Sep 17 00:00:00 2001 From: uid11 Date: Sun, 3 Nov 2024 07:09:15 +0300 Subject: [PATCH 5/6] FI-1506 feat: add bin-command `e2ed-install-browsers` --- README.md | 7 +++++++ scripts/writePrunedPackageJson.ts | 23 +++++++++++++++++++---- 2 files changed, 26 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index b3fbf68b..d633a78b 100644 --- a/README.md +++ b/README.md @@ -31,6 +31,13 @@ Install the latest version of `e2ed` in devDependencies with the exact version: npm install e2ed --save-dev --save-exact ``` +Install [Playwright](https://playwright.dev/) [browsers](https://playwright.dev/docs/browsers) +(only `Chromium` for now): + +```sh +npx e2ed-install-browsers +``` + ### Initialize Initialize `e2ed` in the project; this will add an `autotests` directory diff --git a/scripts/writePrunedPackageJson.ts b/scripts/writePrunedPackageJson.ts index 301d956d..7042893d 100644 --- a/scripts/writePrunedPackageJson.ts +++ b/scripts/writePrunedPackageJson.ts @@ -5,14 +5,29 @@ import {writeFileSync} from 'node:fs'; import {join} from 'node:path'; -import originaPackageJson from '../package.json'; +import originalPackageJson from '../package.json'; + +type OriginalPackageJson = typeof originalPackageJson; + +type PrunedPackageJson = Omit & { + bin: OriginalPackageJson['bin'] & {['e2ed-install-browsers']: string}; + devDependencies: undefined; + scripts: undefined; +}; const prunedPackageJsonPath = join(__dirname, 'node_modules', 'e2ed', 'package.json'); -const prunedPackageJson: Partial = {...originaPackageJson}; +const playwrightVersion = originalPackageJson.dependencies['@playwright/test']; -delete prunedPackageJson.devDependencies; -delete prunedPackageJson.scripts; +const prunedPackageJson: PrunedPackageJson = { + ...originalPackageJson, + bin: { + ...originalPackageJson.bin, + 'e2ed-install-browsers': `npm install --global @playwright/browser-chromium@${playwrightVersion}`, + }, + devDependencies: undefined, + scripts: undefined, +}; const prunedPackageJsonText = JSON.stringify(prunedPackageJson, null, 2); From 33278fa98750915a3cc2747fb53c60db96901b49 Mon Sep 17 00:00:00 2001 From: uid11 Date: Sun, 3 Nov 2024 18:10:47 +0300 Subject: [PATCH 6/6] chore: update Playwright to 1.48.2 chore: update `@types/node` --- package-lock.json | 44 ++++++++++++++++++++++---------------------- package.json | 6 +++--- 2 files changed, 25 insertions(+), 25 deletions(-) diff --git a/package-lock.json b/package-lock.json index f06651d2..9776fb7a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,7 +9,7 @@ "version": "0.18.15", "license": "MIT", "dependencies": { - "@playwright/test": "1.48.1", + "@playwright/test": "1.48.2", "create-locator": "0.0.25", "get-modules-graph": "0.0.9", "globby": "11.1.0" @@ -19,8 +19,8 @@ "e2ed-init": "bin/init.js" }, "devDependencies": { - "@playwright/browser-chromium": "1.48.1", - "@types/node": "22.7.6", + "@playwright/browser-chromium": "1.48.2", + "@types/node": "22.8.7", "@typescript-eslint/eslint-plugin": "7.18.0", "@typescript-eslint/parser": "7.18.0", "assert-modules-support-case-insensitive-fs": "1.0.1", @@ -179,26 +179,26 @@ } }, "node_modules/@playwright/browser-chromium": { - "version": "1.48.1", - "resolved": "https://registry.npmjs.org/@playwright/browser-chromium/-/browser-chromium-1.48.1.tgz", - "integrity": "sha512-WNpIE+CpO4XNhdNTC/mVE8m/1gYR/hPswL8dnIZzElqxsOPm7qxIBdAChWm/7q1kkMiQ+gEGAaXXQfqL0vR28w==", + "version": "1.48.2", + "resolved": "https://registry.npmjs.org/@playwright/browser-chromium/-/browser-chromium-1.48.2.tgz", + "integrity": "sha512-TvYJ5PFaDPYNlKpvPSftBbPTnu75VdRKjoMjmkd7/P79rFIBD+v6K4wU8XR6PlAqqFdPcfLL5XXZnRwTRixbDQ==", "dev": true, "hasInstallScript": true, "license": "Apache-2.0", "dependencies": { - "playwright-core": "1.48.1" + "playwright-core": "1.48.2" }, "engines": { "node": ">=18" } }, "node_modules/@playwright/test": { - "version": "1.48.1", - "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.48.1.tgz", - "integrity": "sha512-s9RtWoxkOLmRJdw3oFvhFbs9OJS0BzrLUc8Hf6l2UdCNd1rqeEyD4BhCJkvzeEoD1FsK4mirsWwGerhVmYKtZg==", + "version": "1.48.2", + "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.48.2.tgz", + "integrity": "sha512-54w1xCWfXuax7dz4W2M9uw0gDyh+ti/0K/MxcCUxChFh37kkdxPdfZDw5QBbuPUJHr1CiHJ1hXgSs+GgeQc5Zw==", "license": "Apache-2.0", "dependencies": { - "playwright": "1.48.1" + "playwright": "1.48.2" }, "bin": { "playwright": "cli.js" @@ -227,13 +227,13 @@ "dev": true }, "node_modules/@types/node": { - "version": "22.7.6", - "resolved": "https://registry.npmjs.org/@types/node/-/node-22.7.6.tgz", - "integrity": "sha512-/d7Rnj0/ExXDMcioS78/kf1lMzYk4BZV8MZGTBKzTGZ6/406ukkbYlIsZmMPhcR5KlkunDHQLrtAVmSq7r+mSw==", + "version": "22.8.7", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.8.7.tgz", + "integrity": "sha512-LidcG+2UeYIWcMuMUpBKOnryBWG/rnmOHQR5apjn8myTQcx3rinFRn7DcIFhMnS0PPFSC6OafdIKEad0lj6U0Q==", "dev": true, "license": "MIT", "dependencies": { - "undici-types": "~6.19.2" + "undici-types": "~6.19.8" } }, "node_modules/@types/semver": { @@ -2729,12 +2729,12 @@ } }, "node_modules/playwright": { - "version": "1.48.1", - "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.48.1.tgz", - "integrity": "sha512-j8CiHW/V6HxmbntOfyB4+T/uk08tBy6ph0MpBXwuoofkSnLmlfdYNNkFTYD6ofzzlSqLA1fwH4vwvVFvJgLN0w==", + "version": "1.48.2", + "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.48.2.tgz", + "integrity": "sha512-NjYvYgp4BPmiwfe31j4gHLa3J7bD2WiBz8Lk2RoSsmX38SVIARZ18VYjxLjAcDsAhA+F4iSEXTSGgjua0rrlgQ==", "license": "Apache-2.0", "dependencies": { - "playwright-core": "1.48.1" + "playwright-core": "1.48.2" }, "bin": { "playwright": "cli.js" @@ -2747,9 +2747,9 @@ } }, "node_modules/playwright-core": { - "version": "1.48.1", - "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.48.1.tgz", - "integrity": "sha512-Yw/t4VAFX/bBr1OzwCuOMZkY1Cnb4z/doAFSwf4huqAGWmf9eMNjmK7NiOljCdLmxeRYcGPPmcDgU0zOlzP0YA==", + "version": "1.48.2", + "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.48.2.tgz", + "integrity": "sha512-sjjw+qrLFlriJo64du+EK0kJgZzoQPsabGF4lBvsid+3CNIZIYLgnMj9V6JY5VhM2Peh20DJWIVpVljLLnlawA==", "license": "Apache-2.0", "bin": { "playwright-core": "cli.js" diff --git a/package.json b/package.json index 103d11de..42e4309f 100644 --- a/package.json +++ b/package.json @@ -24,14 +24,14 @@ "url": "git+https://github.com/joomcode/e2ed.git" }, "dependencies": { - "@playwright/test": "1.48.1", + "@playwright/test": "1.48.2", "create-locator": "0.0.25", "get-modules-graph": "0.0.9", "globby": "11.1.0" }, "devDependencies": { - "@playwright/browser-chromium": "1.48.1", - "@types/node": "22.7.6", + "@playwright/browser-chromium": "1.48.2", + "@types/node": "22.8.7", "@typescript-eslint/eslint-plugin": "7.18.0", "@typescript-eslint/parser": "7.18.0", "assert-modules-support-case-insensitive-fs": "1.0.1",