Use Mitt as the Event Emitter (#5907)

* chore: migrate to Mitt as the EventEmitter

This commit moves us to using Mitt [1] for the event emitter in
Puppeteer. This removes our dependency to Node's EventEmitter which is
part of a larger stream of work to enable a Puppeteer-web version that
doesn't depend on Node.

There are no large breaking changes as we support the main methods that
EventEmitter had, but it also provides some methods that Puppeteer
didn't use. Technically end users could depend on this but it's
unlikely.

[1]: https://github.com/developit/mitt
This commit is contained in:
Jack Franklin 2020-05-29 09:59:26 +01:00 committed by GitHub
parent a2ba6f0abb
commit 1d4d25a0f3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 203 additions and 23 deletions

View File

@ -335,6 +335,13 @@
* [coverage.stopCSSCoverage()](#coveragestopcsscoverage) * [coverage.stopCSSCoverage()](#coveragestopcsscoverage)
* [coverage.stopJSCoverage()](#coveragestopjscoverage) * [coverage.stopJSCoverage()](#coveragestopjscoverage)
- [class: TimeoutError](#class-timeouterror) - [class: TimeoutError](#class-timeouterror)
- [class: EventEmitter](#class-eventemitter)
* [eventEmitter.emit(event, [eventData])](#eventemitteremitevent-eventdata)
* [eventEmitter.listenerCount(event)](#eventemitterlistenercountevent)
* [eventEmitter.off(event, handler)](#eventemitteroffevent-handler)
* [eventEmitter.on(event, handler)](#eventemitteronevent-handler)
* [eventEmitter.once(event, handler)](#eventemitteronceevent-handler)
* [eventEmitter.removeListener(event, handler)](#eventemitterremovelistenerevent-handler)
<!-- GEN:stop --> <!-- GEN:stop -->
### Overview ### Overview
@ -3935,6 +3942,35 @@ reported.
TimeoutError is emitted whenever certain operations are terminated due to timeout, e.g. [page.waitForSelector(selector[, options])](#pagewaitforselectorselector-options) or [puppeteer.launch([options])](#puppeteerlaunchoptions). TimeoutError is emitted whenever certain operations are terminated due to timeout, e.g. [page.waitForSelector(selector[, options])](#pagewaitforselectorselector-options) or [puppeteer.launch([options])](#puppeteerlaunchoptions).
### class: EventEmitter
A small EventEmitter class backed by [Mitt](https://github.com/developit/mitt/).
#### eventEmitter.emit(event, [eventData])
- `event` <[string]|[symbol]> the event to trigger.
- `eventData` <[Object]> additional data to send with the event.
#### eventEmitter.listenerCount(event)
- `event` <[string]|[symbol]> the event to check for listeners.
- returns: <[number]> the number of listeners for the given event.
#### eventEmitter.off(event, handler)
- `event` <[string]|[symbol]> the event to remove the handler from.
- `handler` <[Function]> the event listener that will be removed.
#### eventEmitter.on(event, handler)
- `event` <[string]|[symbol]> the event to add the handler to.
- `handler` <[Function]> the event listener that will be added.
#### eventEmitter.once(event, handler)
- `event` <[string]|[symbol]> the event to add the handler to.
- `handler` <[Function]> the event listener that will be added.
#### eventEmitter.removeListener(event, handler)
- `event` <[string]|[symbol]> the event to remove the handler from.
- `handler` <[Function]> the event listener that will be removed.
This method is identical to `off` and maintained for compatibility with Node's EventEmitter. We recommend using `off` by default.
[AXNode]: #accessibilitysnapshotoptions "AXNode" [AXNode]: #accessibilitysnapshotoptions "AXNode"
@ -3984,4 +4020,5 @@ TimeoutError is emitted whenever certain operations are terminated due to timeou
[selector]: https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors "selector" [selector]: https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors "selector"
[stream.Readable]: https://nodejs.org/api/stream.html#stream_class_stream_readable "stream.Readable" [stream.Readable]: https://nodejs.org/api/stream.html#stream_class_stream_readable "stream.Readable"
[string]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures#String_type "String" [string]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures#String_type "String"
[symbol]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Data_structures#Symbol_type "Symbol"
[xpath]: https://developer.mozilla.org/en-US/docs/Web/XPath "xpath" [xpath]: https://developer.mozilla.org/en-US/docs/Web/XPath "xpath"

View File

@ -48,6 +48,7 @@
"extract-zip": "^2.0.0", "extract-zip": "^2.0.0",
"https-proxy-agent": "^4.0.0", "https-proxy-agent": "^4.0.0",
"mime": "^2.0.3", "mime": "^2.0.3",
"mitt": "^2.0.1",
"progress": "^2.0.1", "progress": "^2.0.1",
"proxy-from-env": "^1.0.0", "proxy-from-env": "^1.0.0",
"rimraf": "^3.0.2", "rimraf": "^3.0.2",

View File

@ -16,7 +16,7 @@
import { helper, assert } from './helper'; import { helper, assert } from './helper';
import { Target } from './Target'; import { Target } from './Target';
import * as EventEmitter from 'events'; import { EventEmitter } from './EventEmitter';
import { Events } from './Events'; import { Events } from './Events';
import Protocol from './protocol'; import Protocol from './protocol';
import { Connection } from './Connection'; import { Connection } from './Connection';

View File

@ -22,11 +22,11 @@ import * as childProcess from 'child_process';
import * as https from 'https'; import * as https from 'https';
import * as http from 'http'; import * as http from 'http';
import * as extractZip from 'extract-zip'; import extractZip from 'extract-zip';
import * as debug from 'debug'; import debug from 'debug';
import * as removeRecursive from 'rimraf'; import removeRecursive from 'rimraf';
import * as URL from 'url'; import * as URL from 'url';
import * as ProxyAgent from 'https-proxy-agent'; import ProxyAgent from 'https-proxy-agent';
import { getProxyForUrl } from 'proxy-from-env'; import { getProxyForUrl } from 'proxy-from-env';
import { helper, assert } from './helper'; import { helper, assert } from './helper';

View File

@ -15,12 +15,12 @@
*/ */
import { assert } from './helper'; import { assert } from './helper';
import { Events } from './Events'; import { Events } from './Events';
import * as debug from 'debug'; import debug from 'debug';
const debugProtocol = debug('puppeteer:protocol'); const debugProtocol = debug('puppeteer:protocol');
import Protocol from './protocol'; import Protocol from './protocol';
import type { ConnectionTransport } from './ConnectionTransport'; import type { ConnectionTransport } from './ConnectionTransport';
import * as EventEmitter from 'events'; import { EventEmitter } from './EventEmitter';
interface ConnectionCallback { interface ConnectionCallback {
resolve: Function; resolve: Function;

61
src/EventEmitter.ts Normal file
View File

@ -0,0 +1,61 @@
import mitt, { Emitter, EventType, Handler } from 'mitt';
export interface CommonEventEmitter {
on(event: EventType, handler: Handler): void;
off(event: EventType, handler: Handler): void;
/* To maintain parity with the built in NodeJS event emitter which uses removeListener
* rather than `off`.
* If you're implementing new code you should use `off`.
*/
removeListener(event: EventType, handler: Handler): void;
emit(event: EventType, eventData?: any): void;
once(event: EventType, handler: Handler): void;
listenerCount(event: string): number;
}
export class EventEmitter implements CommonEventEmitter {
private emitter: Emitter;
private listenerCounts = new Map<EventType, number>();
constructor() {
this.emitter = mitt(new Map());
}
on(event: EventType, handler: Handler): void {
this.emitter.on(event, handler);
const existingCounts = this.listenerCounts.get(event);
if (existingCounts) {
this.listenerCounts.set(event, existingCounts + 1);
} else {
this.listenerCounts.set(event, 1);
}
}
off(event: EventType, handler: Handler): void {
this.emitter.off(event, handler);
const existingCounts = this.listenerCounts.get(event);
this.listenerCounts.set(event, existingCounts - 1);
}
removeListener(event: EventType, handler: Handler): void {
this.off(event, handler);
}
emit(event: EventType, eventData?: any): void {
this.emitter.emit(event, eventData);
}
once(event: EventType, handler: Handler): void {
const onceHandler: Handler = (eventData) => {
handler(eventData);
this.off(event, onceHandler);
};
this.on(event, onceHandler);
}
listenerCount(event: EventType): number {
return this.listenerCounts.get(event) || 0;
}
}

View File

@ -14,7 +14,7 @@
* limitations under the License. * limitations under the License.
*/ */
import * as EventEmitter from 'events'; import { EventEmitter } from './EventEmitter';
import { helper, assert, debugError } from './helper'; import { helper, assert, debugError } from './helper';
import { Events } from './Events'; import { Events } from './Events';
import { ExecutionContext, EVALUATION_SCRIPT_URL } from './ExecutionContext'; import { ExecutionContext, EVALUATION_SCRIPT_URL } from './ExecutionContext';

View File

@ -13,7 +13,7 @@
* See the License for the specific language governing permissions and * See the License for the specific language governing permissions and
* limitations under the License. * limitations under the License.
*/ */
import * as EventEmitter from 'events'; import { EventEmitter } from './EventEmitter';
import { helper, assert, debugError } from './helper'; import { helper, assert, debugError } from './helper';
import Protocol from './protocol'; import Protocol from './protocol';
import { Events } from './Events'; import { Events } from './Events';

View File

@ -15,7 +15,7 @@
*/ */
import * as fs from 'fs'; import * as fs from 'fs';
import * as EventEmitter from 'events'; import { EventEmitter } from './EventEmitter';
import * as mime from 'mime'; import * as mime from 'mime';
import { Events } from './Events'; import { Events } from './Events';
import { Connection, CDPSession } from './Connection'; import { Connection, CDPSession } from './Connection';

View File

@ -13,7 +13,7 @@
* See the License for the specific language governing permissions and * See the License for the specific language governing permissions and
* limitations under the License. * limitations under the License.
*/ */
import * as NodeWebSocket from 'ws'; import NodeWebSocket from 'ws';
import type { ConnectionTransport } from './ConnectionTransport'; import type { ConnectionTransport } from './ConnectionTransport';
export class WebSocketTransport implements ConnectionTransport { export class WebSocketTransport implements ConnectionTransport {

View File

@ -13,7 +13,7 @@
* See the License for the specific language governing permissions and * See the License for the specific language governing permissions and
* limitations under the License. * limitations under the License.
*/ */
import { EventEmitter } from 'events'; import { EventEmitter } from './EventEmitter';
import { debugError } from './helper'; import { debugError } from './helper';
import { ExecutionContext } from './ExecutionContext'; import { ExecutionContext } from './ExecutionContext';
import { JSHandle } from './JSHandle'; import { JSHandle } from './JSHandle';

View File

@ -29,6 +29,7 @@ module.exports = {
Dialog: require('./Dialog').Dialog, Dialog: require('./Dialog').Dialog,
ElementHandle: require('./JSHandle').ElementHandle, ElementHandle: require('./JSHandle').ElementHandle,
ExecutionContext: require('./ExecutionContext').ExecutionContext, ExecutionContext: require('./ExecutionContext').ExecutionContext,
EventEmitter: require('./EventEmitter').EventEmitter,
FileChooser: require('./FileChooser').FileChooser, FileChooser: require('./FileChooser').FileChooser,
Frame: require('./FrameManager').Frame, Frame: require('./FrameManager').Frame,
JSHandle: require('./JSHandle').JSHandle, JSHandle: require('./JSHandle').JSHandle,

View File

@ -14,11 +14,12 @@
* limitations under the License. * limitations under the License.
*/ */
import { TimeoutError } from './Errors'; import { TimeoutError } from './Errors';
import * as debug from 'debug'; import debug from 'debug';
import * as fs from 'fs'; import * as fs from 'fs';
import { CDPSession } from './Connection'; import { CDPSession } from './Connection';
import { promisify } from 'util'; import { promisify } from 'util';
import Protocol from './protocol'; import Protocol from './protocol';
import type { CommonEventEmitter } from './EventEmitter';
const openAsync = promisify(fs.open); const openAsync = promisify(fs.open);
const writeAsync = promisify(fs.write); const writeAsync = promisify(fs.write);
@ -131,13 +132,13 @@ function installAsyncStackHooks(classType: AnyClass): void {
} }
export interface PuppeteerEventListener { export interface PuppeteerEventListener {
emitter: NodeJS.EventEmitter; emitter: CommonEventEmitter;
eventName: string | symbol; eventName: string | symbol;
handler: (...args: any[]) => void; handler: (...args: any[]) => void;
} }
function addEventListener( function addEventListener(
emitter: NodeJS.EventEmitter, emitter: CommonEventEmitter,
eventName: string | symbol, eventName: string | symbol,
handler: (...args: any[]) => void handler: (...args: any[]) => void
): PuppeteerEventListener { ): PuppeteerEventListener {
@ -147,7 +148,7 @@ function addEventListener(
function removeEventListeners( function removeEventListeners(
listeners: Array<{ listeners: Array<{
emitter: NodeJS.EventEmitter; emitter: CommonEventEmitter;
eventName: string | symbol; eventName: string | symbol;
handler: (...args: any[]) => void; handler: (...args: any[]) => void;
}> }>
@ -166,7 +167,7 @@ function isNumber(obj: unknown): obj is number {
} }
async function waitForEvent<T extends any>( async function waitForEvent<T extends any>(
emitter: NodeJS.EventEmitter, emitter: CommonEventEmitter,
eventName: string | symbol, eventName: string | symbol,
predicate: (event: T) => boolean, predicate: (event: T) => boolean,
timeout: number, timeout: number,

View File

@ -14,9 +14,9 @@
* limitations under the License. * limitations under the License.
*/ */
import * as debug from 'debug'; import debug from 'debug';
import * as removeFolder from 'rimraf'; import removeFolder from 'rimraf';
import * as childProcess from 'child_process'; import * as childProcess from 'child_process';
import { helper, assert, debugError } from '../helper'; import { helper, assert, debugError } from '../helper';
import type { LaunchOptions } from './LaunchOptions'; import type { LaunchOptions } from './LaunchOptions';

View File

@ -5,7 +5,8 @@
"outDir": "./lib", "outDir": "./lib",
"target": "ESNext", "target": "ESNext",
"moduleResolution": "node", "moduleResolution": "node",
"module": "CommonJS" "module": "CommonJS",
"esModuleInterop": true
}, },
"include": [ "include": [
"src" "src"

View File

@ -179,6 +179,32 @@ const expectedNonExistingMethods = new Map([
*/ */
['Page', new Set(['emulateMedia'])], ['Page', new Set(['emulateMedia'])],
]); ]);
/* Methods that are defined in code but are not documented */
const expectedNotFoundMethods = new Map([
/* all the methods from our EventEmitter that we don't document for each subclass */
[
'Browser',
new Set(['emit', 'listenerCount', 'off', 'on', 'once', 'removeListener']),
],
[
'BrowserContext',
new Set(['emit', 'listenerCount', 'off', 'on', 'once', 'removeListener']),
],
[
'CDPSession',
new Set(['emit', 'listenerCount', 'off', 'on', 'once', 'removeListener']),
],
[
'Page',
new Set(['emit', 'listenerCount', 'off', 'on', 'once', 'removeListener']),
],
[
'Worker',
new Set(['emit', 'listenerCount', 'off', 'on', 'once', 'removeListener']),
],
]);
/** /**
* @param {!Documentation} actual * @param {!Documentation} actual
* @param {!Documentation} expected * @param {!Documentation} expected
@ -204,14 +230,24 @@ function compareDocumentations(actual, expected) {
const methodDiff = diff(actualMethods, expectedMethods); const methodDiff = diff(actualMethods, expectedMethods);
for (const methodName of methodDiff.extra) { for (const methodName of methodDiff.extra) {
const missingMethodsForClass = expectedNonExistingMethods.get(className); const nonExistingMethodsForClass = expectedNonExistingMethods.get(
if (missingMethodsForClass && missingMethodsForClass.has(methodName)) className
);
if (
nonExistingMethodsForClass &&
nonExistingMethodsForClass.has(methodName)
)
continue; continue;
errors.push(`Non-existing method found: ${className}.${methodName}()`); errors.push(`Non-existing method found: ${className}.${methodName}()`);
} }
for (const methodName of methodDiff.missing)
for (const methodName of methodDiff.missing) {
const missingMethodsForClass = expectedNotFoundMethods.get(className);
if (missingMethodsForClass && missingMethodsForClass.has(methodName))
continue;
errors.push(`Method not found: ${className}.${methodName}()`); errors.push(`Method not found: ${className}.${methodName}()`);
}
for (const methodName of methodDiff.equal) { for (const methodName of methodDiff.equal) {
const actualMethod = actualClass.methods.get(methodName); const actualMethod = actualClass.methods.get(methodName);
@ -611,6 +647,48 @@ function compareDocumentations(actual, expected) {
expectedName: 'VisionDeficiency', expectedName: 'VisionDeficiency',
}, },
], ],
[
'Method EventEmitter.emit() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
[
'Method EventEmitter.listenerCount() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
[
'Method EventEmitter.off() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
[
'Method EventEmitter.on() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
[
'Method EventEmitter.once() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
[
'Method EventEmitter.removeListener() event',
{
actualName: 'string|symbol',
expectedName: 'Object',
},
],
]); ]);
const expectedForSource = expectedNamingMismatches.get(source); const expectedForSource = expectedNamingMismatches.get(source);