Skip to content

Commit 886cefb

Browse files
fix(NODE-6355): respect utf8 validation option when iterating cursors (#4220)
1 parent 25c84a4 commit 886cefb

File tree

14 files changed

+303
-199
lines changed

14 files changed

+303
-199
lines changed

.eslintrc.json

+11-2
Original file line numberDiff line numberDiff line change
@@ -124,7 +124,8 @@
124124
"@typescript-eslint/no-unused-vars": [
125125
"error",
126126
{
127-
"argsIgnorePattern": "^_"
127+
"argsIgnorePattern": "^_",
128+
"varsIgnorePattern": "^_"
128129
}
129130
]
130131
},
@@ -275,7 +276,15 @@
275276
],
276277
"parser": "@typescript-eslint/parser",
277278
"rules": {
278-
"unused-imports/no-unused-imports": "error"
279+
"unused-imports/no-unused-imports": "error",
280+
"@typescript-eslint/consistent-type-imports": [
281+
"error",
282+
{
283+
"prefer": "type-imports",
284+
"disallowTypeAnnotations": false,
285+
"fixStyle": "separate-type-imports"
286+
}
287+
]
279288
}
280289
}
281290
]

src/bson.ts

+11
Original file line numberDiff line numberDiff line change
@@ -133,3 +133,14 @@ export function resolveBSONOptions(
133133
options?.enableUtf8Validation ?? parentOptions?.enableUtf8Validation ?? true
134134
};
135135
}
136+
137+
/** @internal */
138+
export function parseUtf8ValidationOption(options?: { enableUtf8Validation?: boolean }): {
139+
utf8: { writeErrors: false } | false;
140+
} {
141+
const enableUtf8Validation = options?.enableUtf8Validation;
142+
if (enableUtf8Validation === false) {
143+
return { utf8: false };
144+
}
145+
return { utf8: { writeErrors: false } };
146+
}

src/cmap/connection.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import { type DeserializeOptions } from 'bson';
12
import { type Readable, Transform, type TransformCallback } from 'stream';
23
import { clearTimeout, setTimeout } from 'timers';
34

