diff --git a/packages/puppeteer-core/src/common/bidi/Page.ts b/packages/puppeteer-core/src/common/bidi/Page.ts
index 8b09cf38793..8e635aee43f 100644
--- a/packages/puppeteer-core/src/common/bidi/Page.ts
+++ b/packages/puppeteer-core/src/common/bidi/Page.ts
@@ -25,7 +25,9 @@ import {
WaitForOptions,
} from '../../api/Page.js';
import {assert} from '../../util/assert.js';
+import {createDeferred} from '../../util/Deferred.js';
import {ConsoleMessage, ConsoleMessageLocation} from '../ConsoleMessage.js';
+import {TargetCloseError} from '../Errors.js';
import {Handler} from '../EventEmitter.js';
import {FrameManagerEmittedEvents} from '../FrameManager.js';
import {FrameTree} from '../FrameTree.js';
@@ -36,6 +38,8 @@ import {TimeoutSettings} from '../TimeoutSettings.js';
import {EvaluateFunc, HandleFor} from '../types.js';
import {
debugError,
+ isString,
+ waitForEvent,
waitWithTimeout,
withSourcePuppeteerURLIfNone,
} from '../util.js';
@@ -43,6 +47,7 @@ import {
import {BrowsingContext, getBidiHandle} from './BrowsingContext.js';
import {Connection} from './Connection.js';
import {Frame} from './Frame.js';
+import {HTTPRequest} from './HTTPRequest.js';
import {HTTPResponse} from './HTTPResponse.js';
import {NetworkManager} from './NetworkManager.js';
import {BidiSerializer} from './Serializer.js';
@@ -56,7 +61,7 @@ export class Page extends PageBase {
#frameTree = new FrameTree();
#networkManager: NetworkManager;
#viewport: Viewport | null = null;
- #closed = false;
+ #closedDeferred = createDeferred();
#subscribedEvents = new Map>([
['log.entryAdded', this.#onLogEntryAdded.bind(this)],
[
@@ -258,10 +263,10 @@ export class Page extends PageBase {
}
override async close(): Promise {
- if (this.#closed) {
+ if (this.#closedDeferred.finished()) {
return;
}
- this.#closed = true;
+ this.#closedDeferred.resolve(new TargetCloseError('Page closed!'));
this.removeAllListeners();
this.#networkManager.dispose();
@@ -445,6 +450,52 @@ export class Page extends PageBase {
return buffer;
}
+
+ override waitForRequest(
+ urlOrPredicate: string | ((req: HTTPRequest) => boolean | Promise),
+ options: {timeout?: number} = {}
+ ): Promise {
+ const {timeout = this.#timeoutSettings.timeout()} = options;
+ return waitForEvent(
+ this.#networkManager,
+ NetworkManagerEmittedEvents.Request,
+ async request => {
+ if (isString(urlOrPredicate)) {
+ return urlOrPredicate === request.url();
+ }
+ if (typeof urlOrPredicate === 'function') {
+ return !!(await urlOrPredicate(request));
+ }
+ return false;
+ },
+ timeout,
+ this.#closedDeferred.valueOrThrow()
+ );
+ }
+
+ override waitForResponse(
+ urlOrPredicate:
+ | string
+ | ((res: HTTPResponse) => boolean | Promise),
+ options: {timeout?: number} = {}
+ ): Promise {
+ const {timeout = this.#timeoutSettings.timeout()} = options;
+ return waitForEvent(
+ this.#networkManager,
+ NetworkManagerEmittedEvents.Response,
+ async response => {
+ if (isString(urlOrPredicate)) {
+ return urlOrPredicate === response.url();
+ }
+ if (typeof urlOrPredicate === 'function') {
+ return !!(await urlOrPredicate(response));
+ }
+ return false;
+ },
+ timeout,
+ this.#closedDeferred.valueOrThrow()
+ );
+ }
}
function isConsoleLogEntry(
diff --git a/test/TestExpectations.json b/test/TestExpectations.json
index c627d0bb21c..817aafeb0a0 100644
--- a/test/TestExpectations.json
+++ b/test/TestExpectations.json
@@ -129,7 +129,7 @@
"testIdPattern": "[network.spec] network Response.text *",
"platforms": ["darwin", "linux", "win32"],
"parameters": ["webDriverBiDi"],
- "expectations": ["FAIL"]
+ "expectations": ["FAIL", "TIMEOUT"]
},
{
"testIdPattern": "[NetworkManager.spec] NetworkManager *",
@@ -167,6 +167,18 @@
"parameters": ["webDriverBiDi"],
"expectations": ["PASS"]
},
+ {
+ "testIdPattern": "[page.spec] Page Page.waitForRequest *",
+ "platforms": ["darwin", "linux", "win32"],
+ "parameters": ["webDriverBiDi"],
+ "expectations": ["PASS"]
+ },
+ {
+ "testIdPattern": "[page.spec] Page Page.waitForResponse *",
+ "platforms": ["darwin", "linux", "win32"],
+ "parameters": ["webDriverBiDi"],
+ "expectations": ["PASS"]
+ },
{
"testIdPattern": "[queryhandler.spec] *",
"platforms": ["darwin", "linux", "win32"],