refactor: migrate to iterator-based query handlers (#9676)
This commit is contained in:
parent
023c2dcdbc
commit
56f99f7b10
@ -144,6 +144,7 @@ sidebar_label: API
|
|||||||
| ------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
| ------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
||||||
| [ActionResult](./puppeteer.actionresult.md) | |
|
| [ActionResult](./puppeteer.actionresult.md) | |
|
||||||
| [Awaitable](./puppeteer.awaitable.md) | |
|
| [Awaitable](./puppeteer.awaitable.md) | |
|
||||||
|
| [AwaitableIterable](./puppeteer.awaitableiterable.md) | |
|
||||||
| [ChromeReleaseChannel](./puppeteer.chromereleasechannel.md) | |
|
| [ChromeReleaseChannel](./puppeteer.chromereleasechannel.md) | |
|
||||||
| [ConsoleMessageType](./puppeteer.consolemessagetype.md) | The supported types for console messages. |
|
| [ConsoleMessageType](./puppeteer.consolemessagetype.md) | The supported types for console messages. |
|
||||||
| [ElementFor](./puppeteer.elementfor.md) | |
|
| [ElementFor](./puppeteer.elementfor.md) | |
|
||||||
|
11
docs/api/puppeteer.awaitableiterable.md
Normal file
11
docs/api/puppeteer.awaitableiterable.md
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
---
|
||||||
|
sidebar_label: AwaitableIterable
|
||||||
|
---
|
||||||
|
|
||||||
|
# AwaitableIterable type
|
||||||
|
|
||||||
|
#### Signature:
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
export type AwaitableIterable<T> = Iterable<T> | AsyncIterable<T>;
|
||||||
|
```
|
@ -13,6 +13,6 @@ export interface CustomQueryHandler
|
|||||||
## Properties
|
## Properties
|
||||||
|
|
||||||
| Property | Modifiers | Type | Description | Default |
|
| Property | Modifiers | Type | Description | Default |
|
||||||
| ------------------------------------------------------- | --------- | ------------------------------------------------- | ----------------- | ------- |
|
| ------------------------------------------------------- | --------- | --------------------------------------------------------- | ----------------- | ------- |
|
||||||
| [queryAll?](./puppeteer.customqueryhandler.queryall.md) | | (node: Node, selector: string) => Node\[\] | <i>(Optional)</i> | |
|
| [queryAll?](./puppeteer.customqueryhandler.queryall.md) | | (node: Node, selector: string) => Iterable<Node> | <i>(Optional)</i> | |
|
||||||
| [queryOne?](./puppeteer.customqueryhandler.queryone.md) | | (node: Node, selector: string) => Node \| null | <i>(Optional)</i> | |
|
| [queryOne?](./puppeteer.customqueryhandler.queryone.md) | | (node: Node, selector: string) => Node \| null | <i>(Optional)</i> | |
|
||||||
|
@ -8,6 +8,6 @@ sidebar_label: CustomQueryHandler.queryAll
|
|||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
interface CustomQueryHandler {
|
interface CustomQueryHandler {
|
||||||
queryAll?: (node: Node, selector: string) => Node[];
|
queryAll?: (node: Node, selector: string) => Iterable<Node>;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
@ -16,46 +16,43 @@
|
|||||||
|
|
||||||
import {Protocol} from 'devtools-protocol';
|
import {Protocol} from 'devtools-protocol';
|
||||||
|
|
||||||
|
import {ElementHandle} from '../api/ElementHandle.js';
|
||||||
import {assert} from '../util/assert.js';
|
import {assert} from '../util/assert.js';
|
||||||
import {CDPSession} from './Connection.js';
|
import {CDPSession} from './Connection.js';
|
||||||
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
|
||||||
|
|
||||||
import type {ElementHandle} from '../api/ElementHandle.js';
|
|
||||||
import type {PuppeteerQueryHandler} from './QueryHandler.js';
|
|
||||||
import type {Frame} from './Frame.js';
|
import type {Frame} from './Frame.js';
|
||||||
|
import type {WaitForSelectorOptions} from './IsolatedWorld.js';
|
||||||
|
import {IterableUtil} from './IterableUtil.js';
|
||||||
|
import {QueryHandler, QuerySelector} from './QueryHandler.js';
|
||||||
|
import {AwaitableIterable} from './types.js';
|
||||||
|
|
||||||
async function queryAXTree(
|
const queryAXTree = async (
|
||||||
client: CDPSession,
|
client: CDPSession,
|
||||||
element: ElementHandle<Node>,
|
element: ElementHandle<Node>,
|
||||||
accessibleName?: string,
|
accessibleName?: string,
|
||||||
role?: string
|
role?: string
|
||||||
): Promise<Protocol.Accessibility.AXNode[]> {
|
): Promise<Protocol.Accessibility.AXNode[]> => {
|
||||||
const {nodes} = await client.send('Accessibility.queryAXTree', {
|
const {nodes} = await client.send('Accessibility.queryAXTree', {
|
||||||
objectId: element.remoteObject().objectId,
|
objectId: element.remoteObject().objectId,
|
||||||
accessibleName,
|
accessibleName,
|
||||||
role,
|
role,
|
||||||
});
|
});
|
||||||
const filteredNodes: Protocol.Accessibility.AXNode[] = nodes.filter(
|
return nodes.filter((node: Protocol.Accessibility.AXNode) => {
|
||||||
(node: Protocol.Accessibility.AXNode) => {
|
|
||||||
return !node.role || node.role.value !== 'StaticText';
|
return !node.role || node.role.value !== 'StaticText';
|
||||||
}
|
});
|
||||||
);
|
};
|
||||||
return filteredNodes;
|
|
||||||
}
|
type ARIASelector = {name?: string; role?: string};
|
||||||
|
|
||||||
|
const KNOWN_ATTRIBUTES = Object.freeze(['name', 'role']);
|
||||||
|
const isKnownAttribute = (
|
||||||
|
attribute: string
|
||||||
|
): attribute is keyof ARIASelector => {
|
||||||
|
return KNOWN_ATTRIBUTES.includes(attribute);
|
||||||
|
};
|
||||||
|
|
||||||
const normalizeValue = (value: string): string => {
|
const normalizeValue = (value: string): string => {
|
||||||
return value.replace(/ +/g, ' ').trim();
|
return value.replace(/ +/g, ' ').trim();
|
||||||
};
|
};
|
||||||
const knownAttributes = new Set(['name', 'role']);
|
|
||||||
const attributeRegexp =
|
|
||||||
/\[\s*(?<attribute>\w+)\s*=\s*(?<quote>"|')(?<value>\\.|.*?(?=\k<quote>))\k<quote>\s*\]/g;
|
|
||||||
|
|
||||||
type ARIAQueryOption = {name?: string; role?: string};
|
|
||||||
function isKnownAttribute(
|
|
||||||
attribute: string
|
|
||||||
): attribute is keyof ARIAQueryOption {
|
|
||||||
return knownAttributes.has(attribute);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The selectors consist of an accessible name to query for and optionally
|
* The selectors consist of an accessible name to query for and optionally
|
||||||
@ -68,11 +65,13 @@ function isKnownAttribute(
|
|||||||
* - 'label' queries for elements with name 'label' and any role.
|
* - 'label' queries for elements with name 'label' and any role.
|
||||||
* - '[name=""][role="button"]' queries for elements with no name and role 'button'.
|
* - '[name=""][role="button"]' queries for elements with no name and role 'button'.
|
||||||
*/
|
*/
|
||||||
function parseAriaSelector(selector: string): ARIAQueryOption {
|
const ATTRIBUTE_REGEXP =
|
||||||
const queryOptions: ARIAQueryOption = {};
|
/\[\s*(?<attribute>\w+)\s*=\s*(?<quote>"|')(?<value>\\.|.*?(?=\k<quote>))\k<quote>\s*\]/g;
|
||||||
|
const parseARIASelector = (selector: string): ARIASelector => {
|
||||||
|
const queryOptions: ARIASelector = {};
|
||||||
const defaultName = selector.replace(
|
const defaultName = selector.replace(
|
||||||
attributeRegexp,
|
ATTRIBUTE_REGEXP,
|
||||||
(_, attribute: string, _quote: string, value: string) => {
|
(_, attribute, __, value) => {
|
||||||
attribute = attribute.trim();
|
attribute = attribute.trim();
|
||||||
assert(
|
assert(
|
||||||
isKnownAttribute(attribute),
|
isKnownAttribute(attribute),
|
||||||
@ -86,104 +85,56 @@ function parseAriaSelector(selector: string): ARIAQueryOption {
|
|||||||
queryOptions.name = normalizeValue(defaultName);
|
queryOptions.name = normalizeValue(defaultName);
|
||||||
}
|
}
|
||||||
return queryOptions;
|
return queryOptions;
|
||||||
}
|
|
||||||
|
|
||||||
const queryOneId = async (element: ElementHandle<Node>, selector: string) => {
|
|
||||||
const {name, role} = parseAriaSelector(selector);
|
|
||||||
const res = await queryAXTree(element.client, element, name, role);
|
|
||||||
if (!res[0] || !res[0].backendDOMNodeId) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return res[0].backendDOMNodeId;
|
|
||||||
};
|
|
||||||
|
|
||||||
const queryOne: PuppeteerQueryHandler['queryOne'] = async (
|
|
||||||
element,
|
|
||||||
selector
|
|
||||||
) => {
|
|
||||||
const id = await queryOneId(element, selector);
|
|
||||||
if (!id) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return (await element.frame.worlds[MAIN_WORLD].adoptBackendNode(
|
|
||||||
id
|
|
||||||
)) as ElementHandle<Node>;
|
|
||||||
};
|
|
||||||
|
|
||||||
const waitFor: PuppeteerQueryHandler['waitFor'] = async (
|
|
||||||
elementOrFrame,
|
|
||||||
selector,
|
|
||||||
options
|
|
||||||
) => {
|
|
||||||
let frame: Frame;
|
|
||||||
let element: ElementHandle<Node> | undefined;
|
|
||||||
if ('isOOPFrame' in elementOrFrame) {
|
|
||||||
frame = elementOrFrame;
|
|
||||||
} else {
|
|
||||||
frame = elementOrFrame.frame;
|
|
||||||
element = await frame.worlds[PUPPETEER_WORLD].adoptHandle(elementOrFrame);
|
|
||||||
}
|
|
||||||
|
|
||||||
const ariaQuerySelector = async (selector: string) => {
|
|
||||||
const id = await queryOneId(
|
|
||||||
element || (await frame.worlds[PUPPETEER_WORLD].document()),
|
|
||||||
selector
|
|
||||||
);
|
|
||||||
if (!id) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return (await frame.worlds[PUPPETEER_WORLD].adoptBackendNode(
|
|
||||||
id
|
|
||||||
)) as ElementHandle<Node>;
|
|
||||||
};
|
|
||||||
|
|
||||||
const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage(
|
|
||||||
(_: Element, selector: string) => {
|
|
||||||
return (
|
|
||||||
globalThis as unknown as {
|
|
||||||
ariaQuerySelector(selector: string): Node | null;
|
|
||||||
}
|
|
||||||
).ariaQuerySelector(selector);
|
|
||||||
},
|
|
||||||
element,
|
|
||||||
selector,
|
|
||||||
options,
|
|
||||||
new Map([['ariaQuerySelector', ariaQuerySelector]])
|
|
||||||
);
|
|
||||||
if (element) {
|
|
||||||
await element.dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
const handle = result?.asElement();
|
|
||||||
if (!handle) {
|
|
||||||
await result?.dispose();
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return handle.frame.worlds[MAIN_WORLD].transferHandle(handle);
|
|
||||||
};
|
|
||||||
|
|
||||||
const queryAll: PuppeteerQueryHandler['queryAll'] = async (
|
|
||||||
element,
|
|
||||||
selector
|
|
||||||
) => {
|
|
||||||
const exeCtx = element.executionContext();
|
|
||||||
const {name, role} = parseAriaSelector(selector);
|
|
||||||
const res = await queryAXTree(exeCtx._client, element, name, role);
|
|
||||||
const world = exeCtx._world!;
|
|
||||||
return Promise.all(
|
|
||||||
res.map(axNode => {
|
|
||||||
return world.adoptBackendNode(axNode.backendDOMNodeId) as Promise<
|
|
||||||
ElementHandle<Node>
|
|
||||||
>;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
export const ariaHandler: PuppeteerQueryHandler = {
|
export interface ARIAQuerySelectorContext {
|
||||||
queryOne,
|
__ariaQuerySelector(node: Node, selector: string): Promise<Node | null>;
|
||||||
waitFor,
|
}
|
||||||
queryAll,
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class ARIAQueryHandler extends QueryHandler {
|
||||||
|
static override querySelector: QuerySelector = async (node, selector) => {
|
||||||
|
const context = globalThis as unknown as ARIAQuerySelectorContext;
|
||||||
|
return context.__ariaQuerySelector(node, selector);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static override async *queryAll(
|
||||||
|
element: ElementHandle<Node>,
|
||||||
|
selector: string
|
||||||
|
): AwaitableIterable<ElementHandle<Node>> {
|
||||||
|
const context = element.executionContext();
|
||||||
|
const {name, role} = parseARIASelector(selector);
|
||||||
|
const results = await queryAXTree(context._client, element, name, role);
|
||||||
|
const world = context._world!;
|
||||||
|
yield* IterableUtil.map(results, node => {
|
||||||
|
return world.adoptBackendNode(node.backendDOMNodeId) as Promise<
|
||||||
|
ElementHandle<Node>
|
||||||
|
>;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
static override queryOne = async (
|
||||||
|
element: ElementHandle<Node>,
|
||||||
|
selector: string
|
||||||
|
): Promise<ElementHandle<Node> | null> => {
|
||||||
|
return (await IterableUtil.first(this.queryAll(element, selector))) ?? null;
|
||||||
|
};
|
||||||
|
|
||||||
|
static override async waitFor(
|
||||||
|
elementOrFrame: ElementHandle<Node> | Frame,
|
||||||
|
selector: string,
|
||||||
|
options: WaitForSelectorOptions
|
||||||
|
): Promise<ElementHandle<Node> | null> {
|
||||||
|
return super.waitFor(
|
||||||
|
elementOrFrame,
|
||||||
|
selector,
|
||||||
|
options,
|
||||||
|
new Map([['__ariaQuerySelector', this.queryOne]])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
29
packages/puppeteer-core/src/common/CSSQueryHandler.ts
Normal file
29
packages/puppeteer-core/src/common/CSSQueryHandler.ts
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {QueryHandler, QuerySelector, QuerySelectorAll} from './QueryHandler.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class CSSQueryHandler extends QueryHandler {
|
||||||
|
static override querySelector: QuerySelector = (element, selector) => {
|
||||||
|
return (element as Element).querySelector(selector);
|
||||||
|
};
|
||||||
|
static override querySelectorAll: QuerySelectorAll = (element, selector) => {
|
||||||
|
return (element as Element).querySelectorAll(selector);
|
||||||
|
};
|
||||||
|
}
|
95
packages/puppeteer-core/src/common/CustomQueryHandler.ts
Normal file
95
packages/puppeteer-core/src/common/CustomQueryHandler.ts
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {QueryHandler} from './QueryHandler.js';
|
||||||
|
import {getQueryHandlerByName} from './GetQueryHandler.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export const customQueryHandlers = new Map<string, typeof QueryHandler>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export interface CustomQueryHandler {
|
||||||
|
/**
|
||||||
|
* @returns A {@link Node} matching the given `selector` from {@link node}.
|
||||||
|
*/
|
||||||
|
queryOne?: (node: Node, selector: string) => Node | null;
|
||||||
|
/**
|
||||||
|
* @returns Some {@link Node}s matching the given `selector` from {@link node}.
|
||||||
|
*/
|
||||||
|
queryAll?: (node: Node, selector: string) => Iterable<Node>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Import {@link Puppeteer} and use the static method
|
||||||
|
* {@link Puppeteer.registerCustomQueryHandler}
|
||||||
|
*
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export function registerCustomQueryHandler(
|
||||||
|
name: string,
|
||||||
|
handler: CustomQueryHandler
|
||||||
|
): void {
|
||||||
|
if (getQueryHandlerByName(name)) {
|
||||||
|
throw new Error(`A query handler named "${name}" already exists`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const isValidName = /^[a-zA-Z]+$/.test(name);
|
||||||
|
if (!isValidName) {
|
||||||
|
throw new Error(`Custom query handler names may only contain [a-zA-Z]`);
|
||||||
|
}
|
||||||
|
|
||||||
|
customQueryHandlers.set(
|
||||||
|
name,
|
||||||
|
class extends QueryHandler {
|
||||||
|
static override querySelector = handler.queryOne;
|
||||||
|
static override querySelectorAll = handler.queryAll;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Import {@link Puppeteer} and use the static method
|
||||||
|
* {@link Puppeteer.unregisterCustomQueryHandler}
|
||||||
|
*
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export function unregisterCustomQueryHandler(name: string): void {
|
||||||
|
customQueryHandlers.delete(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Import {@link Puppeteer} and use the static method
|
||||||
|
* {@link Puppeteer.customQueryHandlerNames}
|
||||||
|
*
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export function customQueryHandlerNames(): string[] {
|
||||||
|
return [...customQueryHandlers.keys()];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Import {@link Puppeteer} and use the static method
|
||||||
|
* {@link Puppeteer.clearCustomQueryHandlers}
|
||||||
|
*
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export function clearCustomQueryHandlers(): void {
|
||||||
|
customQueryHandlers.clear();
|
||||||
|
}
|
@ -31,9 +31,10 @@ import {CDPSession} from './Connection.js';
|
|||||||
import {ExecutionContext} from './ExecutionContext.js';
|
import {ExecutionContext} from './ExecutionContext.js';
|
||||||
import {Frame} from './Frame.js';
|
import {Frame} from './Frame.js';
|
||||||
import {FrameManager} from './FrameManager.js';
|
import {FrameManager} from './FrameManager.js';
|
||||||
|
import {getQueryHandlerAndSelector} from './GetQueryHandler.js';
|
||||||
import {WaitForSelectorOptions} from './IsolatedWorld.js';
|
import {WaitForSelectorOptions} from './IsolatedWorld.js';
|
||||||
|
import {IterableUtil} from './IterableUtil.js';
|
||||||
import {CDPPage} from './Page.js';
|
import {CDPPage} from './Page.js';
|
||||||
import {getQueryHandlerAndSelector} from './QueryHandler.js';
|
|
||||||
import {
|
import {
|
||||||
ElementFor,
|
ElementFor,
|
||||||
EvaluateFuncWith,
|
EvaluateFuncWith,
|
||||||
@ -183,10 +184,6 @@ export class CDPElementHandle<
|
|||||||
): Promise<CDPElementHandle<NodeFor<Selector>> | null> {
|
): Promise<CDPElementHandle<NodeFor<Selector>> | null> {
|
||||||
const {updatedSelector, queryHandler} =
|
const {updatedSelector, queryHandler} =
|
||||||
getQueryHandlerAndSelector(selector);
|
getQueryHandlerAndSelector(selector);
|
||||||
assert(
|
|
||||||
queryHandler.queryOne,
|
|
||||||
'Cannot handle queries for a single element with the given selector'
|
|
||||||
);
|
|
||||||
return (await queryHandler.queryOne(
|
return (await queryHandler.queryOne(
|
||||||
this,
|
this,
|
||||||
updatedSelector
|
updatedSelector
|
||||||
@ -198,13 +195,9 @@ export class CDPElementHandle<
|
|||||||
): Promise<Array<CDPElementHandle<NodeFor<Selector>>>> {
|
): Promise<Array<CDPElementHandle<NodeFor<Selector>>>> {
|
||||||
const {updatedSelector, queryHandler} =
|
const {updatedSelector, queryHandler} =
|
||||||
getQueryHandlerAndSelector(selector);
|
getQueryHandlerAndSelector(selector);
|
||||||
assert(
|
return IterableUtil.collect(
|
||||||
queryHandler.queryAll,
|
queryHandler.queryAll(this, updatedSelector)
|
||||||
'Cannot handle queries for a multiple element with the given selector'
|
) as Promise<Array<CDPElementHandle<NodeFor<Selector>>>>;
|
||||||
);
|
|
||||||
return (await queryHandler.queryAll(this, updatedSelector)) as Array<
|
|
||||||
CDPElementHandle<NodeFor<Selector>>
|
|
||||||
>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override async $eval<
|
override async $eval<
|
||||||
@ -242,23 +235,14 @@ export class CDPElementHandle<
|
|||||||
pageFunction: Func | string,
|
pageFunction: Func | string,
|
||||||
...args: Params
|
...args: Params
|
||||||
): Promise<Awaited<ReturnType<Func>>> {
|
): Promise<Awaited<ReturnType<Func>>> {
|
||||||
const {updatedSelector, queryHandler} =
|
const results = await this.$$(selector);
|
||||||
getQueryHandlerAndSelector(selector);
|
const elements = await this.evaluateHandle((_, ...elements) => {
|
||||||
assert(
|
|
||||||
queryHandler.queryAll,
|
|
||||||
'Cannot handle queries for a multiple element with the given selector'
|
|
||||||
);
|
|
||||||
const handles = (await queryHandler.queryAll(
|
|
||||||
this,
|
|
||||||
updatedSelector
|
|
||||||
)) as Array<HandleFor<NodeFor<Selector>>>;
|
|
||||||
const elements = (await this.evaluateHandle((_, ...elements) => {
|
|
||||||
return elements;
|
return elements;
|
||||||
}, ...handles)) as JSHandle<Array<NodeFor<Selector>>>;
|
}, ...results);
|
||||||
const [result] = await Promise.all([
|
const [result] = await Promise.all([
|
||||||
elements.evaluate(pageFunction, ...args),
|
elements.evaluate(pageFunction, ...args),
|
||||||
...handles.map(handle => {
|
...results.map(results => {
|
||||||
return handle.dispose();
|
return results.dispose();
|
||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
await elements.dispose();
|
await elements.dispose();
|
||||||
@ -280,7 +264,6 @@ export class CDPElementHandle<
|
|||||||
): Promise<CDPElementHandle<NodeFor<Selector>> | null> {
|
): Promise<CDPElementHandle<NodeFor<Selector>> | null> {
|
||||||
const {updatedSelector, queryHandler} =
|
const {updatedSelector, queryHandler} =
|
||||||
getQueryHandlerAndSelector(selector);
|
getQueryHandlerAndSelector(selector);
|
||||||
assert(queryHandler.waitFor, 'Query handler does not support waiting');
|
|
||||||
return (await queryHandler.waitFor(
|
return (await queryHandler.waitFor(
|
||||||
this,
|
this,
|
||||||
updatedSelector,
|
updatedSelector,
|
||||||
|
@ -17,11 +17,11 @@
|
|||||||
import {Protocol} from 'devtools-protocol';
|
import {Protocol} from 'devtools-protocol';
|
||||||
import {ElementHandle} from '../api/ElementHandle.js';
|
import {ElementHandle} from '../api/ElementHandle.js';
|
||||||
import {Page} from '../api/Page.js';
|
import {Page} from '../api/Page.js';
|
||||||
import {assert} from '../util/assert.js';
|
|
||||||
import {isErrorLike} from '../util/ErrorLike.js';
|
import {isErrorLike} from '../util/ErrorLike.js';
|
||||||
import {CDPSession} from './Connection.js';
|
import {CDPSession} from './Connection.js';
|
||||||
import {ExecutionContext} from './ExecutionContext.js';
|
import {ExecutionContext} from './ExecutionContext.js';
|
||||||
import {FrameManager} from './FrameManager.js';
|
import {FrameManager} from './FrameManager.js';
|
||||||
|
import {getQueryHandlerAndSelector} from './GetQueryHandler.js';
|
||||||
import {HTTPResponse} from './HTTPResponse.js';
|
import {HTTPResponse} from './HTTPResponse.js';
|
||||||
import {MouseButton} from './Input.js';
|
import {MouseButton} from './Input.js';
|
||||||
import {
|
import {
|
||||||
@ -32,7 +32,6 @@ import {
|
|||||||
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
||||||
import {LazyArg} from './LazyArg.js';
|
import {LazyArg} from './LazyArg.js';
|
||||||
import {LifecycleWatcher, PuppeteerLifeCycleEvent} from './LifecycleWatcher.js';
|
import {LifecycleWatcher, PuppeteerLifeCycleEvent} from './LifecycleWatcher.js';
|
||||||
import {getQueryHandlerAndSelector} from './QueryHandler.js';
|
|
||||||
import {EvaluateFunc, EvaluateFuncWith, HandleFor, NodeFor} from './types.js';
|
import {EvaluateFunc, EvaluateFuncWith, HandleFor, NodeFor} from './types.js';
|
||||||
import {importFS} from './util.js';
|
import {importFS} from './util.js';
|
||||||
|
|
||||||
@ -620,7 +619,6 @@ export class Frame {
|
|||||||
): Promise<ElementHandle<NodeFor<Selector>> | null> {
|
): Promise<ElementHandle<NodeFor<Selector>> | null> {
|
||||||
const {updatedSelector, queryHandler} =
|
const {updatedSelector, queryHandler} =
|
||||||
getQueryHandlerAndSelector(selector);
|
getQueryHandlerAndSelector(selector);
|
||||||
assert(queryHandler.waitFor, 'Query handler does not support waiting');
|
|
||||||
return (await queryHandler.waitFor(
|
return (await queryHandler.waitFor(
|
||||||
this,
|
this,
|
||||||
updatedSelector,
|
updatedSelector,
|
||||||
|
68
packages/puppeteer-core/src/common/GetQueryHandler.ts
Normal file
68
packages/puppeteer-core/src/common/GetQueryHandler.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {ARIAQueryHandler} from './AriaQueryHandler.js';
|
||||||
|
import {PierceQueryHandler} from './PierceQueryHandler.js';
|
||||||
|
import {XPathQueryHandler} from './XPathQueryHandler.js';
|
||||||
|
import {TextQueryHandler} from './TextQueryHandler.js';
|
||||||
|
import {CSSQueryHandler} from './CSSQueryHandler.js';
|
||||||
|
import {customQueryHandlers} from './CustomQueryHandler.js';
|
||||||
|
import type {QueryHandler} from './QueryHandler.js';
|
||||||
|
|
||||||
|
export const BUILTIN_QUERY_HANDLERS = Object.freeze({
|
||||||
|
aria: ARIAQueryHandler,
|
||||||
|
pierce: PierceQueryHandler,
|
||||||
|
xpath: XPathQueryHandler,
|
||||||
|
text: TextQueryHandler,
|
||||||
|
});
|
||||||
|
|
||||||
|
const QUERY_SEPARATORS = ['=', '/'];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export function getQueryHandlerByName(
|
||||||
|
name: string
|
||||||
|
): typeof QueryHandler | undefined {
|
||||||
|
if (name in BUILTIN_QUERY_HANDLERS) {
|
||||||
|
return BUILTIN_QUERY_HANDLERS[name as 'aria'];
|
||||||
|
}
|
||||||
|
return customQueryHandlers.get(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export function getQueryHandlerAndSelector(selector: string): {
|
||||||
|
updatedSelector: string;
|
||||||
|
queryHandler: typeof QueryHandler;
|
||||||
|
} {
|
||||||
|
for (const handlerMap of [
|
||||||
|
customQueryHandlers,
|
||||||
|
Object.entries(BUILTIN_QUERY_HANDLERS),
|
||||||
|
]) {
|
||||||
|
for (const [name, queryHandler] of handlerMap) {
|
||||||
|
for (const separator of QUERY_SEPARATORS) {
|
||||||
|
const prefix = `${name}${separator}`;
|
||||||
|
if (selector.startsWith(prefix)) {
|
||||||
|
selector = selector.slice(prefix.length);
|
||||||
|
return {updatedSelector: selector, queryHandler};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {updatedSelector: selector, queryHandler: CSSQueryHandler};
|
||||||
|
}
|
80
packages/puppeteer-core/src/common/HandleIterator.ts
Normal file
80
packages/puppeteer-core/src/common/HandleIterator.ts
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {JSHandle} from '../api/JSHandle.js';
|
||||||
|
import {AwaitableIterable, HandleFor} from './types.js';
|
||||||
|
|
||||||
|
const DEFAULT_BATCH_SIZE = 20;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This will transpose an iterator JSHandle into a fast, Puppeteer-side iterator
|
||||||
|
* of JSHandles.
|
||||||
|
*
|
||||||
|
* @param size - The number of elements to transpose. This should be something
|
||||||
|
* reasonable.
|
||||||
|
*/
|
||||||
|
async function* fastTransposeIteratorHandle<T>(
|
||||||
|
iterator: JSHandle<AwaitableIterator<T>>,
|
||||||
|
size = DEFAULT_BATCH_SIZE
|
||||||
|
) {
|
||||||
|
const array = await iterator.evaluateHandle(async (iterator, size) => {
|
||||||
|
const results = [];
|
||||||
|
while (results.length < size) {
|
||||||
|
const result = await iterator.next();
|
||||||
|
if (result.done) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
results.push(result.value);
|
||||||
|
}
|
||||||
|
return results;
|
||||||
|
}, size);
|
||||||
|
const properties = (await array.getProperties()) as Map<string, HandleFor<T>>;
|
||||||
|
await array.dispose();
|
||||||
|
yield* properties.values();
|
||||||
|
return properties.size === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This will transpose an iterator JSHandle in batches based on the default size
|
||||||
|
* of {@link fastTransposeIteratorHandle}.
|
||||||
|
*/
|
||||||
|
|
||||||
|
async function* transposeIteratorHandle<T>(
|
||||||
|
iterator: JSHandle<AwaitableIterator<T>>
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
while (!(yield* fastTransposeIteratorHandle(iterator))) {}
|
||||||
|
} finally {
|
||||||
|
await iterator.dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type AwaitableIterator<T> = Iterator<T> | AsyncIterator<T>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export async function* transposeIterableHandle<T>(
|
||||||
|
handle: JSHandle<AwaitableIterable<T>>
|
||||||
|
): AsyncIterableIterator<HandleFor<T>> {
|
||||||
|
yield* transposeIteratorHandle(
|
||||||
|
await handle.evaluateHandle(iterable => {
|
||||||
|
return (async function* () {
|
||||||
|
yield* iterable;
|
||||||
|
})();
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
@ -440,7 +440,7 @@ export class IsolatedWorld {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const node = (await PuppeteerUtil.createFunction(query)(
|
const node = (await PuppeteerUtil.createFunction(query)(
|
||||||
root || document,
|
root ?? document,
|
||||||
selector,
|
selector,
|
||||||
PuppeteerUtil
|
PuppeteerUtil
|
||||||
)) as Node | null;
|
)) as Node | null;
|
||||||
@ -533,9 +533,9 @@ export class IsolatedWorld {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async adoptHandle<T extends JSHandle<Node>>(handle: T): Promise<T> {
|
async adoptHandle<T extends JSHandle<Node>>(handle: T): Promise<T> {
|
||||||
const executionContext = await this.executionContext();
|
const context = await this.executionContext();
|
||||||
assert(
|
assert(
|
||||||
handle.executionContext() !== executionContext,
|
handle.executionContext() !== context,
|
||||||
'Cannot adopt handle that already belongs to this execution context'
|
'Cannot adopt handle that already belongs to this execution context'
|
||||||
);
|
);
|
||||||
const nodeInfo = await this.#client.send('DOM.describeNode', {
|
const nodeInfo = await this.#client.send('DOM.describeNode', {
|
||||||
@ -545,9 +545,18 @@ export class IsolatedWorld {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async transferHandle<T extends JSHandle<Node>>(handle: T): Promise<T> {
|
async transferHandle<T extends JSHandle<Node>>(handle: T): Promise<T> {
|
||||||
const result = await this.adoptHandle(handle);
|
const context = await this.executionContext();
|
||||||
|
if (handle.executionContext() === context) {
|
||||||
|
return handle;
|
||||||
|
}
|
||||||
|
const info = await this.#client.send('DOM.describeNode', {
|
||||||
|
objectId: handle.remoteObject().objectId,
|
||||||
|
});
|
||||||
|
const newHandle = (await this.adoptBackendNode(
|
||||||
|
info.node.backendNodeId
|
||||||
|
)) as T;
|
||||||
await handle.dispose();
|
await handle.dispose();
|
||||||
return result;
|
return newHandle;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
48
packages/puppeteer-core/src/common/IterableUtil.ts
Normal file
48
packages/puppeteer-core/src/common/IterableUtil.ts
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type {AwaitableIterable} from './types.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class IterableUtil {
|
||||||
|
static async *map<T, U>(
|
||||||
|
iterable: AwaitableIterable<T>,
|
||||||
|
map: (item: T) => Promise<U>
|
||||||
|
): AwaitableIterable<U> {
|
||||||
|
for await (const value of iterable) {
|
||||||
|
yield await map(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static async collect<T>(iterable: AwaitableIterable<T>): Promise<T[]> {
|
||||||
|
const result = [];
|
||||||
|
for await (const value of iterable) {
|
||||||
|
result.push(value);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static async first<T>(
|
||||||
|
iterable: AwaitableIterable<T>
|
||||||
|
): Promise<T | undefined> {
|
||||||
|
for await (const value of iterable) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
38
packages/puppeteer-core/src/common/PierceQueryHandler.ts
Normal file
38
packages/puppeteer-core/src/common/PierceQueryHandler.ts
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import type PuppeteerUtil from '../injected/injected.js';
|
||||||
|
import {QueryHandler} from './QueryHandler.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class PierceQueryHandler extends QueryHandler {
|
||||||
|
static override querySelector = (
|
||||||
|
element: Node,
|
||||||
|
selector: string,
|
||||||
|
{pierceQuerySelector}: PuppeteerUtil
|
||||||
|
): Node | null => {
|
||||||
|
return pierceQuerySelector(element, selector);
|
||||||
|
};
|
||||||
|
static override querySelectorAll = (
|
||||||
|
element: Node,
|
||||||
|
selector: string,
|
||||||
|
{pierceQuerySelectorAll}: PuppeteerUtil
|
||||||
|
): Iterable<Node> => {
|
||||||
|
return pierceQuerySelectorAll(element, selector);
|
||||||
|
};
|
||||||
|
}
|
@ -13,6 +13,7 @@
|
|||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Browser} from '../api/Browser.js';
|
import {Browser} from '../api/Browser.js';
|
||||||
import {
|
import {
|
||||||
BrowserConnectOptions,
|
BrowserConnectOptions,
|
||||||
@ -25,7 +26,7 @@ import {
|
|||||||
customQueryHandlerNames,
|
customQueryHandlerNames,
|
||||||
registerCustomQueryHandler,
|
registerCustomQueryHandler,
|
||||||
unregisterCustomQueryHandler,
|
unregisterCustomQueryHandler,
|
||||||
} from './QueryHandler.js';
|
} from './CustomQueryHandler.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Settings that are common to the Puppeteer class, regardless of environment.
|
* Settings that are common to the Puppeteer class, regardless of environment.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/**
|
/**
|
||||||
* Copyright 2020 Google Inc. All rights reserved.
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@ -14,315 +14,177 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import PuppeteerUtil from '../injected/injected.js';
|
|
||||||
import {assert} from '../util/assert.js';
|
|
||||||
import {ariaHandler} from './AriaQueryHandler.js';
|
|
||||||
import {ElementHandle} from '../api/ElementHandle.js';
|
import {ElementHandle} from '../api/ElementHandle.js';
|
||||||
import {Frame} from './Frame.js';
|
import type PuppeteerUtil from '../injected/injected.js';
|
||||||
import {WaitForSelectorOptions} from './IsolatedWorld.js';
|
import {assert} from '../util/assert.js';
|
||||||
|
import {createFunction} from '../util/Function.js';
|
||||||
|
import {transposeIterableHandle} from './HandleIterator.js';
|
||||||
|
import type {Frame} from './Frame.js';
|
||||||
|
import type {WaitForSelectorOptions} from './IsolatedWorld.js';
|
||||||
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
||||||
import {LazyArg} from './LazyArg.js';
|
import {LazyArg} from './LazyArg.js';
|
||||||
|
import type {Awaitable, AwaitableIterable} from './types.js';
|
||||||
/**
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
export interface CustomQueryHandler {
|
|
||||||
/**
|
|
||||||
* @returns A {@link Node} matching the given `selector` from {@link node}.
|
|
||||||
*/
|
|
||||||
queryOne?: (node: Node, selector: string) => Node | null;
|
|
||||||
/**
|
|
||||||
* @returns Some {@link Node}s matching the given `selector` from {@link node}.
|
|
||||||
*/
|
|
||||||
queryAll?: (node: Node, selector: string) => Node[];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
export interface InternalQueryHandler {
|
export type QuerySelectorAll = (
|
||||||
/**
|
|
||||||
* @returns A {@link Node} matching the given `selector` from {@link node}.
|
|
||||||
*/
|
|
||||||
queryOne?: (
|
|
||||||
node: Node,
|
node: Node,
|
||||||
selector: string,
|
selector: string,
|
||||||
PuppeteerUtil: PuppeteerUtil
|
PuppeteerUtil: PuppeteerUtil
|
||||||
) => Node | null;
|
) => AwaitableIterable<Node>;
|
||||||
/**
|
|
||||||
* @returns Some {@link Node}s matching the given `selector` from {@link node}.
|
|
||||||
*/
|
|
||||||
queryAll?: (
|
|
||||||
node: Node,
|
|
||||||
selector: string,
|
|
||||||
PuppeteerUtil: PuppeteerUtil
|
|
||||||
) => Node[];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
export interface PuppeteerQueryHandler {
|
export type QuerySelector = (
|
||||||
|
node: Node,
|
||||||
|
selector: string,
|
||||||
|
PuppeteerUtil: PuppeteerUtil
|
||||||
|
) => Awaitable<Node | null>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queries for a single node given a selector and {@link ElementHandle}.
|
* @internal
|
||||||
*
|
|
||||||
* Akin to {@link Window.prototype.querySelector}.
|
|
||||||
*/
|
*/
|
||||||
queryOne?: (
|
export class QueryHandler {
|
||||||
element: ElementHandle<Node>,
|
// Either one of these may be implemented, but at least one must be.
|
||||||
selector: string
|
static querySelectorAll?: QuerySelectorAll;
|
||||||
) => Promise<ElementHandle<Node> | null>;
|
static querySelector?: QuerySelector;
|
||||||
|
|
||||||
|
static get _querySelector(): QuerySelector {
|
||||||
|
if (this.querySelector) {
|
||||||
|
return this.querySelector;
|
||||||
|
}
|
||||||
|
if (!this.querySelectorAll) {
|
||||||
|
throw new Error('Cannot create default query selector');
|
||||||
|
}
|
||||||
|
|
||||||
|
const querySelector: QuerySelector = async (
|
||||||
|
node,
|
||||||
|
selector,
|
||||||
|
PuppeteerUtil
|
||||||
|
) => {
|
||||||
|
const querySelectorAll =
|
||||||
|
'FUNCTION_DEFINITION' as unknown as QuerySelectorAll;
|
||||||
|
const results = querySelectorAll(node, selector, PuppeteerUtil);
|
||||||
|
for await (const result of results) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
return (this.querySelector = createFunction(
|
||||||
|
querySelector
|
||||||
|
.toString()
|
||||||
|
.replace("'FUNCTION_DEFINITION'", this.querySelectorAll.toString())
|
||||||
|
) as typeof querySelector);
|
||||||
|
}
|
||||||
|
|
||||||
|
static get _querySelectorAll(): QuerySelectorAll {
|
||||||
|
if (this.querySelectorAll) {
|
||||||
|
return this.querySelectorAll;
|
||||||
|
}
|
||||||
|
if (!this.querySelector) {
|
||||||
|
throw new Error('Cannot create default query selector');
|
||||||
|
}
|
||||||
|
|
||||||
|
const querySelectorAll: QuerySelectorAll = async function* (
|
||||||
|
node,
|
||||||
|
selector,
|
||||||
|
PuppeteerUtil
|
||||||
|
) {
|
||||||
|
const querySelector = 'FUNCTION_DEFINITION' as unknown as QuerySelector;
|
||||||
|
const result = await querySelector(node, selector, PuppeteerUtil);
|
||||||
|
if (result) {
|
||||||
|
yield result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (this.querySelectorAll = createFunction(
|
||||||
|
querySelectorAll
|
||||||
|
.toString()
|
||||||
|
.replace("'FUNCTION_DEFINITION'", this.querySelector.toString())
|
||||||
|
) as typeof querySelectorAll);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queries for multiple nodes given a selector and {@link ElementHandle}.
|
* Queries for multiple nodes given a selector and {@link ElementHandle}.
|
||||||
*
|
*
|
||||||
* Akin to {@link Window.prototype.querySelectorAll}.
|
* Akin to {@link Window.prototype.querySelectorAll}.
|
||||||
*/
|
*/
|
||||||
queryAll?: (
|
static async *queryAll(
|
||||||
element: ElementHandle<Node>,
|
element: ElementHandle<Node>,
|
||||||
selector: string
|
selector: string
|
||||||
) => Promise<Array<ElementHandle<Node>>>;
|
): AwaitableIterable<ElementHandle<Node>> {
|
||||||
|
const world = element.executionContext()._world;
|
||||||
|
assert(world);
|
||||||
|
const handle = await element.evaluateHandle(
|
||||||
|
this._querySelectorAll,
|
||||||
|
selector,
|
||||||
|
LazyArg.create(context => {
|
||||||
|
return context.puppeteerUtil;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
yield* transposeIterableHandle(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Queries for a single node given a selector and {@link ElementHandle}.
|
||||||
|
*
|
||||||
|
* Akin to {@link Window.prototype.querySelector}.
|
||||||
|
*/
|
||||||
|
static async queryOne(
|
||||||
|
element: ElementHandle<Node>,
|
||||||
|
selector: string
|
||||||
|
): Promise<ElementHandle<Node> | null> {
|
||||||
|
const world = element.executionContext()._world;
|
||||||
|
assert(world);
|
||||||
|
const result = await element.evaluateHandle(
|
||||||
|
this._querySelector,
|
||||||
|
selector,
|
||||||
|
LazyArg.create(context => {
|
||||||
|
return context.puppeteerUtil;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
if (!(result instanceof ElementHandle)) {
|
||||||
|
await result.dispose();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Waits until a single node appears for a given selector and
|
* Waits until a single node appears for a given selector and
|
||||||
* {@link ElementHandle}.
|
* {@link ElementHandle}.
|
||||||
*/
|
*/
|
||||||
waitFor?: (
|
static async waitFor(
|
||||||
elementOrFrame: ElementHandle<Node> | Frame,
|
elementOrFrame: ElementHandle<Node> | Frame,
|
||||||
selector: string,
|
selector: string,
|
||||||
options: WaitForSelectorOptions
|
options: WaitForSelectorOptions,
|
||||||
) => Promise<ElementHandle<Node> | null>;
|
bindings = new Map<string, (...args: never[]) => unknown>()
|
||||||
}
|
): Promise<ElementHandle<Node> | null> {
|
||||||
|
|
||||||
function createPuppeteerQueryHandler(
|
|
||||||
handler: InternalQueryHandler
|
|
||||||
): PuppeteerQueryHandler {
|
|
||||||
const internalHandler: PuppeteerQueryHandler = {};
|
|
||||||
|
|
||||||
if (handler.queryOne) {
|
|
||||||
const queryOne = handler.queryOne;
|
|
||||||
internalHandler.queryOne = async (element, selector) => {
|
|
||||||
const world = element.executionContext()._world;
|
|
||||||
assert(world);
|
|
||||||
const jsHandle = await element.evaluateHandle(
|
|
||||||
queryOne,
|
|
||||||
selector,
|
|
||||||
LazyArg.create(context => {
|
|
||||||
return context.puppeteerUtil;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
const elementHandle = jsHandle.asElement();
|
|
||||||
if (elementHandle) {
|
|
||||||
return elementHandle;
|
|
||||||
}
|
|
||||||
await jsHandle.dispose();
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
internalHandler.waitFor = async (elementOrFrame, selector, options) => {
|
|
||||||
let frame: Frame;
|
let frame: Frame;
|
||||||
let element: ElementHandle<Node> | undefined;
|
let element: ElementHandle<Node> | undefined;
|
||||||
if (elementOrFrame instanceof Frame) {
|
if (!(elementOrFrame instanceof ElementHandle)) {
|
||||||
frame = elementOrFrame;
|
frame = elementOrFrame;
|
||||||
} else {
|
} else {
|
||||||
frame = elementOrFrame.frame;
|
frame = elementOrFrame.frame;
|
||||||
element = await frame.worlds[PUPPETEER_WORLD].adoptHandle(
|
element = await frame.worlds[PUPPETEER_WORLD].adoptHandle(elementOrFrame);
|
||||||
elementOrFrame
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage(
|
const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage(
|
||||||
queryOne,
|
this._querySelector,
|
||||||
element,
|
element,
|
||||||
selector,
|
selector,
|
||||||
options
|
options,
|
||||||
|
bindings
|
||||||
);
|
);
|
||||||
if (element) {
|
if (element) {
|
||||||
await element.dispose();
|
await element.dispose();
|
||||||
}
|
}
|
||||||
if (!result) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
if (!(result instanceof ElementHandle)) {
|
if (!(result instanceof ElementHandle)) {
|
||||||
await result.dispose();
|
await result?.dispose();
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return frame.worlds[MAIN_WORLD].transferHandle(result);
|
return frame.worlds[MAIN_WORLD].transferHandle(result);
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handler.queryAll) {
|
|
||||||
const queryAll = handler.queryAll;
|
|
||||||
internalHandler.queryAll = async (element, selector) => {
|
|
||||||
const world = element.executionContext()._world;
|
|
||||||
assert(world);
|
|
||||||
const jsHandle = await element.evaluateHandle(
|
|
||||||
queryAll,
|
|
||||||
selector,
|
|
||||||
LazyArg.create(context => {
|
|
||||||
return context.puppeteerUtil;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
const properties = await jsHandle.getProperties();
|
|
||||||
await jsHandle.dispose();
|
|
||||||
const result = [];
|
|
||||||
for (const property of properties.values()) {
|
|
||||||
const elementHandle = property.asElement();
|
|
||||||
if (elementHandle) {
|
|
||||||
result.push(elementHandle);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return internalHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
const defaultHandler = createPuppeteerQueryHandler({
|
|
||||||
queryOne: (element, selector) => {
|
|
||||||
if (!('querySelector' in element)) {
|
|
||||||
throw new Error(
|
|
||||||
`Could not invoke \`querySelector\` on node of type ${element.nodeName}.`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
element as unknown as {querySelector(selector: string): Element}
|
|
||||||
).querySelector(selector);
|
|
||||||
},
|
|
||||||
queryAll: (element, selector) => {
|
|
||||||
if (!('querySelectorAll' in element)) {
|
|
||||||
throw new Error(
|
|
||||||
`Could not invoke \`querySelectorAll\` on node of type ${element.nodeName}.`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return [
|
|
||||||
...(
|
|
||||||
element as unknown as {
|
|
||||||
querySelectorAll(selector: string): NodeList;
|
|
||||||
}
|
|
||||||
).querySelectorAll(selector),
|
|
||||||
];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const pierceHandler = createPuppeteerQueryHandler({
|
|
||||||
queryOne: (element, selector, {pierceQuerySelector}) => {
|
|
||||||
return pierceQuerySelector(element, selector);
|
|
||||||
},
|
|
||||||
queryAll: (element, selector, {pierceQuerySelectorAll}) => {
|
|
||||||
return pierceQuerySelectorAll(element, selector);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const xpathHandler = createPuppeteerQueryHandler({
|
|
||||||
queryOne: (element, selector, {xpathQuerySelector}) => {
|
|
||||||
return xpathQuerySelector(element, selector);
|
|
||||||
},
|
|
||||||
queryAll: (element, selector, {xpathQuerySelectorAll}) => {
|
|
||||||
return xpathQuerySelectorAll(element, selector);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const textQueryHandler = createPuppeteerQueryHandler({
|
|
||||||
queryOne: (element, selector, {textQuerySelector}) => {
|
|
||||||
return textQuerySelector(element, selector);
|
|
||||||
},
|
|
||||||
queryAll: (element, selector, {textQuerySelectorAll}) => {
|
|
||||||
return textQuerySelectorAll(element, selector);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
interface RegisteredQueryHandler {
|
|
||||||
handler: PuppeteerQueryHandler;
|
|
||||||
transformSelector?: (selector: string) => string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const INTERNAL_QUERY_HANDLERS = new Map<string, RegisteredQueryHandler>([
|
|
||||||
['aria', {handler: ariaHandler}],
|
|
||||||
['pierce', {handler: pierceHandler}],
|
|
||||||
['xpath', {handler: xpathHandler}],
|
|
||||||
['text', {handler: textQueryHandler}],
|
|
||||||
]);
|
|
||||||
const QUERY_HANDLERS = new Map<string, RegisteredQueryHandler>();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated Import {@link Puppeteer} and use the static method
|
|
||||||
* {@link Puppeteer.registerCustomQueryHandler}
|
|
||||||
*
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
export function registerCustomQueryHandler(
|
|
||||||
name: string,
|
|
||||||
handler: CustomQueryHandler
|
|
||||||
): void {
|
|
||||||
if (INTERNAL_QUERY_HANDLERS.has(name)) {
|
|
||||||
throw new Error(`A query handler named "${name}" already exists`);
|
|
||||||
}
|
|
||||||
if (QUERY_HANDLERS.has(name)) {
|
|
||||||
throw new Error(`A custom query handler named "${name}" already exists`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const isValidName = /^[a-zA-Z]+$/.test(name);
|
|
||||||
if (!isValidName) {
|
|
||||||
throw new Error(`Custom query handler names may only contain [a-zA-Z]`);
|
|
||||||
}
|
|
||||||
|
|
||||||
QUERY_HANDLERS.set(name, {handler: createPuppeteerQueryHandler(handler)});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated Import {@link Puppeteer} and use the static method
|
|
||||||
* {@link Puppeteer.unregisterCustomQueryHandler}
|
|
||||||
*
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
export function unregisterCustomQueryHandler(name: string): void {
|
|
||||||
QUERY_HANDLERS.delete(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated Import {@link Puppeteer} and use the static method
|
|
||||||
* {@link Puppeteer.customQueryHandlerNames}
|
|
||||||
*
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
export function customQueryHandlerNames(): string[] {
|
|
||||||
return [...QUERY_HANDLERS.keys()];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated Import {@link Puppeteer} and use the static method
|
|
||||||
* {@link Puppeteer.clearCustomQueryHandlers}
|
|
||||||
*
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
export function clearCustomQueryHandlers(): void {
|
|
||||||
QUERY_HANDLERS.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
const CUSTOM_QUERY_SEPARATORS = ['=', '/'];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
export function getQueryHandlerAndSelector(selector: string): {
|
|
||||||
updatedSelector: string;
|
|
||||||
queryHandler: PuppeteerQueryHandler;
|
|
||||||
} {
|
|
||||||
for (const handlerMap of [QUERY_HANDLERS, INTERNAL_QUERY_HANDLERS]) {
|
|
||||||
for (const [
|
|
||||||
name,
|
|
||||||
{handler: queryHandler, transformSelector},
|
|
||||||
] of handlerMap) {
|
|
||||||
for (const separator of CUSTOM_QUERY_SEPARATORS) {
|
|
||||||
const prefix = `${name}${separator}`;
|
|
||||||
if (selector.startsWith(prefix)) {
|
|
||||||
selector = selector.slice(prefix.length);
|
|
||||||
if (transformSelector) {
|
|
||||||
selector = transformSelector(selector);
|
|
||||||
}
|
|
||||||
return {updatedSelector: selector, queryHandler};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {updatedSelector: selector, queryHandler: defaultHandler};
|
|
||||||
}
|
|
||||||
|
30
packages/puppeteer-core/src/common/TextQueryHandler.ts
Normal file
30
packages/puppeteer-core/src/common/TextQueryHandler.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {QueryHandler, QuerySelectorAll} from './QueryHandler.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class TextQueryHandler extends QueryHandler {
|
||||||
|
static override querySelectorAll: QuerySelectorAll = (
|
||||||
|
element,
|
||||||
|
selector,
|
||||||
|
{textQuerySelectorAll}
|
||||||
|
) => {
|
||||||
|
return textQuerySelectorAll(element, selector);
|
||||||
|
};
|
||||||
|
}
|
30
packages/puppeteer-core/src/common/XPathQueryHandler.ts
Normal file
30
packages/puppeteer-core/src/common/XPathQueryHandler.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import {QueryHandler, QuerySelectorAll} from './QueryHandler.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export class XPathQueryHandler extends QueryHandler {
|
||||||
|
static override querySelectorAll: QuerySelectorAll = (
|
||||||
|
element,
|
||||||
|
selector,
|
||||||
|
{xpathQuerySelectorAll}
|
||||||
|
) => {
|
||||||
|
return xpathQuerySelectorAll(element, selector);
|
||||||
|
};
|
||||||
|
}
|
@ -25,6 +25,7 @@ export * from './Connection.js';
|
|||||||
export * from './ConnectionTransport.js';
|
export * from './ConnectionTransport.js';
|
||||||
export * from './ConsoleMessage.js';
|
export * from './ConsoleMessage.js';
|
||||||
export * from './Coverage.js';
|
export * from './Coverage.js';
|
||||||
|
export * from './CustomQueryHandler.js';
|
||||||
export * from './Debug.js';
|
export * from './Debug.js';
|
||||||
export * from './Device.js';
|
export * from './Device.js';
|
||||||
export * from './Dialog.js';
|
export * from './Dialog.js';
|
||||||
|
@ -32,6 +32,11 @@ export type BindingPayload = {
|
|||||||
isTrivial: boolean;
|
isTrivial: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @public
|
||||||
|
*/
|
||||||
|
export type AwaitableIterable<T> = Iterable<T> | AsyncIterable<T>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @public
|
* @public
|
||||||
*/
|
*/
|
||||||
|
@ -14,20 +14,21 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Protocol} from 'devtools-protocol';
|
import type {Protocol} from 'devtools-protocol';
|
||||||
import type {Readable} from 'stream';
|
import type {Readable} from 'stream';
|
||||||
|
import type {ElementHandle} from '../api/ElementHandle.js';
|
||||||
|
import type {JSHandle} from '../api/JSHandle.js';
|
||||||
import {isNode} from '../environment.js';
|
import {isNode} from '../environment.js';
|
||||||
import {assert} from '../util/assert.js';
|
import {assert} from '../util/assert.js';
|
||||||
import {isErrorLike} from '../util/ErrorLike.js';
|
import {isErrorLike} from '../util/ErrorLike.js';
|
||||||
import {CDPSession} from './Connection.js';
|
import type {CDPSession} from './Connection.js';
|
||||||
import {debug} from './Debug.js';
|
import {debug} from './Debug.js';
|
||||||
import {ElementHandle} from '../api/ElementHandle.js';
|
|
||||||
import {CDPElementHandle} from './ElementHandle.js';
|
import {CDPElementHandle} from './ElementHandle.js';
|
||||||
import {TimeoutError} from './Errors.js';
|
import {TimeoutError} from './Errors.js';
|
||||||
import {CommonEventEmitter} from './EventEmitter.js';
|
import type {CommonEventEmitter} from './EventEmitter.js';
|
||||||
import {ExecutionContext} from './ExecutionContext.js';
|
import type {ExecutionContext} from './ExecutionContext.js';
|
||||||
import {JSHandle} from '../api/JSHandle.js';
|
|
||||||
import {CDPJSHandle} from './JSHandle.js';
|
import {CDPJSHandle} from './JSHandle.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
|
@ -19,68 +19,38 @@ import {
|
|||||||
isSuitableNodeForTextMatching,
|
isSuitableNodeForTextMatching,
|
||||||
} from './TextContent.js';
|
} from './TextContent.js';
|
||||||
|
|
||||||
/**
|
|
||||||
* Queries the given node for a node matching the given text selector.
|
|
||||||
*
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
export const textQuerySelector = (
|
|
||||||
root: Node,
|
|
||||||
selector: string
|
|
||||||
): Element | null => {
|
|
||||||
for (const node of root.childNodes) {
|
|
||||||
if (node instanceof Element && isSuitableNodeForTextMatching(node)) {
|
|
||||||
let matchedNode: Element | null;
|
|
||||||
if (node.shadowRoot) {
|
|
||||||
matchedNode = textQuerySelector(node.shadowRoot, selector);
|
|
||||||
} else {
|
|
||||||
matchedNode = textQuerySelector(node, selector);
|
|
||||||
}
|
|
||||||
if (matchedNode) {
|
|
||||||
return matchedNode;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (root instanceof Element) {
|
|
||||||
const textContent = createTextContent(root);
|
|
||||||
if (textContent.full.includes(selector)) {
|
|
||||||
return root;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queries the given node for all nodes matching the given text selector.
|
* Queries the given node for all nodes matching the given text selector.
|
||||||
*
|
*
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
export const textQuerySelectorAll = (
|
export const textQuerySelectorAll = function* (
|
||||||
root: Node,
|
root: Node,
|
||||||
selector: string
|
selector: string
|
||||||
): Element[] => {
|
): Generator<Element> {
|
||||||
let results: Element[] = [];
|
let yielded = false;
|
||||||
for (const node of root.childNodes) {
|
for (const node of root.childNodes) {
|
||||||
if (node instanceof Element) {
|
if (node instanceof Element && isSuitableNodeForTextMatching(node)) {
|
||||||
let matchedNodes: Element[];
|
let matches: Generator<Element, boolean>;
|
||||||
if (node.shadowRoot) {
|
if (!node.shadowRoot) {
|
||||||
matchedNodes = textQuerySelectorAll(node.shadowRoot, selector);
|
matches = textQuerySelectorAll(node, selector);
|
||||||
} else {
|
} else {
|
||||||
matchedNodes = textQuerySelectorAll(node, selector);
|
matches = textQuerySelectorAll(node.shadowRoot, selector);
|
||||||
}
|
}
|
||||||
results = results.concat(matchedNodes);
|
for (const match of matches) {
|
||||||
|
yield match;
|
||||||
|
yielded = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (results.length > 0) {
|
}
|
||||||
return results;
|
if (yielded) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (root instanceof Element) {
|
if (root instanceof Element && isSuitableNodeForTextMatching(root)) {
|
||||||
const textContent = createTextContent(root);
|
const textContent = createTextContent(root);
|
||||||
if (textContent.full.includes(selector)) {
|
if (textContent.full.includes(selector)) {
|
||||||
return [root];
|
yield root;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return [];
|
|
||||||
};
|
};
|
||||||
|
@ -17,24 +17,10 @@
|
|||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
export const xpathQuerySelector = (
|
export const xpathQuerySelectorAll = function* (
|
||||||
root: Node,
|
root: Node,
|
||||||
selector: string
|
selector: string
|
||||||
): Node | null => {
|
): Iterable<Node> {
|
||||||
const doc = root.ownerDocument || document;
|
|
||||||
const result = doc.evaluate(
|
|
||||||
selector,
|
|
||||||
root,
|
|
||||||
null,
|
|
||||||
XPathResult.FIRST_ORDERED_NODE_TYPE
|
|
||||||
);
|
|
||||||
return result.singleNodeValue;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
export const xpathQuerySelectorAll = (root: Node, selector: string): Node[] => {
|
|
||||||
const doc = root.ownerDocument || document;
|
const doc = root.ownerDocument || document;
|
||||||
const iterator = doc.evaluate(
|
const iterator = doc.evaluate(
|
||||||
selector,
|
selector,
|
||||||
@ -42,10 +28,8 @@ export const xpathQuerySelectorAll = (root: Node, selector: string): Node[] => {
|
|||||||
null,
|
null,
|
||||||
XPathResult.ORDERED_NODE_ITERATOR_TYPE
|
XPathResult.ORDERED_NODE_ITERATOR_TYPE
|
||||||
);
|
);
|
||||||
const array: Node[] = [];
|
|
||||||
let item;
|
let item;
|
||||||
while ((item = iterator.iterateNext())) {
|
while ((item = iterator.iterateNext())) {
|
||||||
array.push(item);
|
yield item;
|
||||||
}
|
}
|
||||||
return array;
|
|
||||||
};
|
};
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {createDeferredPromise} from '../util/DeferredPromise.js';
|
import {createDeferredPromise} from '../util/DeferredPromise.js';
|
||||||
|
import {createFunction} from '../util/Function.js';
|
||||||
import {RAFPoller, MutationPoller, IntervalPoller} from './Poller.js';
|
import {RAFPoller, MutationPoller, IntervalPoller} from './Poller.js';
|
||||||
import {
|
import {
|
||||||
isSuitableNodeForTextMatching,
|
isSuitableNodeForTextMatching,
|
||||||
@ -33,6 +34,7 @@ const PuppeteerUtil = Object.freeze({
|
|||||||
...TextQuerySelector,
|
...TextQuerySelector,
|
||||||
...XPathQuerySelector,
|
...XPathQuerySelector,
|
||||||
...PierceQuerySelector,
|
...PierceQuerySelector,
|
||||||
|
createFunction,
|
||||||
createDeferredPromise,
|
createDeferredPromise,
|
||||||
createTextContent,
|
createTextContent,
|
||||||
IntervalPoller,
|
IntervalPoller,
|
||||||
|
@ -1,40 +1,3 @@
|
|||||||
/**
|
|
||||||
* Copyright 2022 Google Inc. All rights reserved.
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const createdFunctions = new Map<string, (...args: unknown[]) => unknown>();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a function from a string.
|
|
||||||
*
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
export const createFunction = (
|
|
||||||
functionValue: string
|
|
||||||
): ((...args: unknown[]) => unknown) => {
|
|
||||||
let fn = createdFunctions.get(functionValue);
|
|
||||||
if (fn) {
|
|
||||||
return fn;
|
|
||||||
}
|
|
||||||
fn = new Function(`return ${functionValue}`)() as (
|
|
||||||
...args: unknown[]
|
|
||||||
) => unknown;
|
|
||||||
createdFunctions.set(functionValue, fn);
|
|
||||||
return fn;
|
|
||||||
};
|
|
||||||
|
|
||||||
const HIDDEN_VISIBILITY_VALUES = ['hidden', 'collapse'];
|
const HIDDEN_VISIBILITY_VALUES = ['hidden', 'collapse'];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -25,7 +25,7 @@ export * from './util/util.js';
|
|||||||
/**
|
/**
|
||||||
* @deprecated Use the query handler API defined on {@link Puppeteer}
|
* @deprecated Use the query handler API defined on {@link Puppeteer}
|
||||||
*/
|
*/
|
||||||
export * from './common/QueryHandler.js';
|
export * from './common/CustomQueryHandler.js';
|
||||||
|
|
||||||
import {PuppeteerNode} from './node/PuppeteerNode.js';
|
import {PuppeteerNode} from './node/PuppeteerNode.js';
|
||||||
|
|
||||||
|
35
packages/puppeteer-core/src/util/Function.ts
Normal file
35
packages/puppeteer-core/src/util/Function.ts
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
/**
|
||||||
|
* Copyright 2023 Google Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
const createdFunctions = new Map<string, (...args: unknown[]) => unknown>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a function from a string.
|
||||||
|
*
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
export const createFunction = (
|
||||||
|
functionValue: string
|
||||||
|
): ((...args: unknown[]) => unknown) => {
|
||||||
|
let fn = createdFunctions.get(functionValue);
|
||||||
|
if (fn) {
|
||||||
|
return fn;
|
||||||
|
}
|
||||||
|
fn = new Function(`return ${functionValue}`)() as (
|
||||||
|
...args: unknown[]
|
||||||
|
) => unknown;
|
||||||
|
createdFunctions.set(functionValue, fn);
|
||||||
|
return fn;
|
||||||
|
};
|
@ -15,7 +15,7 @@
|
|||||||
*/
|
*/
|
||||||
import expect from 'expect';
|
import expect from 'expect';
|
||||||
import {Puppeteer} from 'puppeteer';
|
import {Puppeteer} from 'puppeteer';
|
||||||
import {CustomQueryHandler} from 'puppeteer-core/internal/common/QueryHandler.js';
|
import type {CustomQueryHandler} from 'puppeteer-core/internal/common/CustomQueryHandler.js';
|
||||||
import {
|
import {
|
||||||
getTestState,
|
getTestState,
|
||||||
setupTestBrowserHooks,
|
setupTestBrowserHooks,
|
||||||
|
Loading…
Reference in New Issue
Block a user