@@ -487,7 +488,7 @@ export class Connection extends TypedEventEmitter<ConnectionEvents> {
487488

488489
// If `documentsReturnedIn` not set or raw is not enabled, use input bson options
489490
// Otherwise, support raw flag. Raw only works for cursors that hardcode firstBatch/nextBatch fields
490-
const bsonOptions =
491+
const bsonOptions: DeserializeOptions =
491492
options.documentsReturnedIn == null || !options.raw
492493
? options
493494
: {

src/cmap/wire_protocol/on_demand/document.ts

+13-4
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1+
import { type DeserializeOptions } from 'bson';
2+
13
import {
24
Binary,
3-
BSON,
45
type BSONElement,
56
BSONError,
6-
type BSONSerializeOptions,
77
BSONType,
8+
deserialize,
89
getBigInt64LE,
910
getFloat64LE,
1011
getInt32LE,
@@ -43,6 +44,14 @@ export type JSTypeOf = {
4344
/** @internal */
4445
type CachedBSONElement = { element: BSONElement; value: any | undefined };
4546

47+
/**
48+
* @internal
49+
*
50+
* Options for `OnDemandDocument.toObject()`. Validation is required to ensure
51+
* that callers provide utf8 validation options. */
52+
export type OnDemandDocumentDeserializeOptions = Omit<DeserializeOptions, 'validation'> &
53+
Required<Pick<DeserializeOptions, 'validation'>>;
54+
4655
/** @internal */
4756
export class OnDemandDocument {
4857
/**
@@ -329,8 +338,8 @@ export class OnDemandDocument {
329338
* Deserialize this object, DOES NOT cache result so avoid multiple invocations
330339
* @param options - BSON deserialization options
331340
*/
332-
public toObject(options?: BSONSerializeOptions): Record<string, any> {
333-
return BSON.deserialize(this.bson, {
341+
public toObject(options?: OnDemandDocumentDeserializeOptions): Record<string, any> {
342+
return deserialize(this.bson, {
334343
...options,
335344
index: this.offset,
336345
allowObjectSmallerThanBufferSize: true

src/cmap/wire_protocol/responses.ts

+15-16
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,24 @@
1+
import { type DeserializeOptions } from 'bson';
2+
13
import {
24
type BSONElement,
35
type BSONSerializeOptions,
46
BSONType,
57
type Document,
68
Long,
79
parseToElementsToArray,
10+
parseUtf8ValidationOption,
811
pluckBSONSerializeOptions,
912
type Timestamp
1013
} from '../../bson';
1114
import { MongoUnexpectedServerResponseError } from '../../error';
1215
import { type ClusterTime } from '../../sdam/common';
1316
import { decorateDecryptionResult, ns } from '../../utils';
14-
import { type JSTypeOf, OnDemandDocument } from './on_demand/document';
17+
import {
18+
type JSTypeOf,
19+
OnDemandDocument,
20+
type OnDemandDocumentDeserializeOptions
21+
} from './on_demand/document';
1522

1623
// eslint-disable-next-line no-restricted-syntax
1724
const enum BSONElementOffset {
@@ -113,7 +120,8 @@ export class MongoDBResponse extends OnDemandDocument {
113120
this.get('recoveryToken', BSONType.object)?.toObject({
114121
promoteValues: false,
115122
promoteLongs: false,
116-
promoteBuffers: false
123+
promoteBuffers: false,
124+
validation: { utf8: true }
117125
}) ?? null
118126
);
119127
}
@@ -170,20 +178,10 @@ export class MongoDBResponse extends OnDemandDocument {
170178
public override toObject(options?: BSONSerializeOptions): Record<string, any> {
171179
const exactBSONOptions = {
172180
...pluckBSONSerializeOptions(options ?? {}),
173-
validation: this.parseBsonSerializationOptions(options)
181+
validation: parseUtf8ValidationOption(options)
174182
};
175183
return super.toObject(exactBSONOptions);
176184
}
177-
178-
private parseBsonSerializationOptions(options?: { enableUtf8Validation?: boolean }): {
179-
utf8: { writeErrors: false } | false;
180-
} {
181-
const enableUtf8Validation = options?.enableUtf8Validation;
182-
if (enableUtf8Validation === false) {
183-
return { utf8: false };
184-
}
185-
return { utf8: { writeErrors: false } };
186-
}
187185
}
188186

189187
/** @internal */
@@ -267,12 +265,13 @@ export class CursorResponse extends MongoDBResponse {
267265
this.cursor.get('postBatchResumeToken', BSONType.object)?.toObject({
268266
promoteValues: false,
269267
promoteLongs: false,
270-
promoteBuffers: false
268+
promoteBuffers: false,
269+
validation: { utf8: true }
271270
}) ?? null
272271
);
273272
}
274273

275-
public shift(options?: BSONSerializeOptions): any {
274+
public shift(options: OnDemandDocumentDeserializeOptions): any {
276275
if (this.iterated >= this.batchSize) {
277276
return null;
278277
}
@@ -324,7 +323,7 @@ export class ExplainedCursorResponse extends CursorResponse {
324323
return this._length;
325324
}
326325

327-
override shift(options?: BSONSerializeOptions | undefined) {
326+
override shift(options?: DeserializeOptions) {
328327
if (this._length === 0) return null;
329328
this._length -= 1;
330329
return this.toObject(options);

src/cursor/abstract_cursor.ts

+15-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { Readable, Transform } from 'stream';
22

33
import { type BSONSerializeOptions, type Document, Long, pluckBSONSerializeOptions } from '../bson';
4+
import { type OnDemandDocumentDeserializeOptions } from '../cmap/wire_protocol/on_demand/document';
45
import { type CursorResponse } from '../cmap/wire_protocol/responses';
56
import {
67
MongoAPIError,
@@ -153,6 +154,9 @@ export abstract class AbstractCursor<
153154
/** @event */
154155
static readonly CLOSE = 'close' as const;
155156

157+
/** @internal */
158+
protected deserializationOptions: OnDemandDocumentDeserializeOptions;
159+
156160
/** @internal */
157161
protected constructor(
158162
client: MongoClient,
@@ -207,6 +211,13 @@ export abstract class AbstractCursor<
207211
} else {
208212
this.cursorSession = this.cursorClient.startSession({ owner: this, explicit: false });
209213
}
214+
215+
this.deserializationOptions = {
216+
...this.cursorOptions,
217+
validation: {
218+
utf8: options?.enableUtf8Validation === false ? false : true
219+
}
220+
};
210221
}
211222

212223
/**
@@ -289,7 +300,7 @@ export abstract class AbstractCursor<
289300
);
290301

291302
for (let count = 0; count < documentsToRead; count++) {
292-
const document = this.documents?.shift(this.cursorOptions);
303+
const document = this.documents?.shift(this.deserializationOptions);
293304
if (document != null) {
294305
bufferedDocs.push(document);
295306
}
@@ -390,7 +401,7 @@ export abstract class AbstractCursor<
390401
}
391402

392403
do {
393-
const doc = this.documents?.shift(this.cursorOptions);
404+
const doc = this.documents?.shift(this.deserializationOptions);
394405
if (doc != null) {
395406
if (this.transform != null) return await this.transformDocument(doc);
396407
return doc;
@@ -409,15 +420,15 @@ export abstract class AbstractCursor<
409420
throw new MongoCursorExhaustedError();
410421
}
411422

412-
let doc = this.documents?.shift(this.cursorOptions);
423+
let doc = this.documents?.shift(this.deserializationOptions);
413424
if (doc != null) {
414425
if (this.transform != null) return await this.transformDocument(doc);
415426
return doc;
416427
}
417428

418429
await this.fetchBatch();
419430

420-
doc = this.documents?.shift(this.cursorOptions);
431+
doc = this.documents?.shift(this.deserializationOptions);
421432
if (doc != null) {
422433
if (this.transform != null) return await this.transformDocument(doc);
423434
return doc;

src/cursor/aggregation_cursor.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,7 @@ export class AggregationCursor<TSchema = any> extends AbstractCursor<TSchema> {
7474
explain: verbosity ?? true
7575
})
7676
)
77-
).shift(this.aggregateOptions);
77+
).shift(this.deserializationOptions);
7878
}
7979

8080
/** Add a stage to the aggregation pipeline

src/cursor/find_cursor.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,7 @@ export class FindCursor<TSchema = any> extends AbstractCursor<TSchema> {
143143
explain: verbosity ?? true
144144
})
145145
)
146-
).shift(this.findOptions);
146+
).shift(this.deserializationOptions);
147147
}
148148

149149
/** Set the cursor query */

src/index.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -299,7 +299,11 @@ export type { ClientMetadata, ClientMetadataOptions } from './cmap/handshake/cli
299299
export type { ConnectionPoolMetrics } from './cmap/metrics';
300300
export type { StreamDescription, StreamDescriptionOptions } from './cmap/stream_description';
301301
export type { CompressorName } from './cmap/wire_protocol/compression';
302-
export type { JSTypeOf, OnDemandDocument } from './cmap/wire_protocol/on_demand/document';
302+
export type {
303+
JSTypeOf,
304+
OnDemandDocument,
305+
OnDemandDocumentDeserializeOptions
306+
} from './cmap/wire_protocol/on_demand/document';
303307
export type {
304308
CursorResponse,
305309
MongoDBResponse,

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

+1-2
Original file line numberDiff line numberDiff line change
@@ -875,7 +875,7 @@ describe('Change Streams', function () {
875875
await lastWrite().catch(() => null);
876876

877877
let counter = 0;
878-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
878+
879879
for await (const _ of changes) {
880880
counter += 1;
881881
if (counter === 2) {
@@ -1027,7 +1027,6 @@ describe('Change Streams', function () {
10271027
changeStream = collection.watch();
10281028

10291029
const loop = (async function () {
1030-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
10311030
for await (const _change of changeStream) {
10321031
return 'loop entered'; // loop should never be entered
10331032
}

0 commit comments

Comments
 (0)