2023-02-14 21:31:30 +00:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2023-02-15 15:33:18 +00:00
|
|
|
import type PuppeteerUtil from '../injected/injected.js';
|
|
|
|
import {assert} from '../util/assert.js';
|
|
|
|
import {interpolateFunction, stringifyFunction} from '../util/Function.js';
|
|
|
|
import {QueryHandler, QuerySelector, QuerySelectorAll} from './QueryHandler.js';
|
|
|
|
import {scriptInjector} from './ScriptInjector.js';
|
2023-02-14 21:31:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @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>;
|
|
|
|
}
|
|
|
|
|
2023-02-15 15:33:18 +00:00
|
|
|
/**
|
|
|
|
* The registry of {@link CustomQueryHandler | custom query handlers}.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
*
|
|
|
|
* ```ts
|
|
|
|
* Puppeteer.customQueryHandlers.register('lit', { … });
|
|
|
|
* const aHandle = await page.$('lit/…');
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export class CustomQueryHandlerRegistry {
|
|
|
|
#handlers = new Map<
|
|
|
|
string,
|
|
|
|
[registerScript: string, Handler: typeof QueryHandler]
|
|
|
|
>();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
get(name: string): typeof QueryHandler | undefined {
|
|
|
|
const handler = this.#handlers.get(name);
|
|
|
|
return handler ? handler[1] : undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Registers a {@link CustomQueryHandler | custom query handler}.
|
|
|
|
*
|
|
|
|
* @remarks
|
|
|
|
* After registration, the handler can be used everywhere where a selector is
|
|
|
|
* expected by prepending the selection string with `<name>/`. The name is
|
|
|
|
* only allowed to consist of lower- and upper case latin letters.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
*
|
|
|
|
* ```ts
|
|
|
|
* Puppeteer.customQueryHandlers.register('lit', { … });
|
|
|
|
* const aHandle = await page.$('lit/…');
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* @param name - Name to register under.
|
|
|
|
* @param queryHandler - {@link CustomQueryHandler | Custom query handler} to
|
|
|
|
* register.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
register(name: string, handler: CustomQueryHandler): void {
|
|
|
|
if (this.#handlers.has(name)) {
|
|
|
|
throw new Error(`Cannot register over existing handler: ${name}`);
|
|
|
|
}
|
|
|
|
assert(
|
|
|
|
!this.#handlers.has(name),
|
|
|
|
`Cannot register over existing handler: ${name}`
|
|
|
|
);
|
|
|
|
assert(
|
|
|
|
/^[a-zA-Z]+$/.test(name),
|
|
|
|
`Custom query handler names may only contain [a-zA-Z]`
|
|
|
|
);
|
|
|
|
assert(
|
|
|
|
handler.queryAll || handler.queryOne,
|
|
|
|
`At least one query method must be implemented.`
|
|
|
|
);
|
|
|
|
|
|
|
|
const Handler = class extends QueryHandler {
|
|
|
|
static override querySelectorAll: QuerySelectorAll = interpolateFunction(
|
|
|
|
(node, selector, PuppeteerUtil) => {
|
|
|
|
return PuppeteerUtil.customQuerySelectors
|
|
|
|
.get(PLACEHOLDER('name'))!
|
|
|
|
.querySelectorAll(node, selector);
|
|
|
|
},
|
|
|
|
{name: JSON.stringify(name)}
|
|
|
|
);
|
|
|
|
static override querySelector: QuerySelector = interpolateFunction(
|
|
|
|
(node, selector, PuppeteerUtil) => {
|
|
|
|
return PuppeteerUtil.customQuerySelectors
|
|
|
|
.get(PLACEHOLDER('name'))!
|
|
|
|
.querySelector(node, selector);
|
|
|
|
},
|
|
|
|
{name: JSON.stringify(name)}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
const registerScript = interpolateFunction(
|
|
|
|
(PuppeteerUtil: PuppeteerUtil) => {
|
|
|
|
PuppeteerUtil.customQuerySelectors.register(PLACEHOLDER('name'), {
|
|
|
|
queryAll: PLACEHOLDER('queryAll'),
|
|
|
|
queryOne: PLACEHOLDER('queryOne'),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: JSON.stringify(name),
|
|
|
|
queryAll: handler.queryAll
|
|
|
|
? stringifyFunction(handler.queryAll)
|
|
|
|
: String(undefined),
|
|
|
|
queryOne: handler.queryOne
|
|
|
|
? stringifyFunction(handler.queryOne)
|
|
|
|
: String(undefined),
|
|
|
|
}
|
|
|
|
).toString();
|
|
|
|
|
|
|
|
this.#handlers.set(name, [registerScript, Handler]);
|
|
|
|
scriptInjector.append(registerScript);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unregisters the {@link CustomQueryHandler | custom query handler} for the
|
|
|
|
* given name.
|
|
|
|
*
|
|
|
|
* @throws `Error` if there is no handler under the given name.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
unregister(name: string): void {
|
|
|
|
const handler = this.#handlers.get(name);
|
|
|
|
if (!handler) {
|
|
|
|
throw new Error(`Cannot unregister unknown handler: ${name}`);
|
|
|
|
}
|
|
|
|
scriptInjector.pop(handler[0]);
|
|
|
|
this.#handlers.delete(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the names of all {@link CustomQueryHandler | custom query handlers}.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
names(): string[] {
|
|
|
|
return [...this.#handlers.keys()];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unregisters all custom query handlers.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
clear(): void {
|
|
|
|
for (const [registerScript] of this.#handlers) {
|
|
|
|
scriptInjector.pop(registerScript);
|
|
|
|
}
|
|
|
|
this.#handlers.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export const customQueryHandlers = new CustomQueryHandlerRegistry();
|
|
|
|
|
2023-02-14 21:31:30 +00:00
|
|
|
/**
|
|
|
|
* @deprecated Import {@link Puppeteer} and use the static method
|
|
|
|
* {@link Puppeteer.registerCustomQueryHandler}
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export function registerCustomQueryHandler(
|
|
|
|
name: string,
|
|
|
|
handler: CustomQueryHandler
|
|
|
|
): void {
|
2023-02-15 15:33:18 +00:00
|
|
|
customQueryHandlers.register(name, handler);
|
2023-02-14 21:31:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Import {@link Puppeteer} and use the static method
|
|
|
|
* {@link Puppeteer.unregisterCustomQueryHandler}
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export function unregisterCustomQueryHandler(name: string): void {
|
2023-02-15 15:33:18 +00:00
|
|
|
customQueryHandlers.unregister(name);
|
2023-02-14 21:31:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Import {@link Puppeteer} and use the static method
|
|
|
|
* {@link Puppeteer.customQueryHandlerNames}
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export function customQueryHandlerNames(): string[] {
|
2023-02-15 15:33:18 +00:00
|
|
|
return customQueryHandlers.names();
|
2023-02-14 21:31:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Import {@link Puppeteer} and use the static method
|
|
|
|
* {@link Puppeteer.clearCustomQueryHandlers}
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
export function clearCustomQueryHandlers(): void {
|
|
|
|
customQueryHandlers.clear();
|
|
|
|
}
|