From 256223a7b18672ae3890bde312986482ea0fed52 Mon Sep 17 00:00:00 2001 From: Alex Rudenko Date: Wed, 25 May 2022 15:34:11 +0200 Subject: [PATCH] chore: strict-mode TS for DOMWorld (#8398) --- .eslintrc.js | 7 +- src/common/DOMWorld.ts | 165 ++++++++++++++++++++++++++++------------- src/common/helper.ts | 2 +- 3 files changed, 121 insertions(+), 53 deletions(-) diff --git a/.eslintrc.js b/.eslintrc.js index 2c53f10b..ac342c26 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -126,7 +126,10 @@ module.exports = { ], rules: { 'no-unused-vars': 0, - '@typescript-eslint/no-unused-vars': 2, + '@typescript-eslint/no-unused-vars': [ + 'error', + { argsIgnorePattern: '^_' }, + ], 'func-call-spacing': 0, '@typescript-eslint/func-call-spacing': 2, semi: 0, @@ -140,6 +143,8 @@ module.exports = { '@typescript-eslint/explicit-function-return-type': 0, // We know it's bad and use it very sparingly but it's needed :( '@typescript-eslint/ban-ts-ignore': 0, + // We allow non-null assertions if the value was asserted using `assert` API. + '@typescript-eslint/no-non-null-assertion': 0, /** * This is the default options (as per * https://github.com/typescript-eslint/typescript-eslint/blob/HEAD/packages/eslint-plugin/docs/rules/ban-types.md), diff --git a/src/common/DOMWorld.ts b/src/common/DOMWorld.ts index eb56d833..f647524a 100644 --- a/src/common/DOMWorld.ts +++ b/src/common/DOMWorld.ts @@ -46,7 +46,7 @@ declare const predicateQueryHandler: ( selector: string ) => Promise>; declare const checkWaitForOptions: ( - node: Node, + node: Node | null, waitForVisible: boolean, waitForHidden: boolean ) => Element | null | boolean; @@ -77,10 +77,11 @@ export class DOMWorld { private _client: CDPSession; private _frame: Frame; private _timeoutSettings: TimeoutSettings; - private _documentPromise?: Promise = null; - private _contextPromise?: Promise = null; + private _documentPromise: Promise | null = null; + private _contextPromise: Promise | null = null; - private _contextResolveCallback?: (x?: ExecutionContext) => void = null; + private _contextResolveCallback: ((x: ExecutionContext) => void) | null = + null; private _detached = false; /** @@ -119,14 +120,14 @@ export class DOMWorld { return this._frame; } - async _setContext(context?: ExecutionContext): Promise { + async _setContext(context: ExecutionContext | null): Promise { if (context) { assert( this._contextResolveCallback, 'Execution Context has already been set.' ); this._ctxBindings.clear(); - this._contextResolveCallback.call(null, context); + this._contextResolveCallback?.call(null, context); this._contextResolveCallback = null; for (const waitTask of this._waitTasks) waitTask.rerun(); } else { @@ -155,6 +156,8 @@ export class DOMWorld { throw new Error( `Execution context is not available in detached frame "${this._frame.url()}" (are you trying to evaluate?)` ); + if (this._contextPromise === null) + throw new Error(`Execution content promise is missing`); return this._contextPromise; } @@ -189,7 +192,11 @@ export class DOMWorld { if (this._documentPromise) return this._documentPromise; this._documentPromise = this.executionContext().then(async (context) => { const document = await context.evaluateHandle('document'); - return document.asElement(); + const element = document.asElement(); + if (element === null) { + throw new Error('Document is null'); + } + return element; }); return this._documentPromise; } @@ -306,9 +313,17 @@ export class DOMWorld { if (url !== null) { try { const context = await this.executionContext(); - return ( - await context.evaluateHandle(addScriptUrl, url, id, type) - ).asElement(); + const handle = await context.evaluateHandle( + addScriptUrl, + url, + id, + type + ); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Script element is not found'); + } + return elementHandle; } catch (error) { throw new Error(`Loading script from ${url} failed`); } @@ -324,16 +339,32 @@ export class DOMWorld { let contents = await fs.promises.readFile(path, 'utf8'); contents += '//# sourceURL=' + path.replace(/\n/g, ''); const context = await this.executionContext(); - return ( - await context.evaluateHandle(addScriptContent, contents, id, type) - ).asElement(); + const handle = await context.evaluateHandle( + addScriptContent, + contents, + id, + type + ); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Script element is not found'); + } + return elementHandle; } if (content !== null) { const context = await this.executionContext(); - return ( - await context.evaluateHandle(addScriptContent, content, id, type) - ).asElement(); + const handle = await context.evaluateHandle( + addScriptContent, + content, + id, + type + ); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Script element is not found'); + } + return elementHandle; } throw new Error( @@ -394,7 +425,12 @@ export class DOMWorld { if (url !== null) { try { const context = await this.executionContext(); - return (await context.evaluateHandle(addStyleUrl, url)).asElement(); + const handle = await context.evaluateHandle(addStyleUrl, url); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Style element is not found'); + } + return elementHandle; } catch (error) { throw new Error(`Loading style from ${url} failed`); } @@ -410,16 +446,22 @@ export class DOMWorld { let contents = await fs.promises.readFile(path, 'utf8'); contents += '/*# sourceURL=' + path.replace(/\n/g, '') + '*/'; const context = await this.executionContext(); - return ( - await context.evaluateHandle(addStyleContent, contents) - ).asElement(); + const handle = await context.evaluateHandle(addStyleContent, contents); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Style element is not found'); + } + return elementHandle; } if (content !== null) { const context = await this.executionContext(); - return ( - await context.evaluateHandle(addStyleContent, content) - ).asElement(); + const handle = await context.evaluateHandle(addStyleContent, content); + const elementHandle = handle.asElement(); + if (elementHandle === null) { + throw new Error('Style element is not found'); + } + return elementHandle; } throw new Error( @@ -459,36 +501,37 @@ export class DOMWorld { ): Promise { const handle = await this.$(selector); assert(handle, 'No node found for selector: ' + selector); - await handle.click(options); - await handle.dispose(); + await handle!.click(options); + await handle!.dispose(); } async focus(selector: string): Promise { const handle = await this.$(selector); assert(handle, 'No node found for selector: ' + selector); - await handle.focus(); - await handle.dispose(); + await handle!.focus(); + await handle!.dispose(); } async hover(selector: string): Promise { const handle = await this.$(selector); assert(handle, 'No node found for selector: ' + selector); - await handle.hover(); - await handle.dispose(); + await handle!.hover(); + await handle!.dispose(); } async select(selector: string, ...values: string[]): Promise { const handle = await this.$(selector); assert(handle, 'No node found for selector: ' + selector); - const result = await handle.select(...values); - await handle.dispose(); + const result = await handle!.select(...values); + await handle!.dispose(); return result; } async tap(selector: string): Promise { const handle = await this.$(selector); - await handle.tap(); - await handle.dispose(); + assert(handle, 'No node found for selector: ' + selector); + await handle!.tap(); + await handle!.dispose(); } async type( @@ -498,8 +541,8 @@ export class DOMWorld { ): Promise { const handle = await this.$(selector); assert(handle, 'No node found for selector: ' + selector); - await handle.type(text, options); - await handle.dispose(); + await handle!.type(text, options); + await handle!.dispose(); } async waitForSelector( @@ -508,6 +551,9 @@ export class DOMWorld { ): Promise { const { updatedSelector, queryHandler } = getQueryHandlerAndSelector(selector); + if (!queryHandler.waitFor) { + throw new Error('Query handler does not support waitFor'); + } return queryHandler.waitFor(this, updatedSelector, options); } @@ -550,10 +596,10 @@ export class DOMWorld { // We could have tried to evaluate in a context which was already // destroyed. This happens, for example, if the page is navigated while // we are trying to add the binding - const ctxDestroyed = error.message.includes( + const ctxDestroyed = (error as Error).message.includes( 'Execution context was destroyed' ); - const ctxNotFound = error.message.includes( + const ctxNotFound = (error as Error).message.includes( 'Cannot find context with specified id' ); if (ctxDestroyed || ctxNotFound) { @@ -596,7 +642,11 @@ export class DOMWorld { return; if (context._contextId !== event.executionContextId) return; try { - const result = await this._boundFunctions.get(name)(...args); + const fn = this._boundFunctions.get(name); + if (!fn) { + throw new Error(`Bound function $name is not found`); + } + const result = await fn(...args); await context.evaluate(deliverResult, name, seq, result); } catch (error) { // The WaitTask may already have been resolved by timing out, or the @@ -604,11 +654,15 @@ export class DOMWorld { // In both caes, the promises above are rejected with a protocol error. // We can safely ignores these, as the WaitTask is re-installed in // the next execution context if needed. - if (error.message.includes('Protocol error')) return; + if ((error as Error).message.includes('Protocol error')) return; debugError(error); } function deliverResult(name: string, seq: number, result: unknown): void { + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-ignore Code is evaluated in a different context. globalThis[name].callbacks.get(seq).resolve(result); + // eslint-disable-next-line @typescript-eslint/ban-ts-comment + // @ts-ignore Code is evaluated in a different context. globalThis[name].callbacks.delete(seq); } } @@ -749,6 +803,8 @@ export interface WaitTaskOptions { root?: ElementHandle; } +const noop = (): void => {}; + /** * @internal */ @@ -759,14 +815,14 @@ export class WaitTask { _predicateBody: string; _predicateAcceptsContextElement: boolean; _args: SerializableOrJSHandle[]; - _binding: PageBinding; + _binding?: PageBinding; _runCount = 0; promise: Promise; - _resolve: (x: JSHandle) => void; - _reject: (x: Error) => void; + _resolve: (x: JSHandle) => void = noop; + _reject: (x: Error) => void = noop; _timeoutTimer?: NodeJS.Timeout; _terminated = false; - _root: ElementHandle = null; + _root: ElementHandle | null = null; constructor(options: WaitTaskOptions) { if (helper.isString(options.polling)) @@ -789,7 +845,7 @@ export class WaitTask { this._domWorld = options.domWorld; this._polling = options.polling; this._timeout = options.timeout; - this._root = options.root; + this._root = options.root || null; this._predicateBody = getPredicateBody(options.predicateBody); this._predicateAcceptsContextElement = options.predicateAcceptsContextElement; @@ -829,8 +885,8 @@ export class WaitTask { async rerun(): Promise { const runCount = ++this._runCount; - let success: JSHandle = null; - let error: Error = null; + let success: JSHandle | null = null; + let error: Error | null = null; const context = await this._domWorld.executionContext(); if (this._terminated || runCount !== this._runCount) return; if (this._binding) { @@ -848,7 +904,7 @@ export class WaitTask { ...this._args ); } catch (error_) { - error = error_; + error = error_ as Error; } if (this._terminated || runCount !== this._runCount) { @@ -863,6 +919,8 @@ export class WaitTask { !error && (await this._domWorld.evaluate((s) => !s, success).catch(() => true)) ) { + if (!success) + throw new Error('Assertion: result handle is not available'); await success.dispose(); return; } @@ -895,13 +953,15 @@ export class WaitTask { this._reject(error); } else { + if (!success) + throw new Error('Assertion: result handle is not available'); this._resolve(success); } this._cleanup(); } _cleanup(): void { - clearTimeout(this._timeoutTimer); + this._timeoutTimer !== undefined && clearTimeout(this._timeoutTimer); this._domWorld._waitTasks.delete(this); } } @@ -931,7 +991,7 @@ async function waitForPredicatePageFunction( : await predicate(...args); if (success) return Promise.resolve(success); - let fulfill; + let fulfill = (_?: unknown) => {}; const result = new Promise((x) => (fulfill = x)); const observer = new MutationObserver(async () => { if (timedOut) { @@ -946,6 +1006,9 @@ async function waitForPredicatePageFunction( fulfill(success); } }); + if (!root) { + throw new Error('Root element is not found.'); + } observer.observe(root, { childList: true, subtree: true, @@ -955,7 +1018,7 @@ async function waitForPredicatePageFunction( } async function pollRaf(): Promise { - let fulfill; + let fulfill = (_?: unknown): void => {}; const result = new Promise((x) => (fulfill = x)); await onRaf(); return result; @@ -974,7 +1037,7 @@ async function waitForPredicatePageFunction( } async function pollInterval(pollInterval: number): Promise { - let fulfill; + let fulfill = (_?: unknown): void => {}; const result = new Promise((x) => (fulfill = x)); await onTimeout(); return result; diff --git a/src/common/helper.ts b/src/common/helper.ts index 215f182e..12d308e6 100644 --- a/src/common/helper.ts +++ b/src/common/helper.ts @@ -261,7 +261,7 @@ function makePredicateString( predicateQueryHandler?: Function ): string { function checkWaitForOptions( - node: Node, + node: Node | null, waitForVisible: boolean, waitForHidden: boolean ): Node | null | boolean {