mirror of
https://github.com/puppeteer/puppeteer
synced 2024-06-14 14:02:48 +00:00
chore: refactor exposed function for BiDi (#11976)
This commit is contained in:
parent
04392d8f3f
commit
7966dd7262
@ -6,29 +6,24 @@
|
|||||||
|
|
||||||
import * as Bidi from 'chromium-bidi/lib/cjs/protocol/protocol.js';
|
import * as Bidi from 'chromium-bidi/lib/cjs/protocol/protocol.js';
|
||||||
|
|
||||||
|
import {EventEmitter} from '../common/EventEmitter.js';
|
||||||
import type {Awaitable, FlattenHandle} from '../common/types.js';
|
import type {Awaitable, FlattenHandle} from '../common/types.js';
|
||||||
import {debugError} from '../common/util.js';
|
import {debugError} from '../common/util.js';
|
||||||
import {assert} from '../util/assert.js';
|
import {DisposableStack} from '../util/disposable.js';
|
||||||
import {Deferred} from '../util/Deferred.js';
|
|
||||||
import {interpolateFunction, stringifyFunction} from '../util/Function.js';
|
import {interpolateFunction, stringifyFunction} from '../util/Function.js';
|
||||||
|
|
||||||
import type {Connection} from './core/Connection.js';
|
import type {Connection} from './core/Connection.js';
|
||||||
import {BidiDeserializer} from './Deserializer.js';
|
import {BidiElementHandle} from './ElementHandle.js';
|
||||||
import type {BidiFrame} from './Frame.js';
|
import type {BidiFrame} from './Frame.js';
|
||||||
import {BidiSerializer} from './Serializer.js';
|
import {BidiJSHandle} from './JSHandle.js';
|
||||||
|
|
||||||
type SendArgsChannel<Args> = (value: [id: number, args: Args]) => void;
|
type CallbackChannel<Args, Ret> = (
|
||||||
type SendResolveChannel<Ret> = (
|
value: [
|
||||||
value: [id: number, resolve: (ret: FlattenHandle<Awaited<Ret>>) => void]
|
resolve: (ret: FlattenHandle<Awaited<Ret>>) => void,
|
||||||
|
reject: (error: unknown) => void,
|
||||||
|
args: Args,
|
||||||
|
]
|
||||||
) => void;
|
) => void;
|
||||||
type SendRejectChannel = (
|
|
||||||
value: [id: number, reject: (error: unknown) => void]
|
|
||||||
) => void;
|
|
||||||
|
|
||||||
interface RemotePromiseCallbacks {
|
|
||||||
resolve: Deferred<Bidi.Script.RemoteValue>;
|
|
||||||
reject: Deferred<Bidi.Script.RemoteValue>;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
@ -38,65 +33,53 @@ export class ExposeableFunction<Args extends unknown[], Ret> {
|
|||||||
|
|
||||||
readonly name;
|
readonly name;
|
||||||
readonly #apply;
|
readonly #apply;
|
||||||
|
readonly #isolate;
|
||||||
|
|
||||||
readonly #channels;
|
readonly #channel;
|
||||||
readonly #callerInfos = new Map<
|
|
||||||
string,
|
|
||||||
Map<number, RemotePromiseCallbacks>
|
|
||||||
>();
|
|
||||||
|
|
||||||
#preloadScriptId?: Bidi.Script.PreloadScript;
|
#scripts: Array<[BidiFrame, Bidi.Script.PreloadScript]> = [];
|
||||||
|
#disposables = new DisposableStack();
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
frame: BidiFrame,
|
frame: BidiFrame,
|
||||||
name: string,
|
name: string,
|
||||||
apply: (...args: Args) => Awaitable<Ret>
|
apply: (...args: Args) => Awaitable<Ret>,
|
||||||
|
isolate = false
|
||||||
) {
|
) {
|
||||||
this.#frame = frame;
|
this.#frame = frame;
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.#apply = apply;
|
this.#apply = apply;
|
||||||
|
this.#isolate = isolate;
|
||||||
|
|
||||||
this.#channels = {
|
this.#channel = `__puppeteer__${this.#frame._id}_page_exposeFunction_${this.name}`;
|
||||||
args: `__puppeteer__${this.#frame._id}_page_exposeFunction_${this.name}_args`,
|
|
||||||
resolve: `__puppeteer__${this.#frame._id}_page_exposeFunction_${this.name}_resolve`,
|
|
||||||
reject: `__puppeteer__${this.#frame._id}_page_exposeFunction_${this.name}_reject`,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async expose(): Promise<void> {
|
async expose(): Promise<void> {
|
||||||
const connection = this.#connection;
|
const connection = this.#connection;
|
||||||
const channelArguments = this.#channelArguments;
|
const channel = {
|
||||||
|
type: 'channel' as const,
|
||||||
|
value: {
|
||||||
|
channel: this.#channel,
|
||||||
|
ownership: Bidi.Script.ResultOwnership.Root,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
// TODO(jrandolf): Implement cleanup with removePreloadScript.
|
const connectionEmitter = this.#disposables.use(
|
||||||
connection.on(
|
new EventEmitter(connection)
|
||||||
Bidi.ChromiumBidi.Script.EventNames.Message,
|
|
||||||
this.#handleArgumentsMessage
|
|
||||||
);
|
);
|
||||||
connection.on(
|
connectionEmitter.on(
|
||||||
Bidi.ChromiumBidi.Script.EventNames.Message,
|
Bidi.ChromiumBidi.Script.EventNames.Message,
|
||||||
this.#handleResolveMessage
|
this.#handleMessage
|
||||||
);
|
|
||||||
connection.on(
|
|
||||||
Bidi.ChromiumBidi.Script.EventNames.Message,
|
|
||||||
this.#handleRejectMessage
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const functionDeclaration = stringifyFunction(
|
const functionDeclaration = stringifyFunction(
|
||||||
interpolateFunction(
|
interpolateFunction(
|
||||||
(
|
(callback: CallbackChannel<Args, Ret>) => {
|
||||||
sendArgs: SendArgsChannel<Args>,
|
|
||||||
sendResolve: SendResolveChannel<Ret>,
|
|
||||||
sendReject: SendRejectChannel
|
|
||||||
) => {
|
|
||||||
let id = 0;
|
|
||||||
Object.assign(globalThis, {
|
Object.assign(globalThis, {
|
||||||
[PLACEHOLDER('name') as string]: function (...args: Args) {
|
[PLACEHOLDER('name') as string]: function (...args: Args) {
|
||||||
return new Promise<FlattenHandle<Awaited<Ret>>>(
|
return new Promise<FlattenHandle<Awaited<Ret>>>(
|
||||||
(resolve, reject) => {
|
(resolve, reject) => {
|
||||||
sendArgs([id, args]);
|
callback([resolve, reject, args]);
|
||||||
sendResolve([id, resolve]);
|
|
||||||
sendReject([id, reject]);
|
|
||||||
++id;
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -106,179 +89,140 @@ export class ExposeableFunction<Args extends unknown[], Ret> {
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
const {result} = await connection.send('script.addPreloadScript', {
|
const frames = [this.#frame];
|
||||||
functionDeclaration,
|
for (const frame of frames) {
|
||||||
arguments: channelArguments,
|
frames.push(...frame.childFrames());
|
||||||
contexts: [this.#frame.page().mainFrame()._id],
|
}
|
||||||
});
|
|
||||||
this.#preloadScriptId = result.script;
|
|
||||||
|
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
this.#frame
|
frames.map(async frame => {
|
||||||
.page()
|
const realm = this.#isolate ? frame.isolatedRealm() : frame.mainRealm();
|
||||||
.frames()
|
try {
|
||||||
.map(async frame => {
|
this.#scripts.push([
|
||||||
return await connection.send('script.callFunction', {
|
frame,
|
||||||
functionDeclaration,
|
await frame.browsingContext.addPreloadScript(functionDeclaration, {
|
||||||
arguments: channelArguments,
|
arguments: [channel],
|
||||||
awaitPromise: false,
|
sandbox: realm.sandbox,
|
||||||
target: frame.mainRealm().realm.target,
|
}),
|
||||||
|
]);
|
||||||
|
} catch (error) {
|
||||||
|
// If it errors, the frame probably doesn't support adding preload
|
||||||
|
// scripts. We fail gracefully.
|
||||||
|
debugError(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await realm.realm.callFunction(functionDeclaration, false, {
|
||||||
|
arguments: [channel],
|
||||||
});
|
});
|
||||||
})
|
} catch (error) {
|
||||||
|
// If it errors, the frame probably doesn't support call function. We
|
||||||
|
// fail gracefully.
|
||||||
|
debugError(error);
|
||||||
|
}
|
||||||
|
})
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#handleArgumentsMessage = async (params: Bidi.Script.MessageParameters) => {
|
|
||||||
if (params.channel !== this.#channels.args) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const connection = this.#connection;
|
|
||||||
const {callbacks, remoteValue} = this.#getCallbacksAndRemoteValue(params);
|
|
||||||
const args = remoteValue.value?.[1];
|
|
||||||
assert(args);
|
|
||||||
try {
|
|
||||||
const result = await this.#apply(...BidiDeserializer.deserialize(args));
|
|
||||||
await connection.send('script.callFunction', {
|
|
||||||
functionDeclaration: stringifyFunction(([_, resolve]: any, result) => {
|
|
||||||
resolve(result);
|
|
||||||
}),
|
|
||||||
arguments: [
|
|
||||||
(await callbacks.resolve.valueOrThrow()) as Bidi.Script.LocalValue,
|
|
||||||
BidiSerializer.serialize(result),
|
|
||||||
],
|
|
||||||
awaitPromise: false,
|
|
||||||
target: {
|
|
||||||
realm: params.source.realm,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} catch (error) {
|
|
||||||
try {
|
|
||||||
if (error instanceof Error) {
|
|
||||||
await connection.send('script.callFunction', {
|
|
||||||
functionDeclaration: stringifyFunction(
|
|
||||||
(
|
|
||||||
[_, reject]: [unknown, (error: Error) => void],
|
|
||||||
name: string,
|
|
||||||
message: string,
|
|
||||||
stack?: string
|
|
||||||
) => {
|
|
||||||
const error = new Error(message);
|
|
||||||
error.name = name;
|
|
||||||
if (stack) {
|
|
||||||
error.stack = stack;
|
|
||||||
}
|
|
||||||
reject(error);
|
|
||||||
}
|
|
||||||
),
|
|
||||||
arguments: [
|
|
||||||
(await callbacks.reject.valueOrThrow()) as Bidi.Script.LocalValue,
|
|
||||||
BidiSerializer.serialize(error.name),
|
|
||||||
BidiSerializer.serialize(error.message),
|
|
||||||
BidiSerializer.serialize(error.stack),
|
|
||||||
],
|
|
||||||
awaitPromise: false,
|
|
||||||
target: {
|
|
||||||
realm: params.source.realm,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
await connection.send('script.callFunction', {
|
|
||||||
functionDeclaration: stringifyFunction(
|
|
||||||
(
|
|
||||||
[_, reject]: [unknown, (error: unknown) => void],
|
|
||||||
error: unknown
|
|
||||||
) => {
|
|
||||||
reject(error);
|
|
||||||
}
|
|
||||||
),
|
|
||||||
arguments: [
|
|
||||||
(await callbacks.reject.valueOrThrow()) as Bidi.Script.LocalValue,
|
|
||||||
BidiSerializer.serialize(error),
|
|
||||||
],
|
|
||||||
awaitPromise: false,
|
|
||||||
target: {
|
|
||||||
realm: params.source.realm,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
debugError(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
get #connection(): Connection {
|
get #connection(): Connection {
|
||||||
return this.#frame.page().browser().connection;
|
return this.#frame.page().browser().connection;
|
||||||
}
|
}
|
||||||
|
|
||||||
get #channelArguments() {
|
#handleMessage = async (params: Bidi.Script.MessageParameters) => {
|
||||||
return [
|
if (params.channel !== this.#channel) {
|
||||||
{
|
return;
|
||||||
type: 'channel' as const,
|
}
|
||||||
value: {
|
const realm = this.#getRealm(params.source);
|
||||||
channel: this.#channels.args,
|
if (!realm) {
|
||||||
ownership: Bidi.Script.ResultOwnership.Root,
|
// Unrelated message.
|
||||||
},
|
return;
|
||||||
},
|
}
|
||||||
{
|
|
||||||
type: 'channel' as const,
|
using dataHandle = BidiJSHandle.from<
|
||||||
value: {
|
[
|
||||||
channel: this.#channels.resolve,
|
resolve: (ret: FlattenHandle<Awaited<Ret>>) => void,
|
||||||
ownership: Bidi.Script.ResultOwnership.Root,
|
reject: (error: unknown) => void,
|
||||||
},
|
args: Args,
|
||||||
},
|
]
|
||||||
{
|
>(params.data, realm);
|
||||||
type: 'channel' as const,
|
|
||||||
value: {
|
using argsHandle = await dataHandle.evaluateHandle(([, , args]) => {
|
||||||
channel: this.#channels.reject,
|
return args;
|
||||||
ownership: Bidi.Script.ResultOwnership.Root,
|
});
|
||||||
},
|
|
||||||
},
|
using stack = new DisposableStack();
|
||||||
];
|
const args = [];
|
||||||
|
for (const [index, handle] of await argsHandle.getProperties()) {
|
||||||
|
stack.use(handle);
|
||||||
|
|
||||||
|
// Element handles are passed as is.
|
||||||
|
if (handle instanceof BidiElementHandle) {
|
||||||
|
args[+index] = handle;
|
||||||
|
stack.use(handle);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Everything else is passed as the JS value.
|
||||||
|
args[+index] = handle.jsonValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
let result;
|
||||||
|
try {
|
||||||
|
result = await this.#apply(...((await Promise.all(args)) as Args));
|
||||||
|
} catch (error) {
|
||||||
|
try {
|
||||||
|
if (error instanceof Error) {
|
||||||
|
await dataHandle.evaluate(
|
||||||
|
([, reject], name, message, stack) => {
|
||||||
|
const error = new Error(message);
|
||||||
|
error.name = name;
|
||||||
|
if (stack) {
|
||||||
|
error.stack = stack;
|
||||||
|
}
|
||||||
|
reject(error);
|
||||||
|
},
|
||||||
|
error.name,
|
||||||
|
error.message,
|
||||||
|
error.stack
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
await dataHandle.evaluate(([, reject], error) => {
|
||||||
|
reject(error);
|
||||||
|
}, error);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
debugError(error);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await dataHandle.evaluate(([resolve], result) => {
|
||||||
|
resolve(result);
|
||||||
|
}, result);
|
||||||
|
} catch (error) {
|
||||||
|
debugError(error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#getRealm(source: Bidi.Script.Source) {
|
||||||
|
const frame = this.#findFrame(source.context as string);
|
||||||
|
if (!frame) {
|
||||||
|
// Unrelated message.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
return frame.realm(source.realm);
|
||||||
}
|
}
|
||||||
|
|
||||||
#handleResolveMessage = (params: Bidi.Script.MessageParameters) => {
|
#findFrame(id: string) {
|
||||||
if (params.channel !== this.#channels.resolve) {
|
const frames = [this.#frame];
|
||||||
return;
|
for (const frame of frames) {
|
||||||
|
if (frame._id === id) {
|
||||||
|
return frame;
|
||||||
|
}
|
||||||
|
frames.push(...frame.childFrames());
|
||||||
}
|
}
|
||||||
const {callbacks, remoteValue} = this.#getCallbacksAndRemoteValue(params);
|
return;
|
||||||
callbacks.resolve.resolve(remoteValue);
|
|
||||||
};
|
|
||||||
|
|
||||||
#handleRejectMessage = (params: Bidi.Script.MessageParameters) => {
|
|
||||||
if (params.channel !== this.#channels.reject) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const {callbacks, remoteValue} = this.#getCallbacksAndRemoteValue(params);
|
|
||||||
callbacks.reject.resolve(remoteValue);
|
|
||||||
};
|
|
||||||
|
|
||||||
#getCallbacksAndRemoteValue(params: Bidi.Script.MessageParameters) {
|
|
||||||
const {data, source} = params;
|
|
||||||
assert(data.type === 'array');
|
|
||||||
assert(data.value);
|
|
||||||
|
|
||||||
const callerIdRemote = data.value[0];
|
|
||||||
assert(callerIdRemote);
|
|
||||||
assert(callerIdRemote.type === 'number');
|
|
||||||
assert(typeof callerIdRemote.value === 'number');
|
|
||||||
|
|
||||||
let bindingMap = this.#callerInfos.get(source.realm);
|
|
||||||
if (!bindingMap) {
|
|
||||||
bindingMap = new Map();
|
|
||||||
this.#callerInfos.set(source.realm, bindingMap);
|
|
||||||
}
|
|
||||||
|
|
||||||
const callerId = callerIdRemote.value;
|
|
||||||
let callbacks = bindingMap.get(callerId);
|
|
||||||
if (!callbacks) {
|
|
||||||
callbacks = {
|
|
||||||
resolve: new Deferred(),
|
|
||||||
reject: new Deferred(),
|
|
||||||
};
|
|
||||||
bindingMap.set(callerId, callbacks);
|
|
||||||
}
|
|
||||||
return {callbacks, remoteValue: data};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Symbol.dispose](): void {
|
[Symbol.dispose](): void {
|
||||||
@ -286,10 +230,11 @@ export class ExposeableFunction<Args extends unknown[], Ret> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async [Symbol.asyncDispose](): Promise<void> {
|
async [Symbol.asyncDispose](): Promise<void> {
|
||||||
if (this.#preloadScriptId) {
|
this.#disposables.dispose();
|
||||||
await this.#connection.send('script.removePreloadScript', {
|
await Promise.all(
|
||||||
script: this.#preloadScriptId,
|
this.#scripts.map(async ([frame, script]) => {
|
||||||
});
|
await frame.browsingContext.removePreloadScript(script);
|
||||||
}
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -224,14 +224,23 @@ export class BidiFrame extends Frame {
|
|||||||
return this.page()._timeoutSettings;
|
return this.page()._timeoutSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
override mainRealm(): BidiRealm {
|
override mainRealm(): BidiFrameRealm {
|
||||||
return this.realms.default;
|
return this.realms.default;
|
||||||
}
|
}
|
||||||
|
|
||||||
override isolatedRealm(): BidiRealm {
|
override isolatedRealm(): BidiFrameRealm {
|
||||||
return this.realms.internal;
|
return this.realms.internal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
realm(id: string): BidiRealm | undefined {
|
||||||
|
for (const realm of Object.values(this.realms)) {
|
||||||
|
if (realm.realm.id === id) {
|
||||||
|
return realm;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
override page(): BidiPage {
|
override page(): BidiPage {
|
||||||
let parent = this.#parent;
|
let parent = this.#parent;
|
||||||
while (parent instanceof BidiFrame) {
|
while (parent instanceof BidiFrame) {
|
||||||
|
@ -2871,12 +2871,6 @@
|
|||||||
"parameters": ["cdp", "firefox"],
|
"parameters": ["cdp", "firefox"],
|
||||||
"expectations": ["SKIP"]
|
"expectations": ["SKIP"]
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"testIdPattern": "[page.spec] Page Page.exposeFunction should be callable from-inside evaluateOnNewDocument",
|
|
||||||
"platforms": ["darwin", "linux", "win32"],
|
|
||||||
"parameters": ["chrome", "webDriverBiDi"],
|
|
||||||
"expectations": ["FAIL"]
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"testIdPattern": "[page.spec] Page Page.metrics metrics event fired on console.timeStamp",
|
"testIdPattern": "[page.spec] Page Page.metrics metrics event fired on console.timeStamp",
|
||||||
"platforms": ["darwin", "linux", "win32"],
|
"platforms": ["darwin", "linux", "win32"],
|
||||||
|
@ -15,6 +15,7 @@ import type {HTTPRequest} from 'puppeteer-core/internal/api/HTTPRequest.js';
|
|||||||
import type {Metrics, Page} from 'puppeteer-core/internal/api/Page.js';
|
import type {Metrics, Page} from 'puppeteer-core/internal/api/Page.js';
|
||||||
import type {CdpPage} from 'puppeteer-core/internal/cdp/Page.js';
|
import type {CdpPage} from 'puppeteer-core/internal/cdp/Page.js';
|
||||||
import type {ConsoleMessage} from 'puppeteer-core/internal/common/ConsoleMessage.js';
|
import type {ConsoleMessage} from 'puppeteer-core/internal/common/ConsoleMessage.js';
|
||||||
|
import {Deferred} from 'puppeteer-core/internal/util/Deferred.js';
|
||||||
import sinon from 'sinon';
|
import sinon from 'sinon';
|
||||||
|
|
||||||
import {getTestState, setupTestBrowserHooks} from './mocha-utils.js';
|
import {getTestState, setupTestBrowserHooks} from './mocha-utils.js';
|
||||||
@ -1030,15 +1031,15 @@ describe('Page', function () {
|
|||||||
it('should be callable from-inside evaluateOnNewDocument', async () => {
|
it('should be callable from-inside evaluateOnNewDocument', async () => {
|
||||||
const {page} = await getTestState();
|
const {page} = await getTestState();
|
||||||
|
|
||||||
let called = false;
|
const called = new Deferred<void>();
|
||||||
await page.exposeFunction('woof', function () {
|
await page.exposeFunction('woof', function () {
|
||||||
called = true;
|
called.resolve();
|
||||||
});
|
});
|
||||||
await page.evaluateOnNewDocument(() => {
|
await page.evaluateOnNewDocument(() => {
|
||||||
return (globalThis as any).woof();
|
return (globalThis as any).woof();
|
||||||
});
|
});
|
||||||
await page.reload();
|
await page.reload();
|
||||||
expect(called).toBe(true);
|
await called.valueOrThrow();
|
||||||
});
|
});
|
||||||
it('should survive navigation', async () => {
|
it('should survive navigation', async () => {
|
||||||
const {page, server} = await getTestState();
|
const {page, server} = await getTestState();
|
||||||
|
Loading…
Reference in New Issue
Block a user