2022-09-16 05:35:51 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2017 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.
|
|
|
|
*/
|
|
|
|
|
2022-08-17 15:45:45 +00:00
|
|
|
import {Protocol} from 'devtools-protocol';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
import {Frame, throwIfDetached} from '../api/Frame.js';
|
2023-03-15 16:51:34 +00:00
|
|
|
import {HTTPResponse} from '../api/HTTPResponse.js';
|
2023-03-21 09:21:10 +00:00
|
|
|
import {Page, WaitTimeoutOptions} from '../api/Page.js';
|
|
|
|
import {assert} from '../util/assert.js';
|
2023-05-31 21:36:19 +00:00
|
|
|
import {Deferred} from '../util/Deferred.js';
|
2022-08-17 15:45:45 +00:00
|
|
|
import {isErrorLike} from '../util/ErrorLike.js';
|
2023-02-15 23:09:31 +00:00
|
|
|
|
2022-08-17 15:45:45 +00:00
|
|
|
import {CDPSession} from './Connection.js';
|
2023-03-21 09:21:10 +00:00
|
|
|
import {
|
|
|
|
DeviceRequestPrompt,
|
|
|
|
DeviceRequestPromptManager,
|
|
|
|
} from './DeviceRequestPrompt.js';
|
2022-08-17 15:45:45 +00:00
|
|
|
import {ExecutionContext} from './ExecutionContext.js';
|
|
|
|
import {FrameManager} from './FrameManager.js';
|
2023-07-18 16:28:03 +00:00
|
|
|
import {IsolatedWorld} from './IsolatedWorld.js';
|
2022-11-10 16:11:18 +00:00
|
|
|
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
2022-08-17 15:45:45 +00:00
|
|
|
import {LifecycleWatcher, PuppeteerLifeCycleEvent} from './LifecycleWatcher.js';
|
|
|
|
|
2023-08-08 14:42:45 +00:00
|
|
|
/**
|
|
|
|
* We use symbols to prevent external parties listening to these events.
|
|
|
|
* They are internal to Puppeteer.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export const FrameEmittedEvents = {
|
|
|
|
FrameNavigated: Symbol('Frame.FrameNavigated'),
|
|
|
|
FrameSwapped: Symbol('Frame.FrameSwapped'),
|
|
|
|
LifecycleEvent: Symbol('Frame.LifecycleEvent'),
|
|
|
|
FrameNavigatedWithinDocument: Symbol('Frame.FrameNavigatedWithinDocument'),
|
|
|
|
FrameDetached: Symbol('Frame.FrameDetached'),
|
2023-08-28 06:20:57 +00:00
|
|
|
FrameSwappedByActivation: Symbol('Frame.FrameSwappedByActivation'),
|
2023-08-08 14:42:45 +00:00
|
|
|
};
|
|
|
|
|
2022-08-17 15:45:45 +00:00
|
|
|
/**
|
2023-05-15 14:39:47 +00:00
|
|
|
* @internal
|
2022-08-17 15:45:45 +00:00
|
|
|
*/
|
2023-08-30 12:20:25 +00:00
|
|
|
export class CDPFrame extends Frame {
|
2022-08-17 15:45:45 +00:00
|
|
|
#url = '';
|
|
|
|
#detached = false;
|
|
|
|
#client!: CDPSession;
|
|
|
|
|
|
|
|
_frameManager: FrameManager;
|
2023-05-15 14:39:47 +00:00
|
|
|
override _id: string;
|
2022-08-17 15:45:45 +00:00
|
|
|
_loaderId = '';
|
|
|
|
_lifecycleEvents = new Set<string>();
|
2023-05-15 14:39:47 +00:00
|
|
|
override _parentId?: string;
|
2022-08-17 15:45:45 +00:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
frameManager: FrameManager,
|
|
|
|
frameId: string,
|
2022-09-16 05:35:51 +00:00
|
|
|
parentFrameId: string | undefined,
|
2022-08-17 15:45:45 +00:00
|
|
|
client: CDPSession
|
|
|
|
) {
|
2023-05-15 14:39:47 +00:00
|
|
|
super();
|
2022-08-17 15:45:45 +00:00
|
|
|
this._frameManager = frameManager;
|
|
|
|
this.#url = '';
|
|
|
|
this._id = frameId;
|
2022-09-16 05:35:51 +00:00
|
|
|
this._parentId = parentFrameId;
|
2022-08-17 15:45:45 +00:00
|
|
|
this.#detached = false;
|
|
|
|
|
|
|
|
this._loaderId = '';
|
|
|
|
|
|
|
|
this.updateClient(client);
|
2023-08-28 06:20:57 +00:00
|
|
|
|
|
|
|
this.on(FrameEmittedEvents.FrameSwappedByActivation, () => {
|
|
|
|
// Emulate loading process for swapped frames.
|
|
|
|
this._onLoadingStarted();
|
|
|
|
this._onLoadingStopped();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the frame ID with the new ID. This happens when the main frame is
|
|
|
|
* replaced by a different frame.
|
|
|
|
*/
|
|
|
|
updateId(id: string): void {
|
|
|
|
this._id = id;
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
|
2023-08-28 06:20:57 +00:00
|
|
|
updateClient(client: CDPSession, keepWorlds = false): void {
|
2022-08-17 15:45:45 +00:00
|
|
|
this.#client = client;
|
2023-08-28 06:20:57 +00:00
|
|
|
if (!keepWorlds) {
|
|
|
|
this.worlds = {
|
|
|
|
[MAIN_WORLD]: new IsolatedWorld(this),
|
|
|
|
[PUPPETEER_WORLD]: new IsolatedWorld(this),
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
this.worlds[MAIN_WORLD].frameUpdated();
|
|
|
|
this.worlds[PUPPETEER_WORLD].frameUpdated();
|
|
|
|
}
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
|
2023-05-15 14:39:47 +00:00
|
|
|
override page(): Page {
|
2022-08-17 15:45:45 +00:00
|
|
|
return this._frameManager.page();
|
|
|
|
}
|
|
|
|
|
2023-05-15 14:39:47 +00:00
|
|
|
override isOOPFrame(): boolean {
|
2022-08-17 15:45:45 +00:00
|
|
|
return this.#client !== this._frameManager.client;
|
|
|
|
}
|
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
@throwIfDetached
|
2023-05-15 14:39:47 +00:00
|
|
|
override async goto(
|
2022-08-17 15:45:45 +00:00
|
|
|
url: string,
|
|
|
|
options: {
|
|
|
|
referer?: string;
|
2023-01-23 11:11:20 +00:00
|
|
|
referrerPolicy?: string;
|
2022-08-17 15:45:45 +00:00
|
|
|
timeout?: number;
|
|
|
|
waitUntil?: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[];
|
|
|
|
} = {}
|
|
|
|
): Promise<HTTPResponse | null> {
|
|
|
|
const {
|
|
|
|
referer = this._frameManager.networkManager.extraHTTPHeaders()['referer'],
|
2023-01-23 11:11:20 +00:00
|
|
|
referrerPolicy = this._frameManager.networkManager.extraHTTPHeaders()[
|
|
|
|
'referer-policy'
|
|
|
|
],
|
2022-08-17 15:45:45 +00:00
|
|
|
waitUntil = ['load'],
|
|
|
|
timeout = this._frameManager.timeoutSettings.navigationTimeout(),
|
|
|
|
} = options;
|
|
|
|
|
|
|
|
let ensureNewDocumentNavigation = false;
|
|
|
|
const watcher = new LifecycleWatcher(
|
2023-08-08 14:42:45 +00:00
|
|
|
this._frameManager.networkManager,
|
2022-08-17 15:45:45 +00:00
|
|
|
this,
|
|
|
|
waitUntil,
|
|
|
|
timeout
|
|
|
|
);
|
2023-05-31 21:36:19 +00:00
|
|
|
let error = await Deferred.race([
|
2023-01-23 11:11:20 +00:00
|
|
|
navigate(
|
|
|
|
this.#client,
|
|
|
|
url,
|
|
|
|
referer,
|
|
|
|
referrerPolicy as Protocol.Page.ReferrerPolicy,
|
|
|
|
this._id
|
|
|
|
),
|
2023-07-03 07:30:30 +00:00
|
|
|
watcher.terminationPromise(),
|
2022-08-17 15:45:45 +00:00
|
|
|
]);
|
|
|
|
if (!error) {
|
2023-05-31 21:36:19 +00:00
|
|
|
error = await Deferred.race([
|
2023-07-03 07:30:30 +00:00
|
|
|
watcher.terminationPromise(),
|
2022-08-17 15:45:45 +00:00
|
|
|
ensureNewDocumentNavigation
|
|
|
|
? watcher.newDocumentNavigationPromise()
|
|
|
|
: watcher.sameDocumentNavigationPromise(),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (error) {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return await watcher.navigationResponse();
|
|
|
|
} finally {
|
|
|
|
watcher.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
async function navigate(
|
|
|
|
client: CDPSession,
|
|
|
|
url: string,
|
|
|
|
referrer: string | undefined,
|
2023-01-23 11:11:20 +00:00
|
|
|
referrerPolicy: Protocol.Page.ReferrerPolicy | undefined,
|
2022-08-17 15:45:45 +00:00
|
|
|
frameId: string
|
|
|
|
): Promise<Error | null> {
|
|
|
|
try {
|
|
|
|
const response = await client.send('Page.navigate', {
|
|
|
|
url,
|
|
|
|
referrer,
|
|
|
|
frameId,
|
2023-01-23 11:11:20 +00:00
|
|
|
referrerPolicy,
|
2022-08-17 15:45:45 +00:00
|
|
|
});
|
|
|
|
ensureNewDocumentNavigation = !!response.loaderId;
|
2023-02-06 09:40:51 +00:00
|
|
|
if (response.errorText === 'net::ERR_HTTP_RESPONSE_CODE_FAILURE') {
|
|
|
|
return null;
|
|
|
|
}
|
2022-08-17 15:45:45 +00:00
|
|
|
return response.errorText
|
|
|
|
? new Error(`${response.errorText} at ${url}`)
|
|
|
|
: null;
|
|
|
|
} catch (error) {
|
|
|
|
if (isErrorLike(error)) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
@throwIfDetached
|
2023-05-15 14:39:47 +00:00
|
|
|
override async waitForNavigation(
|
2022-08-17 15:45:45 +00:00
|
|
|
options: {
|
|
|
|
timeout?: number;
|
|
|
|
waitUntil?: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[];
|
|
|
|
} = {}
|
|
|
|
): Promise<HTTPResponse | null> {
|
|
|
|
const {
|
|
|
|
waitUntil = ['load'],
|
|
|
|
timeout = this._frameManager.timeoutSettings.navigationTimeout(),
|
|
|
|
} = options;
|
|
|
|
const watcher = new LifecycleWatcher(
|
2023-08-08 14:42:45 +00:00
|
|
|
this._frameManager.networkManager,
|
2022-08-17 15:45:45 +00:00
|
|
|
this,
|
|
|
|
waitUntil,
|
|
|
|
timeout
|
|
|
|
);
|
2023-05-31 21:36:19 +00:00
|
|
|
const error = await Deferred.race([
|
2023-07-03 07:30:30 +00:00
|
|
|
watcher.terminationPromise(),
|
2022-08-17 15:45:45 +00:00
|
|
|
watcher.sameDocumentNavigationPromise(),
|
|
|
|
watcher.newDocumentNavigationPromise(),
|
|
|
|
]);
|
|
|
|
try {
|
|
|
|
if (error) {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return await watcher.navigationResponse();
|
|
|
|
} finally {
|
|
|
|
watcher.dispose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-15 14:39:47 +00:00
|
|
|
override _client(): CDPSession {
|
2022-08-17 15:45:45 +00:00
|
|
|
return this.#client;
|
|
|
|
}
|
|
|
|
|
2023-05-15 14:39:47 +00:00
|
|
|
override executionContext(): Promise<ExecutionContext> {
|
2022-08-17 15:45:45 +00:00
|
|
|
return this.worlds[MAIN_WORLD].executionContext();
|
|
|
|
}
|
|
|
|
|
2023-06-16 07:16:04 +00:00
|
|
|
override mainRealm(): IsolatedWorld {
|
|
|
|
return this.worlds[MAIN_WORLD];
|
|
|
|
}
|
|
|
|
|
|
|
|
override isolatedRealm(): IsolatedWorld {
|
|
|
|
return this.worlds[PUPPETEER_WORLD];
|
|
|
|
}
|
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
@throwIfDetached
|
2023-05-15 14:39:47 +00:00
|
|
|
override async setContent(
|
2022-08-17 15:45:45 +00:00
|
|
|
html: string,
|
|
|
|
options: {
|
|
|
|
timeout?: number;
|
|
|
|
waitUntil?: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[];
|
|
|
|
} = {}
|
|
|
|
): Promise<void> {
|
2023-09-01 07:49:33 +00:00
|
|
|
return await this.isolatedRealm().setContent(html, options);
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
|
2023-05-15 14:39:47 +00:00
|
|
|
override url(): string {
|
2022-08-17 15:45:45 +00:00
|
|
|
return this.#url;
|
|
|
|
}
|
|
|
|
|
2023-08-30 12:20:25 +00:00
|
|
|
override parentFrame(): CDPFrame | null {
|
2022-09-16 05:35:51 +00:00
|
|
|
return this._frameManager._frameTree.parentFrame(this._id) || null;
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 12:20:25 +00:00
|
|
|
override childFrames(): CDPFrame[] {
|
2022-09-16 05:35:51 +00:00
|
|
|
return this._frameManager._frameTree.childFrames(this._id);
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 11:09:27 +00:00
|
|
|
#deviceRequestPromptManager(): DeviceRequestPromptManager {
|
2023-03-21 09:21:10 +00:00
|
|
|
if (this.isOOPFrame()) {
|
|
|
|
return this._frameManager._deviceRequestPromptManager(this.#client);
|
|
|
|
}
|
|
|
|
const parentFrame = this.parentFrame();
|
|
|
|
assert(parentFrame !== null);
|
2023-08-30 11:09:27 +00:00
|
|
|
return parentFrame.#deviceRequestPromptManager();
|
2023-03-21 09:21:10 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
@throwIfDetached
|
2023-05-15 14:39:47 +00:00
|
|
|
override waitForDevicePrompt(
|
2023-03-21 09:21:10 +00:00
|
|
|
options: WaitTimeoutOptions = {}
|
|
|
|
): Promise<DeviceRequestPrompt> {
|
2023-08-30 11:09:27 +00:00
|
|
|
return this.#deviceRequestPromptManager().waitForDevicePrompt(options);
|
2023-03-21 09:21:10 +00:00
|
|
|
}
|
|
|
|
|
2022-08-17 15:45:45 +00:00
|
|
|
_navigated(framePayload: Protocol.Page.Frame): void {
|
|
|
|
this._name = framePayload.name;
|
|
|
|
this.#url = `${framePayload.url}${framePayload.urlFragment || ''}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
_navigatedWithinDocument(url: string): void {
|
|
|
|
this.#url = url;
|
|
|
|
}
|
|
|
|
|
|
|
|
_onLifecycleEvent(loaderId: string, name: string): void {
|
|
|
|
if (name === 'init') {
|
|
|
|
this._loaderId = loaderId;
|
|
|
|
this._lifecycleEvents.clear();
|
|
|
|
}
|
|
|
|
this._lifecycleEvents.add(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onLoadingStopped(): void {
|
|
|
|
this._lifecycleEvents.add('DOMContentLoaded');
|
|
|
|
this._lifecycleEvents.add('load');
|
|
|
|
}
|
|
|
|
|
|
|
|
_onLoadingStarted(): void {
|
|
|
|
this._hasStartedLoading = true;
|
|
|
|
}
|
|
|
|
|
2023-08-31 14:39:32 +00:00
|
|
|
override get detached(): boolean {
|
|
|
|
return this.#detached;
|
|
|
|
}
|
|
|
|
|
2023-08-30 11:09:27 +00:00
|
|
|
[Symbol.dispose](): void {
|
2023-08-31 14:39:32 +00:00
|
|
|
if (this.#detached) {
|
|
|
|
return;
|
|
|
|
}
|
2022-08-17 15:45:45 +00:00
|
|
|
this.#detached = true;
|
2023-08-30 10:24:38 +00:00
|
|
|
this.worlds[MAIN_WORLD][Symbol.dispose]();
|
|
|
|
this.worlds[PUPPETEER_WORLD][Symbol.dispose]();
|
2022-08-17 15:45:45 +00:00
|
|
|
}
|
|
|
|
}
|