puppeteer/packages/puppeteer-core/src/common/LifecycleWatcher.ts

308 lines
8.7 KiB
TypeScript
Raw Normal View History

/**
* Copyright 2019 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 {HTTPResponse} from '../api/HTTPResponse.js';
import {assert} from '../util/assert.js';
import {
DeferredPromise,
createDeferredPromise,
} from '../util/DeferredPromise.js';
import {CDPSessionEmittedEvents} from './Connection.js';
import {TimeoutError} from './Errors.js';
import {Frame} from './Frame.js';
import {FrameManager, FrameManagerEmittedEvents} from './FrameManager.js';
import {HTTPRequest} from './HTTPRequest.js';
import {NetworkManagerEmittedEvents} from './NetworkManager.js';
import {
addEventListener,
PuppeteerEventListener,
removeEventListeners,
} from './util.js';
/**
* @public
*/
2020-05-07 10:54:55 +00:00
export type PuppeteerLifeCycleEvent =
| 'load'
| 'domcontentloaded'
| 'networkidle0'
| 'networkidle2';
/**
* @public
*/
export type ProtocolLifeCycleEvent =
2020-05-07 10:54:55 +00:00
| 'load'
| 'DOMContentLoaded'
| 'networkIdle'
| 'networkAlmostIdle';
const puppeteerToProtocolLifecycle = new Map<
PuppeteerLifeCycleEvent,
ProtocolLifeCycleEvent
>([
['load', 'load'],
['domcontentloaded', 'DOMContentLoaded'],
['networkidle0', 'networkIdle'],
['networkidle2', 'networkAlmostIdle'],
]);
const noop = (): void => {};
/**
* @internal
*/
export class LifecycleWatcher {
2022-06-13 09:16:25 +00:00
#expectedLifecycle: ProtocolLifeCycleEvent[];
#frameManager: FrameManager;
#frame: Frame;
#timeout: number;
#navigationRequest: HTTPRequest | null = null;
#eventListeners: PuppeteerEventListener[];
#initialLoaderId: string;
#sameDocumentNavigationPromise = createDeferredPromise<Error | undefined>();
#lifecyclePromise = createDeferredPromise<void>();
#newDocumentNavigationPromise = createDeferredPromise<Error | undefined>();
#terminationPromise = createDeferredPromise<Error | undefined>();
2022-06-13 09:16:25 +00:00
#timeoutPromise: Promise<TimeoutError | undefined>;
2022-06-13 09:16:25 +00:00
#maximumTimer?: NodeJS.Timeout;
#hasSameDocumentNavigation?: boolean;
#swapped?: boolean;
#navigationResponseReceived?: DeferredPromise<void>;
2020-05-07 10:54:55 +00:00
constructor(
frameManager: FrameManager,
frame: Frame,
waitUntil: PuppeteerLifeCycleEvent | PuppeteerLifeCycleEvent[],
timeout: number
) {
2022-06-14 11:55:35 +00:00
if (Array.isArray(waitUntil)) {
waitUntil = waitUntil.slice();
} else if (typeof waitUntil === 'string') {
waitUntil = [waitUntil];
}
this.#initialLoaderId = frame._loaderId;
this.#expectedLifecycle = waitUntil.map(value => {
const protocolEvent = puppeteerToProtocolLifecycle.get(value);
assert(protocolEvent, 'Unknown value for options.waitUntil: ' + value);
return protocolEvent as ProtocolLifeCycleEvent;
});
2022-06-13 09:16:25 +00:00
this.#frameManager = frameManager;
this.#frame = frame;
this.#timeout = timeout;
this.#eventListeners = [
addEventListener(
frameManager.client,
CDPSessionEmittedEvents.Disconnected,
2022-06-13 09:16:25 +00:00
this.#terminate.bind(
2022-06-09 17:00:50 +00:00
this,
new Error('Navigation failed because browser has disconnected!')
)
2020-05-07 10:54:55 +00:00
),
addEventListener(
2022-06-13 09:16:25 +00:00
this.#frameManager,
FrameManagerEmittedEvents.LifecycleEvent,
2022-06-13 09:16:25 +00:00
this.#checkLifecycleComplete.bind(this)
2020-05-07 10:54:55 +00:00
),
addEventListener(
2022-06-13 09:16:25 +00:00
this.#frameManager,
FrameManagerEmittedEvents.FrameNavigatedWithinDocument,
2022-06-13 09:16:25 +00:00
this.#navigatedWithinDocument.bind(this)
2020-05-07 10:54:55 +00:00
),
addEventListener(
2022-06-13 09:16:25 +00:00
this.#frameManager,
FrameManagerEmittedEvents.FrameNavigated,
2022-06-13 09:16:25 +00:00
this.#navigated.bind(this)
),
addEventListener(
2022-06-13 09:16:25 +00:00
this.#frameManager,
FrameManagerEmittedEvents.FrameSwapped,
2022-06-13 09:16:25 +00:00
this.#frameSwapped.bind(this)
),
addEventListener(
2022-06-13 09:16:25 +00:00
this.#frameManager,
FrameManagerEmittedEvents.FrameDetached,
2022-06-13 09:16:25 +00:00
this.#onFrameDetached.bind(this)
2020-05-07 10:54:55 +00:00
),
addEventListener(
this.#frameManager.networkManager,
NetworkManagerEmittedEvents.Request,
2022-06-13 09:16:25 +00:00
this.#onRequest.bind(this)
2020-05-07 10:54:55 +00:00
),
addEventListener(
this.#frameManager.networkManager,
NetworkManagerEmittedEvents.Response,
this.#onResponse.bind(this)
),
addEventListener(
this.#frameManager.networkManager,
NetworkManagerEmittedEvents.RequestFailed,
this.#onRequestFailed.bind(this)
),
];
2022-06-13 09:16:25 +00:00
this.#timeoutPromise = this.#createTimeoutPromise();
this.#checkLifecycleComplete();
}
2022-06-13 09:16:25 +00:00
#onRequest(request: HTTPRequest): void {
2022-06-14 11:55:35 +00:00
if (request.frame() !== this.#frame || !request.isNavigationRequest()) {
return;
2022-06-14 11:55:35 +00:00
}
2022-06-13 09:16:25 +00:00
this.#navigationRequest = request;
// Resolve previous navigation response in case there are multiple
// navigation requests reported by the backend. This generally should not
// happen by it looks like it's possible.
this.#navigationResponseReceived?.resolve();
this.#navigationResponseReceived = createDeferredPromise();
if (request.response() !== null) {
this.#navigationResponseReceived?.resolve();
}
}
#onRequestFailed(request: HTTPRequest): void {
if (this.#navigationRequest?._requestId !== request._requestId) {
return;
}
this.#navigationResponseReceived?.resolve();
}
#onResponse(response: HTTPResponse): void {
if (this.#navigationRequest?._requestId !== response.request()._requestId) {
return;
}
this.#navigationResponseReceived?.resolve();
}
2022-06-13 09:16:25 +00:00
#onFrameDetached(frame: Frame): void {
if (this.#frame === frame) {
this.#terminationPromise.resolve(
2020-05-07 10:54:55 +00:00
new Error('Navigating frame was detached')
);
return;
}
2022-06-13 09:16:25 +00:00
this.#checkLifecycleComplete();
}
async navigationResponse(): Promise<HTTPResponse | null> {
// Continue with a possibly null response.
await this.#navigationResponseReceived?.valueOrThrow();
2022-06-13 09:16:25 +00:00
return this.#navigationRequest ? this.#navigationRequest.response() : null;
}
2022-06-13 09:16:25 +00:00
#terminate(error: Error): void {
this.#terminationPromise.resolve(error);
}
sameDocumentNavigationPromise(): Promise<Error | undefined> {
return this.#sameDocumentNavigationPromise.valueOrThrow();
}
newDocumentNavigationPromise(): Promise<Error | undefined> {
return this.#newDocumentNavigationPromise.valueOrThrow();
}
lifecyclePromise(): Promise<void> {
return this.#lifecyclePromise.valueOrThrow();
}
timeoutOrTerminationPromise(): Promise<Error | TimeoutError | undefined> {
return Promise.race([
this.#timeoutPromise,
this.#terminationPromise.valueOrThrow(),
]);
}
2022-06-13 09:16:25 +00:00
async #createTimeoutPromise(): Promise<TimeoutError | undefined> {
2022-06-14 11:55:35 +00:00
if (!this.#timeout) {
return new Promise(noop);
}
2020-05-07 10:54:55 +00:00
const errorMessage =
2022-06-13 09:16:25 +00:00
'Navigation timeout of ' + this.#timeout + ' ms exceeded';
await new Promise(fulfill => {
return (this.#maximumTimer = setTimeout(fulfill, this.#timeout));
});
2022-06-13 09:16:25 +00:00
return new TimeoutError(errorMessage);
}
2022-06-13 09:16:25 +00:00
#navigatedWithinDocument(frame: Frame): void {
2022-06-14 11:55:35 +00:00
if (frame !== this.#frame) {
return;
}
2022-06-13 09:16:25 +00:00
this.#hasSameDocumentNavigation = true;
this.#checkLifecycleComplete();
}
2022-06-13 09:16:25 +00:00
#navigated(frame: Frame): void {
2022-06-14 11:55:35 +00:00
if (frame !== this.#frame) {
return;
}
2022-06-13 09:16:25 +00:00
this.#checkLifecycleComplete();
}
2022-06-13 09:16:25 +00:00
#frameSwapped(frame: Frame): void {
2022-06-14 11:55:35 +00:00
if (frame !== this.#frame) {
return;
}
2022-06-13 09:16:25 +00:00
this.#swapped = true;
this.#checkLifecycleComplete();
}
2022-06-13 09:16:25 +00:00
#checkLifecycleComplete(): void {
// We expect navigation to commit.
2022-06-14 11:55:35 +00:00
if (!checkLifecycle(this.#frame, this.#expectedLifecycle)) {
return;
}
this.#lifecyclePromise.resolve();
2022-06-14 11:55:35 +00:00
if (this.#hasSameDocumentNavigation) {
this.#sameDocumentNavigationPromise.resolve(undefined);
2022-06-14 11:55:35 +00:00
}
if (this.#swapped || this.#frame._loaderId !== this.#initialLoaderId) {
this.#newDocumentNavigationPromise.resolve(undefined);
2022-06-14 11:55:35 +00:00
}
2020-05-07 10:54:55 +00:00
function checkLifecycle(
frame: Frame,
expectedLifecycle: ProtocolLifeCycleEvent[]
): boolean {
for (const event of expectedLifecycle) {
2022-06-14 11:55:35 +00:00
if (!frame._lifecycleEvents.has(event)) {
return false;
}
}
for (const child of frame.childFrames()) {
if (
child._hasStartedLoading &&
!checkLifecycle(child, expectedLifecycle)
) {
return false;
}
}
return true;
}
}
dispose(): void {
removeEventListeners(this.#eventListeners);
2022-06-13 09:16:25 +00:00
this.#maximumTimer !== undefined && clearTimeout(this.#maximumTimer);
}
}