Skip to content

Commit 3ba9778

Browse files
authored
Merge pull request #14423 from Automattic/vkarpov15/gh-14400
types(model): make `bulkWrite()` types more flexible to account for casting
2 parents cce65d1 + 7835396 commit 3ba9778

File tree

4 files changed

+111
-12
lines changed

4 files changed

+111
-12
lines changed

lib/model.js

+12-2
Original file line numberDiff line numberDiff line change
@@ -3375,6 +3375,7 @@ function _setIsNew(doc, val) {
33753375
* trip to MongoDB.
33763376
*
33773377
* Mongoose will perform casting on all operations you provide.
3378+
* The only exception is [setting the `update` operator for `updateOne` or `updateMany` to a pipeline](https://www.mongodb.com/docs/manual/reference/method/db.collection.bulkWrite/#updateone-and-updatemany): Mongoose does **not** cast update pipelines.
33783379
*
33793380
* This function does **not** trigger any middleware, neither `save()`, nor `update()`.
33803381
* If you need to trigger
@@ -3410,6 +3411,15 @@ function _setIsNew(doc, val) {
34103411
* console.log(res.insertedCount, res.modifiedCount, res.deletedCount);
34113412
* });
34123413
*
3414+
* // Mongoose does **not** cast update pipelines, so no casting for the `update` option below.
3415+
* // Mongoose does still cast `filter`
3416+
* await Character.bulkWrite([{
3417+
* updateOne: {
3418+
* filter: { name: 'Annika Hansen' },
3419+
* update: [{ $set: { name: 7 } }] // Array means update pipeline, so Mongoose skips casting
3420+
* }
3421+
* }]);
3422+
*
34133423
* The [supported operations](https://www.mongodb.com/docs/manual/reference/method/db.collection.bulkWrite/#db.collection.bulkWrite) are:
34143424
*
34153425
* - `insertOne`
@@ -3939,7 +3949,7 @@ Model.hydrate = function(obj, projection, options) {
39393949
* - `updateMany()`
39403950
*
39413951
* @param {Object} filter
3942-
* @param {Object|Array} update
3952+
* @param {Object|Array} update. If array, this update will be treated as an update pipeline and not casted.
39433953
* @param {Object} [options] optional see [`Query.prototype.setOptions()`](https://mongoosejs.com/docs/api/query.html#Query.prototype.setOptions())
39443954
* @param {Boolean|String} [options.strict] overwrites the schema's [strict mode option](https://mongoosejs.com/docs/guide.html#strict)
39453955
* @param {Boolean} [options.upsert=false] if true, and no documents found, insert a new document
@@ -3979,7 +3989,7 @@ Model.updateMany = function updateMany(conditions, doc, options) {
39793989
* - `updateOne()`
39803990
*
39813991
* @param {Object} filter
3982-
* @param {Object|Array} update
3992+
* @param {Object|Array} update. If array, this update will be treated as an update pipeline and not casted.
39833993
* @param {Object} [options] optional see [`Query.prototype.setOptions()`](https://mongoosejs.com/docs/api/query.html#Query.prototype.setOptions())
39843994
* @param {Boolean|String} [options.strict] overwrites the schema's [strict mode option](https://mongoosejs.com/docs/guide.html#strict)
39853995
* @param {Boolean} [options.upsert=false] if true, and no documents found, insert a new document

lib/query.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -3880,7 +3880,7 @@ Query.prototype._replaceOne = async function _replaceOne() {
38803880
* - `updateMany()`
38813881
*
38823882
* @param {Object} [filter]
3883-
* @param {Object|Array} [update] the update command
3883+
* @param {Object|Array} [update] the update command. If array, this update will be treated as an update pipeline and not casted.
38843884
* @param {Object} [options]
38853885
* @param {Boolean} [options.multipleCastError] by default, mongoose only returns the first error that occurred in casting the query. Turn on this option to aggregate all the cast errors.
38863886
* @param {Boolean|String} [options.strict] overwrites the schema's [strict mode option](https://mongoosejs.com/docs/guide.html#strict)
@@ -3950,7 +3950,7 @@ Query.prototype.updateMany = function(conditions, doc, options, callback) {
39503950
* - `updateOne()`
39513951
*
39523952
* @param {Object} [filter]
3953-
* @param {Object|Array} [update] the update command
3953+
* @param {Object|Array} [update] the update command. If array, this update will be treated as an update pipeline and not casted.
39543954
* @param {Object} [options]
39553955
* @param {Boolean} [options.multipleCastError] by default, mongoose only returns the first error that occurred in casting the query. Turn on this option to aggregate all the cast errors.
39563956
* @param {Boolean|String} [options.strict] overwrites the schema's [strict mode option](https://mongoosejs.com/docs/guide.html#strict)

test/types/models.test.ts

+16
Original file line numberDiff line numberDiff line change
@@ -897,6 +897,22 @@ function gh4727() {
897897
const company = { _id: new mongoose.Types.ObjectId(), name: 'Booster', users: [users[0]] };
898898

899899
return Company.hydrate(company, {}, { hydratedPopulatedDocs: true });
900+
}
901+
902+
async function gh14440() {
903+
const testSchema = new Schema({
904+
dateProperty: { type: Date }
905+
});
900906

907+
const TestModel = model('Test', testSchema);
901908

909+
const doc = new TestModel();
910+
await TestModel.bulkWrite([
911+
{
912+
updateOne: {
913+
filter: { _id: doc._id },
914+
update: { dateProperty: (new Date('2023-06-01')).toISOString() }
915+
}
916+
}
917+
]);
902918
}

types/models.d.ts

+81-8
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,85 @@ declare module 'mongoose' {
156156

157157
const Model: Model<any>;
158158

159+
export type AnyBulkWriteOperation<TSchema = AnyObject> = {
160+
insertOne: InsertOneModel<TSchema>;
161+
} | {
162+
replaceOne: ReplaceOneModel<TSchema>;
163+
} | {
164+
updateOne: UpdateOneModel<TSchema>;
165+
} | {
166+
updateMany: UpdateManyModel<TSchema>;
167+
} | {
168+
deleteOne: DeleteOneModel<TSchema>;
169+
} | {
170+
deleteMany: DeleteManyModel<TSchema>;
171+
};
172+
173+
export interface InsertOneModel<TSchema> {
174+
document: mongodb.OptionalId<TSchema>
175+
}
176+
177+
export interface ReplaceOneModel<TSchema = AnyObject> {
178+
/** The filter to limit the replaced document. */
179+
filter: FilterQuery<TSchema>;
180+
/** The document with which to replace the matched document. */
181+
replacement: mongodb.WithoutId<TSchema>;
182+
/** Specifies a collation. */
183+
collation?: mongodb.CollationOptions;
184+
/** The index to use. If specified, then the query system will only consider plans using the hinted index. */
185+
hint?: mongodb.Hint;
186+
/** When true, creates a new document if no document matches the query. */
187+
upsert?: boolean;
188+
}
189+
190+
export interface UpdateOneModel<TSchema = AnyObject> {
191+
/** The filter to limit the updated documents. */
192+
filter: FilterQuery<TSchema>;
193+
/** A document or pipeline containing update operators. */
194+
update: UpdateQuery<TSchema>;
195+
/** A set of filters specifying to which array elements an update should apply. */
196+
arrayFilters?: AnyObject[];
197+
/** Specifies a collation. */
198+
collation?: mongodb.CollationOptions;
199+
/** The index to use. If specified, then the query system will only consider plans using the hinted index. */
200+
hint?: mongodb.Hint;
201+
/** When true, creates a new document if no document matches the query. */
202+
upsert?: boolean;
203+
}
204+
205+
export interface UpdateManyModel<TSchema = AnyObject> {
206+
/** The filter to limit the updated documents. */
207+
filter: FilterQuery<TSchema>;
208+
/** A document or pipeline containing update operators. */
209+
update: UpdateQuery<TSchema>;
210+
/** A set of filters specifying to which array elements an update should apply. */
211+
arrayFilters?: AnyObject[];
212+
/** Specifies a collation. */
213+
collation?: mongodb.CollationOptions;
214+
/** The index to use. If specified, then the query system will only consider plans using the hinted index. */
215+
hint?: mongodb.Hint;
216+
/** When true, creates a new document if no document matches the query. */
217+
upsert?: boolean;
218+
}
219+
220+
export interface DeleteOneModel<TSchema = AnyObject> {
221+
/** The filter to limit the deleted documents. */
222+
filter: FilterQuery<TSchema>;
223+
/** Specifies a collation. */
224+
collation?: mongodb.CollationOptions;
225+
/** The index to use. If specified, then the query system will only consider plans using the hinted index. */
226+
hint?: mongodb.Hint;
227+
}
228+
229+
export interface DeleteManyModel<TSchema = AnyObject> {
230+
/** The filter to limit the deleted documents. */
231+
filter: FilterQuery<TSchema>;
232+
/** Specifies a collation. */
233+
collation?: mongodb.CollationOptions;
234+
/** The index to use. If specified, then the query system will only consider plans using the hinted index. */
235+
hint?: mongodb.Hint;
236+
}
237+
159238
/**
160239
* Models are fancy constructors compiled from `Schema` definitions.
161240
* An instance of a model is called a document.
@@ -201,17 +280,11 @@ declare module 'mongoose' {
201280
* round trip to the MongoDB server.
202281
*/
203282
bulkWrite<DocContents = TRawDocType>(
204-
writes: Array<
205-
mongodb.AnyBulkWriteOperation<
206-
DocContents extends mongodb.Document ? DocContents : any
207-
> & MongooseBulkWritePerWriteOptions>,
283+
writes: Array<AnyBulkWriteOperation<DocContents extends Document ? any : (DocContents extends {} ? DocContents : any)>>,
208284
options: mongodb.BulkWriteOptions & MongooseBulkWriteOptions & { ordered: false }
209285
): Promise<mongodb.BulkWriteResult & { mongoose?: { validationErrors: Error[] } }>;
210286
bulkWrite<DocContents = TRawDocType>(
211-
writes: Array<
212-
mongodb.AnyBulkWriteOperation<
213-
DocContents extends mongodb.Document ? DocContents : any
214-
> & MongooseBulkWritePerWriteOptions>,
287+
writes: Array<AnyBulkWriteOperation<DocContents extends Document ? any : (DocContents extends {} ? DocContents : any)>>,
215288
options?: mongodb.BulkWriteOptions & MongooseBulkWriteOptions
216289
): Promise<mongodb.BulkWriteResult>;
217290

0 commit comments

Comments
 (0)