chore: remove src/TaskQueue (#5826)

* chore: Remove src/TaskQueue

The only place it's used is in `src/Page.ts` to have a chain of
screenshot promises. Rather than initialize a task queue in `Browser`
and pass it through a chain of constructors we instead move the class
into `src/Page` and define it inline.

In the future we might want to create a helpers folder to contain small
utilities like that (`src/Page.ts` is already far too large) but I'm
leaving that for a future PR.

`TaskQueue` isn't documented in `api.md` so I don't think this is a
breaking change.

I updated the type of `screenshot()` to return `Promise<string | Buffer
| void>` because if a promise rejects it's silently swallowed. I'd like
to change this behaviour but one step at a time. This type only had to
change as now we type the screenshot task queue correctly rather than
using `any` which then exposed the incorrect `screenshot()` types.
This commit is contained in:
Jack Franklin 2020-05-07 14:49:42 +01:00 committed by GitHub
parent 4fdb1e3cab
commit 49ce65943c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 30 additions and 67 deletions

View File

@ -17,7 +17,6 @@
import { helper, assert } from './helper'; import { helper, assert } from './helper';
import { Target } from './Target'; import { Target } from './Target';
import * as EventEmitter from 'events'; import * as EventEmitter from 'events';
import { TaskQueue } from './TaskQueue';
import { Events } from './Events'; import { Events } from './Events';
import { Connection } from './Connection'; import { Connection } from './Connection';
import { Page } from './Page'; import { Page } from './Page';
@ -49,7 +48,6 @@ export class Browser extends EventEmitter {
_ignoreHTTPSErrors: boolean; _ignoreHTTPSErrors: boolean;
_defaultViewport?: Viewport; _defaultViewport?: Viewport;
_process?: ChildProcess; _process?: ChildProcess;
_screenshotTaskQueue = new TaskQueue();
_connection: Connection; _connection: Connection;
_closeCallback: BrowserCloseCallback; _closeCallback: BrowserCloseCallback;
_defaultContext: BrowserContext; _defaultContext: BrowserContext;
@ -69,7 +67,6 @@ export class Browser extends EventEmitter {
this._ignoreHTTPSErrors = ignoreHTTPSErrors; this._ignoreHTTPSErrors = ignoreHTTPSErrors;
this._defaultViewport = defaultViewport; this._defaultViewport = defaultViewport;
this._process = process; this._process = process;
this._screenshotTaskQueue = new TaskQueue();
this._connection = connection; this._connection = connection;
this._closeCallback = closeCallback || function (): void {}; this._closeCallback = closeCallback || function (): void {};
@ -146,8 +143,7 @@ export class Browser extends EventEmitter {
context, context,
() => this._connection.createSession(targetInfo), () => this._connection.createSession(targetInfo),
this._ignoreHTTPSErrors, this._ignoreHTTPSErrors,
this._defaultViewport, this._defaultViewport
this._screenshotTaskQueue
); );
assert( assert(
!this._targets.has(event.targetInfo.targetId), !this._targets.has(event.targetInfo.targetId),

View File

@ -454,7 +454,7 @@ export class ElementHandle extends JSHandle {
}; };
} }
async screenshot(options = {}): Promise<string | Buffer> { async screenshot(options = {}): Promise<string | Buffer | void> {
let needsViewportReset = false; let needsViewportReset = false;
let boundingBox = await this.boundingBox(); let boundingBox = await this.boundingBox();

View File

@ -39,7 +39,6 @@ import {
import { Accessibility } from './Accessibility'; import { Accessibility } from './Accessibility';
import { TimeoutSettings } from './TimeoutSettings'; import { TimeoutSettings } from './TimeoutSettings';
import { PuppeteerLifeCycleEvent } from './LifecycleWatcher'; import { PuppeteerLifeCycleEvent } from './LifecycleWatcher';
import { TaskQueue } from './TaskQueue';
const writeFileAsync = helper.promisify(fs.writeFile); const writeFileAsync = helper.promisify(fs.writeFile);
@ -128,20 +127,30 @@ const paperFormats: Record<string, PaperFormat> = {
a6: { width: 4.13, height: 5.83 }, a6: { width: 4.13, height: 5.83 },
} as const; } as const;
class ScreenshotTaskQueue {
_chain: Promise<Buffer | string | void>;
constructor() {
this._chain = Promise.resolve<Buffer | string | void>(undefined);
}
public postTask(
task: () => Promise<Buffer | string>
): Promise<Buffer | string | void> {
const result = this._chain.then(task);
this._chain = result.catch(() => {});
return result;
}
}
export class Page extends EventEmitter { export class Page extends EventEmitter {
static async create( static async create(
client: CDPSession, client: CDPSession,
target: Target, target: Target,
ignoreHTTPSErrors: boolean, ignoreHTTPSErrors: boolean,
defaultViewport: Viewport | null, defaultViewport: Viewport | null
screenshotTaskQueue: TaskQueue
): Promise<Page> { ): Promise<Page> {
const page = new Page( const page = new Page(client, target, ignoreHTTPSErrors);
client,
target,
ignoreHTTPSErrors,
screenshotTaskQueue
);
await page._initialize(); await page._initialize();
if (defaultViewport) await page.setViewport(defaultViewport); if (defaultViewport) await page.setViewport(defaultViewport);
return page; return page;
@ -162,19 +171,14 @@ export class Page extends EventEmitter {
_coverage: Coverage; _coverage: Coverage;
_javascriptEnabled = true; _javascriptEnabled = true;
_viewport: Viewport | null; _viewport: Viewport | null;
_screenshotTaskQueue: TaskQueue; _screenshotTaskQueue: ScreenshotTaskQueue;
_workers = new Map<string, PuppeteerWorker>(); _workers = new Map<string, PuppeteerWorker>();
// TODO: improve this typedef - it's a function that takes a file chooser or something? // TODO: improve this typedef - it's a function that takes a file chooser or something?
_fileChooserInterceptors = new Set<Function>(); _fileChooserInterceptors = new Set<Function>();
_disconnectPromise?: Promise<Error>; _disconnectPromise?: Promise<Error>;
constructor( constructor(client: CDPSession, target: Target, ignoreHTTPSErrors: boolean) {
client: CDPSession,
target: Target,
ignoreHTTPSErrors: boolean,
screenshotTaskQueue: TaskQueue
) {
super(); super();
this._client = client; this._client = client;
this._target = target; this._target = target;
@ -191,7 +195,7 @@ export class Page extends EventEmitter {
this._emulationManager = new EmulationManager(client); this._emulationManager = new EmulationManager(client);
this._tracing = new Tracing(client); this._tracing = new Tracing(client);
this._coverage = new Coverage(client); this._coverage = new Coverage(client);
this._screenshotTaskQueue = screenshotTaskQueue; this._screenshotTaskQueue = new ScreenshotTaskQueue();
this._viewport = null; this._viewport = null;
client.on('Target.attachedToTarget', (event) => { client.on('Target.attachedToTarget', (event) => {
@ -948,7 +952,9 @@ export class Page extends EventEmitter {
await this._frameManager.networkManager().setCacheEnabled(enabled); await this._frameManager.networkManager().setCacheEnabled(enabled);
} }
async screenshot(options: ScreenshotOptions = {}): Promise<Buffer | string> { async screenshot(
options: ScreenshotOptions = {}
): Promise<Buffer | string | void> {
let screenshotType = null; let screenshotType = null;
// options.type takes precedence over inferring the type from options.path // options.type takes precedence over inferring the type from options.path
// because it may be a 0-length file with no extension created beforehand (i.e. as a temp file). // because it may be a 0-length file with no extension created beforehand (i.e. as a temp file).
@ -1023,8 +1029,8 @@ export class Page extends EventEmitter {
'Expected options.clip.height not to be 0.' 'Expected options.clip.height not to be 0.'
); );
} }
return this._screenshotTaskQueue.postTask( return this._screenshotTaskQueue.postTask(() =>
this._screenshotTask.bind(this, screenshotType, options) this._screenshotTask(screenshotType, options)
); );
} }

View File

@ -18,7 +18,6 @@ import { Events } from './Events';
import { Page } from './Page'; import { Page } from './Page';
import { Worker as PuppeteerWorker } from './Worker'; import { Worker as PuppeteerWorker } from './Worker';
import { CDPSession } from './Connection'; import { CDPSession } from './Connection';
import { TaskQueue } from './TaskQueue';
import { Browser, BrowserContext } from './Browser'; import { Browser, BrowserContext } from './Browser';
import type { Viewport } from './PuppeteerViewport'; import type { Viewport } from './PuppeteerViewport';
@ -29,7 +28,6 @@ export class Target {
_sessionFactory: () => Promise<CDPSession>; _sessionFactory: () => Promise<CDPSession>;
_ignoreHTTPSErrors: boolean; _ignoreHTTPSErrors: boolean;
_defaultViewport?: Viewport; _defaultViewport?: Viewport;
_screenshotTaskQueue: TaskQueue;
_pagePromise?: Promise<Page>; _pagePromise?: Promise<Page>;
_workerPromise?: Promise<PuppeteerWorker>; _workerPromise?: Promise<PuppeteerWorker>;
_initializedPromise: Promise<boolean>; _initializedPromise: Promise<boolean>;
@ -43,8 +41,7 @@ export class Target {
browserContext: BrowserContext, browserContext: BrowserContext,
sessionFactory: () => Promise<CDPSession>, sessionFactory: () => Promise<CDPSession>,
ignoreHTTPSErrors: boolean, ignoreHTTPSErrors: boolean,
defaultViewport: Viewport | null, defaultViewport: Viewport | null
screenshotTaskQueue: TaskQueue
) { ) {
this._targetInfo = targetInfo; this._targetInfo = targetInfo;
this._browserContext = browserContext; this._browserContext = browserContext;
@ -52,7 +49,6 @@ export class Target {
this._sessionFactory = sessionFactory; this._sessionFactory = sessionFactory;
this._ignoreHTTPSErrors = ignoreHTTPSErrors; this._ignoreHTTPSErrors = ignoreHTTPSErrors;
this._defaultViewport = defaultViewport; this._defaultViewport = defaultViewport;
this._screenshotTaskQueue = screenshotTaskQueue;
/** @type {?Promise<!Puppeteer.Page>} */ /** @type {?Promise<!Puppeteer.Page>} */
this._pagePromise = null; this._pagePromise = null;
/** @type {?Promise<!PuppeteerWorker>} */ /** @type {?Promise<!PuppeteerWorker>} */
@ -93,8 +89,7 @@ export class Target {
client, client,
this, this,
this._ignoreHTTPSErrors, this._ignoreHTTPSErrors,
this._defaultViewport, this._defaultViewport
this._screenshotTaskQueue
) )
); );
} }

View File

@ -1,34 +0,0 @@
/**
* Copyright 2020 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.
*/
/* TODO(jacktfranklin@): once we are calling this from TS files we can
* avoid the horrible void | any type and instead make use of generics
* to make this into TaskQueue<T> and let the caller tell us what types
* the promise in the queue should return.
*/
export class TaskQueue {
_chain: Promise<void | any>;
constructor() {
this._chain = Promise.resolve();
}
public postTask(task: () => any): Promise<void | any> {
const result = this._chain.then(task);
this._chain = result.catch(() => {});
return result;
}
}