diff --git a/packages/puppeteer-core/src/api/Page.ts b/packages/puppeteer-core/src/api/Page.ts index 18e88cd9214..9d0b30464eb 100644 --- a/packages/puppeteer-core/src/api/Page.ts +++ b/packages/puppeteer-core/src/api/Page.ts @@ -28,7 +28,11 @@ import { merge, Observable, raceWith, - timer, + delay, + filter, + of, + switchMap, + startWith, } from '../../third_party/rxjs/rxjs.js'; import type {HTTPRequest} from '../api/HTTPRequest.js'; import type {HTTPResponse} from '../api/HTTPResponse.js'; @@ -38,7 +42,7 @@ import type {ConsoleMessage} from '../common/ConsoleMessage.js'; import type {Coverage} from '../common/Coverage.js'; import {Device} from '../common/Device.js'; import {DeviceRequestPrompt} from '../common/DeviceRequestPrompt.js'; -import {TargetCloseError, TimeoutError} from '../common/Errors.js'; +import {TargetCloseError} from '../common/Errors.js'; import {EventEmitter, Handler} from '../common/EventEmitter.js'; import type {FileChooser} from '../common/FileChooser.js'; import type {WaitForSelectorOptions} from '../common/IsolatedWorld.js'; @@ -68,7 +72,7 @@ import { importFSPromises, isNumber, isString, - waitForEvent, + timeout, withSourcePuppeteerURLIfNone, } from '../common/util.js'; import type {WebWorker} from '../common/WebWorker.js'; @@ -1696,62 +1700,33 @@ export abstract class Page inFlightRequestsCount: () => number; }, idleTime: number, - timeout: number, + ms: number, closedDeferred: Deferred ): Promise { - const idleDeferred = Deferred.create(); - const abortDeferred = Deferred.create(); - - let idleTimer: NodeJS.Timeout | undefined; - const cleanup = () => { - clearTimeout(idleTimer); - abortDeferred.reject(new Error('abort')); - }; - - const evaluate = () => { - clearTimeout(idleTimer); - - if (networkManager.inFlightRequestsCount() === 0) { - idleTimer = setTimeout(() => { - return idleDeferred.resolve(); - }, idleTime); - } - }; - - const listenToEvent = (event: symbol) => { - return waitForEvent( - networkManager, - event, - () => { - evaluate(); - return false; - }, - timeout, - abortDeferred - ); - }; - - const eventPromises = [ - listenToEvent(NetworkManagerEmittedEvents.Request), - listenToEvent(NetworkManagerEmittedEvents.Response), - listenToEvent(NetworkManagerEmittedEvents.RequestFailed), - ]; - - evaluate(); - - // We don't want to reject the closed deferred when - // the race if finished so we pass the Promise instead - const closedPromise = closedDeferred.valueOrThrow(); - - await Deferred.race([idleDeferred, ...eventPromises, closedPromise]).then( - r => { - cleanup(); - return r; - }, - error => { - cleanup(); - throw error; - } + await firstValueFrom( + merge( + fromEvent( + networkManager, + NetworkManagerEmittedEvents.Request as unknown as string + ), + fromEvent( + networkManager, + NetworkManagerEmittedEvents.Response as unknown as string + ), + fromEvent( + networkManager, + NetworkManagerEmittedEvents.RequestFailed as unknown as string + ) + ).pipe( + startWith(null), + filter(() => { + return networkManager.inFlightRequestsCount() === 0; + }), + switchMap(v => { + return of(v).pipe(delay(idleTime)); + }), + raceWith(timeout(ms), from(closedDeferred.valueOrThrow())) + ) ); } @@ -1787,11 +1762,7 @@ export abstract class Page filterAsync(urlOrPredicate), first(), raceWith( - timer(ms === 0 ? Infinity : ms).pipe( - map(() => { - throw new TimeoutError(`Timed out after waiting ${ms}ms`); - }) - ), + timeout(ms), fromEvent(this, PageEmittedEvents.Close).pipe( map(() => { throw new TargetCloseError('Page closed.'); diff --git a/packages/puppeteer-core/src/api/locators/Locator.ts b/packages/puppeteer-core/src/api/locators/Locator.ts index ef0f2d347a9..c03d121e20d 100644 --- a/packages/puppeteer-core/src/api/locators/Locator.ts +++ b/packages/puppeteer-core/src/api/locators/Locator.ts @@ -36,12 +36,10 @@ import { raceWith, retry, tap, - timer, } from '../../../third_party/rxjs/rxjs.js'; -import {TimeoutError} from '../../common/Errors.js'; import {EventEmitter} from '../../common/EventEmitter.js'; import {HandleFor} from '../../common/types.js'; -import {debugError} from '../../common/util.js'; +import {debugError, timeout} from '../../common/util.js'; import {BoundingBox, ClickOptions, ElementHandle} from '../ElementHandle.js'; import { @@ -213,17 +211,7 @@ export abstract class Locator extends EventEmitter { ) ); } - if (this._timeout > 0) { - candidates.push( - timer(this._timeout).pipe( - map(() => { - throw new TimeoutError( - `Timed out after waiting ${this._timeout}ms` - ); - }) - ) - ); - } + candidates.push(timeout(this._timeout)); return pipe( retry({delay: RETRY_DELAY}), raceWith(...candidates) diff --git a/packages/puppeteer-core/src/common/util.ts b/packages/puppeteer-core/src/common/util.ts index a9d77e95f8a..67c29058c33 100644 --- a/packages/puppeteer-core/src/common/util.ts +++ b/packages/puppeteer-core/src/common/util.ts @@ -18,6 +18,7 @@ import type {Readable} from 'stream'; import type {Protocol} from 'devtools-protocol'; +import {map, NEVER, Observable, timer} from '../../third_party/rxjs/rxjs.js'; import type {ElementHandle} from '../api/ElementHandle.js'; import type {JSHandle} from '../api/JSHandle.js'; import {Page} from '../api/Page.js'; @@ -29,6 +30,7 @@ import {isErrorLike} from '../util/ErrorLike.js'; import type {CDPSession} from './Connection.js'; import {debug} from './Debug.js'; import {CDPElementHandle} from './ElementHandle.js'; +import {TimeoutError} from './Errors.js'; import type {CommonEventEmitter} from './EventEmitter.js'; import {IsolatedWorld} from './IsolatedWorld.js'; import {CDPJSHandle} from './JSHandle.js'; @@ -708,3 +710,16 @@ export class Mutex { resolve(); } } + +/** + * @internal + */ +export function timeout(ms: number): Observable { + return ms === 0 + ? NEVER + : timer(ms).pipe( + map(() => { + throw new TimeoutError(`Timed out after waiting ${ms}ms`); + }) + ); +} diff --git a/packages/puppeteer-core/third_party/rxjs/rxjs.ts b/packages/puppeteer-core/third_party/rxjs/rxjs.ts index f657023ecf8..a2ab61437d8 100644 --- a/packages/puppeteer-core/third_party/rxjs/rxjs.ts +++ b/packages/puppeteer-core/third_party/rxjs/rxjs.ts @@ -16,28 +16,33 @@ export { catchError, defaultIfEmpty, + defer, + delay, + EMPTY, filter, first, - ignoreElements, - map, - mergeMap, - raceWith, - retry, - tap, - throwIfEmpty, firstValueFrom, - defer, - EMPTY, from, fromEvent, - merge, - race, - timer, - OperatorFunction, identity, + ignoreElements, + map, + merge, + mergeMap, + NEVER, noop, - pipe, Observable, + of, + OperatorFunction, + pipe, + race, + raceWith, + retry, + startWith, + switchMap, + tap, + throwIfEmpty, + timer, } from 'rxjs'; import {mergeMap, from, filter, map, type Observable} from 'rxjs';