Skip to content

Commit 8c6f0fe

Browse files
committed
test(NODE-6692): check that event emitters have error listeners
1 parent 6b15f20 commit 8c6f0fe

25 files changed

+126
-79
lines changed

.mocharc.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
"source-map-support/register",
55
"ts-node/register",
66
"test/tools/runner/chai_addons.ts",
7-
"test/tools/runner/hooks/unhandled_checker.ts"
7+
"test/tools/runner/ee_checker.ts"
88
],
99
"extension": [
1010
"js",

src/cmap/connection.ts

+2
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ import {
4646
HostAddress,
4747
maxWireVersion,
4848
type MongoDBNamespace,
49+
noop,
4950
now,
5051
once,
5152
squashError,
@@ -229,6 +230,7 @@ export class Connection extends TypedEventEmitter<ConnectionEvents> {
229230

230231
constructor(stream: Stream, options: ConnectionOptions) {
231232
super();
233+
this.on('error', noop);
232234

233235
this.socket = stream;
234236
this.id = options.id;

src/cmap/connection_pool.ts

+2
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ import {
3434
kDispose,
3535
List,
3636
makeCounter,
37+
noop,
3738
now,
3839
promiseWithResolvers
3940
} from '../utils';
@@ -200,6 +201,7 @@ export class ConnectionPool extends TypedEventEmitter<ConnectionPoolEvents> {
200201

201202
constructor(server: Server, options: ConnectionPoolOptions) {
202203
super();
204+
this.on('error', noop);
203205

204206
this.options = Object.freeze({
205207
connectionType: Connection,

src/cursor/abstract_cursor.ts

+2
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ import {
2727
type Disposable,
2828
kDispose,
2929
type MongoDBNamespace,
30+
noop,
3031
squashError
3132
} from '../utils';
3233

@@ -267,6 +268,7 @@ export abstract class AbstractCursor<
267268
options: AbstractCursorOptions & Abortable = {}
268269
) {
269270
super();
271+
this.on('error', noop);
270272

271273
if (!client.s.isMongoClient) {
272274
throw new MongoRuntimeError('Cursor must be constructed with MongoClient');

src/gridfs/index.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import { type Filter, TypedEventEmitter } from '../mongo_types';
77
import type { ReadPreference } from '../read_preference';
88
import type { Sort } from '../sort';
99
import { CSOTTimeoutContext } from '../timeout';
10-
import { resolveOptions } from '../utils';
10+
import { noop, resolveOptions } from '../utils';
1111
import { WriteConcern, type WriteConcernOptions } from '../write_concern';
1212
import type { FindOptions } from './../operations/find';
1313
import {
@@ -87,6 +87,7 @@ export class GridFSBucket extends TypedEventEmitter<GridFSBucketEvents> {
8787

8888
constructor(db: Db, options?: GridFSBucketOptions) {
8989
super();
90+
this.on('error', noop);
9091
this.setMaxListeners(0);
9192
const privateOptions = resolveOptions(db, {
9293
...DEFAULT_GRIDFS_BUCKET_OPTIONS,

src/mongo_client.ts

+2
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ import {
5858
hostMatchesWildcards,
5959
isHostMatch,
6060
type MongoDBNamespace,
61+
noop,
6162
ns,
6263
resolveOptions,
6364
squashError
@@ -386,6 +387,7 @@ export class MongoClient extends TypedEventEmitter<MongoClientEvents> implements
386387

387388
constructor(url: string, options?: MongoClientOptions) {
388389
super();
390+
this.on('error', noop);
389391

390392
this.options = parseOptions(url, this, options);
391393

src/mongo_types.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ import {
2424
type MongoLogger
2525
} from './mongo_logger';
2626
import type { Sort } from './sort';
27+
import { noop } from './utils';
2728

2829
/** @internal */
2930
export type TODO_NODE_3286 = any;
@@ -472,7 +473,12 @@ export class TypedEventEmitter<Events extends EventsDescription> extends EventEm
472473
}
473474

474475
/** @public */
475-
export class CancellationToken extends TypedEventEmitter<{ cancel(): void }> {}
476+
export class CancellationToken extends TypedEventEmitter<{ cancel(): void }> {
477+
constructor(...args: any[]) {
478+
super(...args);
479+
this.on('error', noop);
480+
}
481+
}
476482

477483
/** @public */
478484
export type Abortable = {

src/sdam/monitor.ts

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import {
1313
type Callback,
1414
type EventEmitterWithState,
1515
makeStateMachine,
16+
noop,
1617
now,
1718
ns
1819
} from '../utils';
@@ -102,6 +103,7 @@ export class Monitor extends TypedEventEmitter<MonitorEvents> {
102103

103104
constructor(server: Server, options: MonitorOptions) {
104105
super();
106+
this.on('error', noop);
105107

106108
this.server = server;
107109
this.connection = null;

src/sdam/server.ts

+2
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@ import {
4747
makeStateMachine,
4848
maxWireVersion,
4949
type MongoDBNamespace,
50+
noop,
5051
supportsRetryableWrites
5152
} from '../utils';
5253
import { throwIfWriteConcernError } from '../write_concern';
@@ -142,6 +143,7 @@ export class Server extends TypedEventEmitter<ServerEvents> {
142143
*/
143144
constructor(topology: Topology, description: ServerDescription, options: ServerOptions) {
144145
super();
146+
this.on('error', noop);
145147

146148
this.serverApi = options.serverApi;
147149

src/sdam/srv_polling.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import { clearTimeout, setTimeout } from 'timers';
33

44
import { MongoRuntimeError } from '../error';
55
import { TypedEventEmitter } from '../mongo_types';
6-
import { checkParentDomainMatch, HostAddress, squashError } from '../utils';
6+
import { checkParentDomainMatch, HostAddress, noop, squashError } from '../utils';
77

88
/**
99
* @internal
@@ -49,6 +49,7 @@ export class SrvPoller extends TypedEventEmitter<SrvPollerEvents> {
4949

5050
constructor(options: SrvPollerOptions) {
5151
super();
52+
this.on('error', noop);
5253

5354
if (!options || !options.srvHost) {
5455
throw new MongoRuntimeError('Options for SrvPoller must exist and include srvHost');

src/sdam/topology.ts

+2
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ import {
4444
kDispose,
4545
List,
4646
makeStateMachine,
47+
noop,
4748
now,
4849
ns,
4950
promiseWithResolvers,
@@ -248,6 +249,7 @@ export class Topology extends TypedEventEmitter<TopologyEvents> {
248249
options: TopologyOptions
249250
) {
250251
super();
252+
this.on('error', noop);
251253

252254
this.client = client;
253255
// Options should only be undefined in tests, MongoClient will always have defined options

src/sessions.ts

+2
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ import {
4343
isPromiseLike,
4444
List,
4545
maxWireVersion,
46+
noop,
4647
now,
4748
squashError,
4849
uuidV4
@@ -161,6 +162,7 @@ export class ClientSession
161162
clientOptions: MongoOptions
162163
) {
163164
super();
165+
this.on('error', noop);
164166

165167
if (client == null) {
166168
// TODO(NODE-3483)

test/integration/change-streams/change_stream.test.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -736,6 +736,7 @@ describe('Change Streams', function () {
736736
// ChangeStream detects emitter usage via 'newListener' event
737737
// so this covers all emitter methods
738738
});
739+
changeStream.on('error', () => null); // one must listen for errors if they use EE mode.
739740

740741
await once(changeStream.cursor, 'init');
741742
expect(changeStream).to.have.property('mode', 'emitter');
@@ -971,7 +972,7 @@ describe('Change Streams', function () {
971972
{ requires: { topology: '!single' } },
972973
async function () {
973974
changeStream = collection.watch([]);
974-
changeStream.on('change', sinon.stub());
975+
changeStream.on('change', sinon.stub()).on('error', () => null);
975976

976977
try {
977978
// eslint-disable-next-line @typescript-eslint/no-unused-vars

test/integration/change-streams/change_streams.prose.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -858,6 +858,7 @@ describe('Change Stream prose tests', function () {
858858
expect(err).to.not.exist;
859859
coll = client.db('integration_tests').collection('setupAfterTest');
860860
const changeStream = coll.watch();
861+
changeStream.on('error', done);
861862
waitForStarted(changeStream, () => {
862863
coll.insertOne({ x: 1 }, { writeConcern: { w: 'majority', j: true } }, err => {
863864
expect(err).to.not.exist;
@@ -932,6 +933,7 @@ describe('Change Stream prose tests', function () {
932933
let events = [];
933934
client.on('commandStarted', e => recordEvent(events, e));
934935
const changeStream = coll.watch([], { startAfter });
936+
changeStream.on('error', done);
935937
this.defer(() => changeStream.close());
936938

937939
changeStream.on('change', change => {

test/integration/crud/crud_api.test.ts

+7-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { expect } from 'chai';
2-
import { on } from 'events';
2+
import { once } from 'events';
33
import * as semver from 'semver';
44
import * as sinon from 'sinon';
55

@@ -307,13 +307,15 @@ describe('CRUD API', function () {
307307

308308
describe('#stream()', () => {
309309
it('creates a node stream that emits data events', async () => {
310-
const count = 0;
310+
let count = 0;
311311
const cursor = makeCursor();
312312
const stream = cursor.stream();
313-
on(stream, 'data');
314-
cursor.once('close', function () {
315-
expect(count).to.equal(2);
313+
const willClose = once(cursor, 'close');
314+
stream.on('data', () => {
315+
count++;
316316
});
317+
await willClose;
318+
expect(count).to.equal(2);
317319
});
318320
});
319321

test/integration/crud/misc_cursors.test.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -1993,15 +1993,15 @@ describe('Cursor', function () {
19931993
expect(res).property('insertedId').to.exist;
19941994
}, 300);
19951995

1996-
const start = new Date();
1996+
const start = performance.now();
19971997
const doc1 = await cursor.next();
19981998
expect(doc1).to.have.property('b', 2);
1999-
const end = new Date();
1999+
const end = performance.now();
20002000

20012001
await later; // make sure this finished, without a failure
20022002

20032003
// We should see here that cursor.next blocked for at least 300ms
2004-
expect(end.getTime() - start.getTime()).to.be.at.least(300);
2004+
expect(end - start).to.be.at.least(290);
20052005
}
20062006
}
20072007
);

test/integration/node-specific/examples/change_streams.test.js

+37-17
Original file line numberDiff line numberDiff line change
@@ -66,9 +66,13 @@ maybeDescribe('examples(change-stream):', function () {
6666
// Start Changestream Example 1
6767
const collection = db.collection('inventory');
6868
const changeStream = collection.watch();
69-
changeStream.on('change', next => {
70-
// process next document
71-
});
69+
changeStream
70+
.on('change', next => {
71+
// process next document
72+
})
73+
.once('error', () => {
74+
// handle error
75+
});
7276
// End Changestream Example 1
7377

7478
const changeStreamIterator = collection.watch();
@@ -113,9 +117,13 @@ maybeDescribe('examples(change-stream):', function () {
113117
// Start Changestream Example 2
114118
const collection = db.collection('inventory');
115119
const changeStream = collection.watch([], { fullDocument: 'updateLookup' });
116-
changeStream.on('change', next => {
117-
// process next document
118-
});
120+
changeStream
121+
.on('change', next => {
122+
// process next document
123+
})
124+
.once('error', error => {
125+
// handle error
126+
});
119127
// End Changestream Example 2
120128

121129
// Start Changestream Example 2 Alternative
@@ -151,15 +159,23 @@ maybeDescribe('examples(change-stream):', function () {
151159
const changeStream = collection.watch();
152160

153161
let newChangeStream;
154-
changeStream.once('change', next => {
155-
const resumeToken = changeStream.resumeToken;
156-
changeStream.close();
157-
158-
newChangeStream = collection.watch([], { resumeAfter: resumeToken });
159-
newChangeStream.on('change', next => {
160-
processChange(next);
162+
changeStream
163+
.once('change', next => {
164+
const resumeToken = changeStream.resumeToken;
165+
changeStream.close();
166+
167+
newChangeStream = collection.watch([], { resumeAfter: resumeToken });
168+
newChangeStream
169+
.on('change', next => {
170+
processChange(next);
171+
})
172+
.once('error', error => {
173+
// handle error
174+
});
175+
})
176+
.once('error', error => {
177+
// handle error
161178
});
162-
});
163179
// End Changestream Example 3
164180

165181
// Start Changestream Example 3 Alternative
@@ -200,9 +216,13 @@ maybeDescribe('examples(change-stream):', function () {
200216

201217
const collection = db.collection('inventory');
202218
const changeStream = collection.watch(pipeline);
203-
changeStream.on('change', next => {
204-
// process next document
205-
});
219+
changeStream
220+
.on('change', next => {
221+
// process next document
222+
})
223+
.once('error', error => {
224+
// handle error
225+
});
206226
// End Changestream Example 4
207227

208228
// Start Changestream Example 4 Alternative

test/integration/sessions/sessions.test.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -430,13 +430,14 @@ describe('Sessions Spec', function () {
430430
});
431431
});
432432

433-
context('when using a LegacyMongoClient', () => {
433+
context.skip('when using a LegacyMongoClient', () => {
434434
let legacyClient;
435435
beforeEach(async function () {
436436
const options = this.configuration.serverApi
437437
? { serverApi: this.configuration.serverApi }
438438
: {};
439439
legacyClient = new LegacyMongoClient(this.configuration.url(), options);
440+
legacyClient.on('error', () => null); // Uses released version of the driver so it won't be fixed until the error listeners are published
440441
});
441442

442443
afterEach(async function () {

test/mocha_mongodb.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,9 @@
33
"require": [
44
"source-map-support/register",
55
"ts-node/register",
6+
"test/tools/runner/ee_checker.ts",
67
"test/tools/runner/chai_addons.ts",
78
"test/tools/runner/hooks/configuration.ts",
8-
"test/tools/runner/hooks/unhandled_checker.ts",
99
"test/tools/runner/hooks/leak_checker.ts",
1010
"test/tools/runner/hooks/legacy_crud_shims.ts"
1111
],

test/tools/cmap_spec_runner.ts

+1
Original file line numberDiff line numberDiff line change
@@ -295,6 +295,7 @@ export class ThreadContext {
295295
poolOptions: Partial<ConnectionPoolOptions> = {},
296296
contextOptions: { injectPoolStats: boolean }
297297
) {
298+
this.poolEventsEventEmitter.on('error', () => null);
298299
this.#poolOptions = poolOptions;
299300
this.#hostAddress = hostAddress;
300301
this.#server = server;

0 commit comments

Comments
 (0)