From 1c66c547b165e1eb2910468caa38fe68fda1bd1c Mon Sep 17 00:00:00 2001 From: jrandolf <101637635+jrandolf@users.noreply.github.com> Date: Thu, 14 Sep 2023 11:57:06 +0200 Subject: [PATCH] chore: use `throwIfDisposed` in `ElementHandle` and `JSHandle` (#10904) --- .../puppeteer-core/src/api/ElementHandle.ts | 44 ++++++++++++++----- packages/puppeteer-core/src/api/JSHandle.ts | 4 +- .../src/common/bidi/ElementHandle.ts | 3 ++ 3 files changed, 40 insertions(+), 11 deletions(-) diff --git a/packages/puppeteer-core/src/api/ElementHandle.ts b/packages/puppeteer-core/src/api/ElementHandle.ts index f7d5479a3bb..bb586fafbab 100644 --- a/packages/puppeteer-core/src/api/ElementHandle.ts +++ b/packages/puppeteer-core/src/api/ElementHandle.ts @@ -222,16 +222,7 @@ export abstract class ElementHandle< /** * @internal */ - override async getProperty( - propertyName: HandleOr - ): Promise>; - /** - * @internal - */ - override async getProperty(propertyName: string): Promise>; - /** - * @internal - */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle override async getProperty( propertyName: HandleOr @@ -242,6 +233,7 @@ export abstract class ElementHandle< /** * @internal */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle override async getProperties(): Promise> { return await this.handle.getProperties(); @@ -260,6 +252,10 @@ export abstract class ElementHandle< pageFunction: Func | string, ...args: Params ): Promise>> { + pageFunction = withSourcePuppeteerURLIfNone( + this.evaluate.name, + pageFunction + ); return await this.handle.evaluate(pageFunction, ...args); } @@ -276,12 +272,17 @@ export abstract class ElementHandle< pageFunction: Func | string, ...args: Params ): Promise>>> { + pageFunction = withSourcePuppeteerURLIfNone( + this.evaluateHandle.name, + pageFunction + ); return await this.handle.evaluateHandle(pageFunction, ...args); } /** * @internal */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle override async jsonValue(): Promise { return await this.handle.jsonValue(); @@ -327,6 +328,7 @@ export abstract class ElementHandle< * @returns A {@link ElementHandle | element handle} to the first element * matching the given selector. Otherwise, `null`. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async $( selector: Selector @@ -346,6 +348,7 @@ export abstract class ElementHandle< * @returns An array of {@link ElementHandle | element handles} that point to * elements matching the given selector. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async $$( selector: Selector @@ -479,6 +482,7 @@ export abstract class ElementHandle< * If there are no such elements, the method will resolve to an empty array. * @param expression - Expression to {@link https://developer.mozilla.org/en-US/docs/Web/API/Document/evaluate | evaluate} */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async $x(expression: string): Promise>> { if (expression.startsWith('//')) { @@ -524,6 +528,7 @@ export abstract class ElementHandle< * @returns An element matching the given selector. * @throws Throws if an element matching the given selector doesn't appear. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async waitForSelector( selector: Selector, @@ -554,6 +559,7 @@ export abstract class ElementHandle< * Checks if an element is visible using the same mechanism as * {@link ElementHandle.waitForSelector}. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async isVisible(): Promise { return await this.#checkVisibility(true); @@ -563,6 +569,7 @@ export abstract class ElementHandle< * Checks if an element is hidden using the same mechanism as * {@link ElementHandle.waitForSelector}. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async isHidden(): Promise { return await this.#checkVisibility(false); @@ -630,6 +637,7 @@ export abstract class ElementHandle< * default value can be changed by using the {@link Page.setDefaultTimeout} * method. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async waitForXPath( xpath: string, @@ -663,6 +671,7 @@ export abstract class ElementHandle< * @throws An error if the handle does not match. **The handle will not be * automatically disposed.** */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async toElement< K extends keyof HTMLElementTagNameMap | keyof SVGElementTagNameMap, @@ -686,6 +695,7 @@ export abstract class ElementHandle< /** * Returns the middle point within an element unless a specific offset is provided. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async clickablePoint(offset?: Offset): Promise { const box = await this.#clickableBox(); @@ -709,6 +719,7 @@ export abstract class ElementHandle< * uses {@link Page} to hover over the center of the element. * If the element is detached from DOM, the method throws an error. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async hover(this: ElementHandle): Promise { await this.scrollIntoViewIfNeeded(); @@ -721,6 +732,7 @@ export abstract class ElementHandle< * uses {@link Page | Page.mouse} to click in the center of the element. * If the element is detached from DOM, the method throws an error. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async click( this: ElementHandle, @@ -878,6 +890,7 @@ export abstract class ElementHandle< * `multiple` attribute, all values are considered, otherwise only the first * one is taken into account. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async select(...values: string[]): Promise { for (const value of values) { @@ -947,6 +960,7 @@ export abstract class ElementHandle< * {@link Touchscreen.tap} to tap in the center of the element. * If the element is detached from DOM, the method throws an error. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async tap(this: ElementHandle): Promise { await this.scrollIntoViewIfNeeded(); @@ -955,6 +969,7 @@ export abstract class ElementHandle< await this.frame.page().touchscreen.touchEnd(); } + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async touchStart(this: ElementHandle): Promise { await this.scrollIntoViewIfNeeded(); @@ -962,6 +977,7 @@ export abstract class ElementHandle< await this.frame.page().touchscreen.touchStart(x, y); } + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async touchMove(this: ElementHandle): Promise { await this.scrollIntoViewIfNeeded(); @@ -969,6 +985,7 @@ export abstract class ElementHandle< await this.frame.page().touchscreen.touchMove(x, y); } + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async touchEnd(this: ElementHandle): Promise { await this.scrollIntoViewIfNeeded(); @@ -978,6 +995,7 @@ export abstract class ElementHandle< /** * Calls {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/focus | focus} on the element. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async focus(): Promise { await this.evaluate(element => { @@ -1013,6 +1031,7 @@ export abstract class ElementHandle< * * @param options - Delay in milliseconds. Defaults to 0. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async type( text: string, @@ -1036,6 +1055,7 @@ export abstract class ElementHandle< * @param key - Name of key to press, such as `ArrowLeft`. * See {@link KeyInput} for a list of all key names. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async press( key: KeyInput, @@ -1125,6 +1145,7 @@ export abstract class ElementHandle< * This method returns the bounding box of the element (relative to the main frame), * or `null` if the element is not visible. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async boundingBox(): Promise { const box = await this.evaluate(element => { @@ -1161,6 +1182,7 @@ export abstract class ElementHandle< * Boxes are represented as an array of points; * Each Point is an object `{x, y}`. Box points are sorted clock-wise. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async boxModel(): Promise { const model = await this.evaluate(element => { @@ -1349,6 +1371,7 @@ export abstract class ElementHandle< * @param options - Threshold for the intersection between 0 (no intersection) and 1 * (full intersection). Defaults to 1. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async isIntersectingViewport( this: ElementHandle, @@ -1379,6 +1402,7 @@ export abstract class ElementHandle< * Scrolls the element into view using either the automation protocol client * or by calling element.scrollIntoView. */ + @throwIfDisposed() @ElementHandle.bindIsolatedHandle async scrollIntoView(this: ElementHandle): Promise { await this.assertConnectedElement(); diff --git a/packages/puppeteer-core/src/api/JSHandle.ts b/packages/puppeteer-core/src/api/JSHandle.ts index 843d42e5388..cd5ba9ee4bb 100644 --- a/packages/puppeteer-core/src/api/JSHandle.ts +++ b/packages/puppeteer-core/src/api/JSHandle.ts @@ -24,7 +24,7 @@ import { Moveable, } from '../common/types.js'; import {debugError, withSourcePuppeteerURLIfNone} from '../common/util.js'; -import {moveable} from '../util/decorators.js'; +import {moveable, throwIfDisposed} from '../util/decorators.js'; import {ElementHandle} from './ElementHandle.js'; import {Realm} from './Realm.js'; @@ -124,6 +124,7 @@ export abstract class JSHandle /** * @internal */ + @throwIfDisposed() async getProperty( propertyName: HandleOr ): Promise> { @@ -150,6 +151,7 @@ export abstract class JSHandle * children; // holds elementHandles to all children of document.body * ``` */ + @throwIfDisposed() async getProperties(): Promise> { const propertyNames = await this.evaluate(object => { const enumerableProperties = []; diff --git a/packages/puppeteer-core/src/common/bidi/ElementHandle.ts b/packages/puppeteer-core/src/common/bidi/ElementHandle.ts index ac2627b311f..af2ed44ce76 100644 --- a/packages/puppeteer-core/src/common/bidi/ElementHandle.ts +++ b/packages/puppeteer-core/src/common/bidi/ElementHandle.ts @@ -17,6 +17,7 @@ import * as Bidi from 'chromium-bidi/lib/cjs/protocol/protocol.js'; import {AutofillData, ElementHandle} from '../../api/ElementHandle.js'; +import {throwIfDisposed} from '../../util/decorators.js'; import {BidiFrame} from './Frame.js'; import {BidiJSHandle} from './JSHandle.js'; @@ -55,6 +56,7 @@ export class BidiElementHandle< return this.handle.remoteValue(); } + @throwIfDisposed() override async autofill(data: AutofillData): Promise { const client = this.frame.client; const nodeInfo = await client.send('DOM.describeNode', { @@ -72,6 +74,7 @@ export class BidiElementHandle< override async contentFrame( this: BidiElementHandle ): Promise; + @throwIfDisposed() @ElementHandle.bindIsolatedHandle override async contentFrame(): Promise { using handle = (await this.evaluateHandle(element => {