import * as Bidi from 'chromium-bidi/lib/cjs/protocol/protocol.js'; import ProtocolMapping from 'devtools-protocol/types/protocol-mapping.js'; import {assert} from '../../util/assert.js'; import {stringifyFunction} from '../../util/Function.js'; import {ProtocolError, TimeoutError} from '../Errors.js'; import {EventEmitter} from '../EventEmitter.js'; import {PuppeteerLifeCycleEvent} from '../LifecycleWatcher.js'; import {TimeoutSettings} from '../TimeoutSettings.js'; import {EvaluateFunc, HandleFor} from '../types.js'; import { PuppeteerURL, getSourcePuppeteerURLIfAvailable, isString, setPageContent, waitWithTimeout, } from '../util.js'; import {Connection} from './Connection.js'; import {ElementHandle} from './ElementHandle.js'; import {JSHandle} from './JSHandle.js'; import {BidiSerializer} from './Serializer.js'; import {createEvaluationError} from './utils.js'; const SOURCE_URL_REGEX = /^[\040\t]*\/\/[@#] sourceURL=\s*(\S*?)\s*$/m; const getSourceUrlComment = (url: string) => { return `//# sourceURL=${url}`; }; /** * @internal */ const lifeCycleToSubscribedEvent = new Map([ ['load', 'browsingContext.load'], ['domcontentloaded', 'browsingContext.domContentLoaded'], ]); /** * @internal */ const lifeCycleToReadinessState = new Map< PuppeteerLifeCycleEvent, Bidi.BrowsingContext.ReadinessState >([ ['load', 'complete'], ['domcontentloaded', 'interactive'], ]); /** * @internal */ export class BrowsingContext extends EventEmitter { connection: Connection; #timeoutSettings: TimeoutSettings; #id: string; #url = 'about:blank'; constructor( connection: Connection, timeoutSettings: TimeoutSettings, info: Bidi.BrowsingContext.Info ) { super(); this.connection = connection; this.#timeoutSettings = timeoutSettings; this.#id = info.context; } get url(): string { return this.#url; } get id(): string { return this.#id; } async goto( url: string, options: { referer?: string; referrerPolicy?: string; timeout?: number; waitUntil?: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[]; } = {} ): Promise { const { waitUntil = 'load', timeout = this.#timeoutSettings.navigationTimeout(), } = options; const readinessState = lifeCycleToReadinessState.get( getWaitUntilSingle(waitUntil) ) as Bidi.BrowsingContext.ReadinessState; try { const {result} = await waitWithTimeout( this.connection.send('browsingContext.navigate', { url: url, context: this.#id, wait: readinessState, }), 'Navigation', timeout ); this.#url = result.url; return result.navigation; } catch (error) { if (error instanceof ProtocolError) { error.message += ` at ${url}`; } else if (error instanceof TimeoutError) { error.message = 'Navigation timeout of ' + timeout + ' ms exceeded'; } throw error; } } async evaluateHandle< Params extends unknown[], Func extends EvaluateFunc = EvaluateFunc >( pageFunction: Func | string, ...args: Params ): Promise>>> { return this.#evaluate(false, pageFunction, ...args); } async evaluate< Params extends unknown[], Func extends EvaluateFunc = EvaluateFunc >( pageFunction: Func | string, ...args: Params ): Promise>> { return this.#evaluate(true, pageFunction, ...args); } async #evaluate< Params extends unknown[], Func extends EvaluateFunc = EvaluateFunc >( returnByValue: true, pageFunction: Func | string, ...args: Params ): Promise>>; async #evaluate< Params extends unknown[], Func extends EvaluateFunc = EvaluateFunc >( returnByValue: false, pageFunction: Func | string, ...args: Params ): Promise>>>; async #evaluate< Params extends unknown[], Func extends EvaluateFunc = EvaluateFunc >( returnByValue: boolean, pageFunction: Func | string, ...args: Params ): Promise>> | Awaited>> { const sourceUrlComment = getSourceUrlComment( getSourcePuppeteerURLIfAvailable(pageFunction)?.toString() ?? PuppeteerURL.INTERNAL_URL ); let responsePromise; const resultOwnership = returnByValue ? 'none' : 'root'; if (isString(pageFunction)) { const expression = SOURCE_URL_REGEX.test(pageFunction) ? pageFunction : `${pageFunction}\n${sourceUrlComment}\n`; responsePromise = this.connection.send('script.evaluate', { expression, target: {context: this.#id}, resultOwnership, awaitPromise: true, }); } else { let functionDeclaration = stringifyFunction(pageFunction); functionDeclaration = SOURCE_URL_REGEX.test(functionDeclaration) ? functionDeclaration : `${functionDeclaration}\n${sourceUrlComment}\n`; responsePromise = this.connection.send('script.callFunction', { functionDeclaration, arguments: await Promise.all( args.map(arg => { return BidiSerializer.serialize(arg, this); }) ), target: {context: this.#id}, resultOwnership, awaitPromise: true, }); } const {result} = await responsePromise; if ('type' in result && result.type === 'exception') { throw createEvaluationError(result.exceptionDetails); } return returnByValue ? BidiSerializer.deserialize(result.result) : getBidiHandle(this, result.result); } async setContent( html: string, options: { timeout?: number; waitUntil?: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[]; } ): Promise { const { waitUntil = 'load', timeout = this.#timeoutSettings.navigationTimeout(), } = options; const waitUntilCommand = lifeCycleToSubscribedEvent.get( getWaitUntilSingle(waitUntil) ) as string; await Promise.all([ setPageContent(this, html), waitWithTimeout( new Promise(resolve => { this.once(waitUntilCommand, () => { resolve(); }); }), waitUntilCommand, timeout ), ]); } async content(): Promise { return await this.evaluate(() => { let retVal = ''; if (document.doctype) { retVal = new XMLSerializer().serializeToString(document.doctype); } if (document.documentElement) { retVal += document.documentElement.outerHTML; } return retVal; }); } async sendCDPCommand( method: keyof ProtocolMapping.Commands, params: object = {} ): Promise { const session = await this.connection.send('cdp.getSession', { context: this.#id, }); // TODO: remove any once chromium-bidi types are updated. const sessionId = (session.result as any).cdpSession; return await this.connection.send('cdp.sendCommand', { cdpMethod: method, cdpParams: params, cdpSession: sessionId, }); } dispose(): void { this.removeAllListeners(); this.connection.unregisterBrowsingContexts(this.#id); } } /** * @internal */ export function getBidiHandle( context: BrowsingContext, result: Bidi.CommonDataTypes.RemoteValue ): JSHandle | ElementHandle { if (result.type === 'node' || result.type === 'window') { return new ElementHandle(context, result); } return new JSHandle(context, result); } /** * @internal */ export function getWaitUntilSingle( event: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[] ): Extract { if (Array.isArray(event) && event.length > 1) { throw new Error('BiDi support only single `waitUntil` argument'); } const waitUntilSingle = Array.isArray(event) ? (event.find(lifecycle => { return lifecycle === 'domcontentloaded' || lifecycle === 'load'; }) as PuppeteerLifeCycleEvent) : event; if ( waitUntilSingle === 'networkidle0' || waitUntilSingle === 'networkidle2' ) { throw new Error(`BiDi does not support 'waitUntil' ${waitUntilSingle}`); } assert(waitUntilSingle, `Invalid waitUntil option ${waitUntilSingle}`); return waitUntilSingle; }