Skip to content

Commit 4cc7af6

Browse files
authored
Merge pull request #15324 from Automattic/vkarpov15/async-savesubdocs
BREAKING CHANGE: use kareem@v3 promise-based execPre() for document hooks
2 parents e995641 + eb0368f commit 4cc7af6

16 files changed

+172
-184
lines changed

docs/migrating_to_9.md

+31
Original file line numberDiff line numberDiff line change
@@ -36,3 +36,34 @@ try {
3636
// Handle validation error
3737
}
3838
```
39+
40+
## Errors in middleware functions take priority over `next()` calls
41+
42+
In Mongoose 8.x, if a middleware function threw an error after calling `next()`, that error would be ignored.
43+
44+
```javascript
45+
schema.pre('save', function(next) {
46+
next();
47+
// In Mongoose 8, this error will not get reported, because you already called next()
48+
throw new Error('woops!');
49+
});
50+
```
51+
52+
In Mongoose 9, errors in the middleware function take priority, so the above `save()` would throw an error.
53+
54+
## `next()` no longer supports passing arguments to the next middleware
55+
56+
Previously, you could call `next(null, 'new arg')` in a hook and the args to the next middleware would get overwritten by 'new arg'.
57+
58+
```javascript
59+
schema.pre('save', function(next, options) {
60+
options; // options passed to `save()`
61+
next(null, 'new arg');
62+
});
63+
64+
schema.pre('save', function(next, arg) {
65+
arg; // In Mongoose 8, this would be 'new arg', overwrote the options passed to `save()`
66+
});
67+
```
68+
69+
In Mongoose 9, `next(null, 'new arg')` doesn't overwrite the args to the next middleware.

lib/aggregate.js

+22-20
Original file line numberDiff line numberDiff line change
@@ -800,18 +800,19 @@ Aggregate.prototype.explain = async function explain(verbosity) {
800800

801801
prepareDiscriminatorPipeline(this._pipeline, this._model.schema);
802802

803-
await new Promise((resolve, reject) => {
804-
model.hooks.execPre('aggregate', this, error => {
805-
if (error) {
806-
const _opts = { error: error };
807-
return model.hooks.execPost('aggregate', this, [null], _opts, error => {
808-
reject(error);
809-
});
810-
} else {
803+
try {
804+
await model.hooks.execPre('aggregate', this);
805+
} catch (error) {
806+
const _opts = { error: error };
807+
return await new Promise((resolve, reject) => {
808+
model.hooks.execPost('aggregate', this, [null], _opts, error => {
809+
if (error) {
810+
return reject(error);
811+
}
811812
resolve();
812-
}
813+
});
813814
});
814-
});
815+
}
815816

816817
const cursor = model.collection.aggregate(this._pipeline, this.options);
817818

@@ -1079,18 +1080,19 @@ Aggregate.prototype.exec = async function exec() {
10791080
prepareDiscriminatorPipeline(this._pipeline, this._model.schema);
10801081
stringifyFunctionOperators(this._pipeline);
10811082

1082-
await new Promise((resolve, reject) => {
1083-
model.hooks.execPre('aggregate', this, error => {
1084-
if (error) {
1085-
const _opts = { error: error };
1086-
return model.hooks.execPost('aggregate', this, [null], _opts, error => {
1087-
reject(error);
1088-
});
1089-
} else {
1083+
try {
1084+
await model.hooks.execPre('aggregate', this);
1085+
} catch (error) {
1086+
const _opts = { error: error };
1087+
return await new Promise((resolve, reject) => {
1088+
model.hooks.execPost('aggregate', this, [null], _opts, error => {
1089+
if (error) {
1090+
return reject(error);
1091+
}
10901092
resolve();
1091-
}
1093+
});
10921094
});
1093-
});
1095+
}
10941096

10951097
if (!this._pipeline.length) {
10961098
throw new MongooseError('Aggregate has empty pipeline');

lib/browserDocument.js

+2-10
Original file line numberDiff line numberDiff line change
@@ -97,16 +97,8 @@ Document.$emitter = new EventEmitter();
9797
* ignore
9898
*/
9999

100-
Document.prototype._execDocumentPreHooks = function _execDocumentPreHooks(opName) {
101-
return new Promise((resolve, reject) => {
102-
this._middleware.execPre(opName, this, [], (error) => {
103-
if (error != null) {
104-
reject(error);
105-
return;
106-
}
107-
resolve();
108-
});
109-
});
100+
Document.prototype._execDocumentPreHooks = async function _execDocumentPreHooks(opName) {
101+
return this._middleware.execPre(opName, this, []);
110102
};
111103

112104
/*!

lib/cursor/aggregationCursor.js

+14-24
Original file line numberDiff line numberDiff line change
@@ -63,34 +63,24 @@ util.inherits(AggregationCursor, Readable);
6363

6464
function _init(model, c, agg) {
6565
if (!model.collection.buffer) {
66-
model.hooks.execPre('aggregate', agg, function(err) {
67-
if (err != null) {
68-
_handlePreHookError(c, err);
69-
return;
70-
}
71-
if (typeof agg.options?.cursor?.transform === 'function') {
72-
c._transforms.push(agg.options.cursor.transform);
73-
}
74-
75-
c.cursor = model.collection.aggregate(agg._pipeline, agg.options || {});
76-
c.emit('cursor', c.cursor);
77-
});
66+
model.hooks.execPre('aggregate', agg).then(() => onPreComplete(null), err => onPreComplete(err));
7867
} else {
7968
model.collection.emitter.once('queue', function() {
80-
model.hooks.execPre('aggregate', agg, function(err) {
81-
if (err != null) {
82-
_handlePreHookError(c, err);
83-
return;
84-
}
69+
model.hooks.execPre('aggregate', agg).then(() => onPreComplete(null), err => onPreComplete(err));
70+
});
71+
}
8572

86-
if (typeof agg.options?.cursor?.transform === 'function') {
87-
c._transforms.push(agg.options.cursor.transform);
88-
}
73+
function onPreComplete(err) {
74+
if (err != null) {
75+
_handlePreHookError(c, err);
76+
return;
77+
}
78+
if (typeof agg.options?.cursor?.transform === 'function') {
79+
c._transforms.push(agg.options.cursor.transform);
80+
}
8981

90-
c.cursor = model.collection.aggregate(agg._pipeline, agg.options || {});
91-
c.emit('cursor', c.cursor);
92-
});
93-
});
82+
c.cursor = model.collection.aggregate(agg._pipeline, agg.options || {});
83+
c.emit('cursor', c.cursor);
9484
}
9585
}
9686

lib/cursor/queryCursor.js

+5-2
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,8 @@ function QueryCursor(query) {
4949
this._transforms = [];
5050
this.model = model;
5151
this.options = {};
52-
model.hooks.execPre('find', query, (err) => {
52+
53+
const onPreComplete = (err) => {
5354
if (err != null) {
5455
if (err instanceof kareem.skipWrappedFunction) {
5556
const resultValue = err.args[0];
@@ -91,7 +92,9 @@ function QueryCursor(query) {
9192
} else {
9293
_getRawCursor(query, this);
9394
}
94-
});
95+
};
96+
97+
model.hooks.execPre('find', query).then(() => onPreComplete(null), err => onPreComplete(err));
9598
}
9699

97100
util.inherits(QueryCursor, Readable);

lib/document.js

+4-12
Original file line numberDiff line numberDiff line change
@@ -847,8 +847,8 @@ function init(self, obj, doc, opts, prefix) {
847847
Document.prototype.updateOne = function updateOne(doc, options, callback) {
848848
const query = this.constructor.updateOne({ _id: this._doc._id }, doc, options);
849849
const self = this;
850-
query.pre(function queryPreUpdateOne(cb) {
851-
self.constructor._middleware.execPre('updateOne', self, [self], cb);
850+
query.pre(function queryPreUpdateOne() {
851+
return self.constructor._middleware.execPre('updateOne', self, [self]);
852852
});
853853
query.post(function queryPostUpdateOne(cb) {
854854
self.constructor._middleware.execPost('updateOne', self, [self], {}, cb);
@@ -2880,16 +2880,8 @@ function _pushNestedArrayPaths(val, paths, path) {
28802880
* ignore
28812881
*/
28822882

2883-
Document.prototype._execDocumentPreHooks = function _execDocumentPreHooks(opName, ...args) {
2884-
return new Promise((resolve, reject) => {
2885-
this.constructor._middleware.execPre(opName, this, [...args], (error) => {
2886-
if (error != null) {
2887-
reject(error);
2888-
return;
2889-
}
2890-
resolve();
2891-
});
2892-
});
2883+
Document.prototype._execDocumentPreHooks = async function _execDocumentPreHooks(opName, ...args) {
2884+
return this.constructor._middleware.execPre(opName, this, [...args]);
28932885
};
28942886

28952887
/*!

lib/helpers/model/applyHooks.js

+34-18
Original file line numberDiff line numberDiff line change
@@ -51,25 +51,11 @@ function applyHooks(model, schema, options) {
5151
for (const key of Object.keys(schema.paths)) {
5252
let type = schema.paths[key];
5353
let childModel = null;
54-
if (type.$isSingleNested) {
55-
childModel = type.caster;
56-
} else if (type.$isMongooseDocumentArray) {
57-
childModel = type.Constructor;
58-
} else if (type.instance === 'Array') {
59-
let curType = type;
60-
// Drill into nested arrays to check if nested array contains document array
61-
while (curType.instance === 'Array') {
62-
if (curType.$isMongooseDocumentArray) {
63-
childModel = curType.Constructor;
64-
type = curType;
65-
break;
66-
}
67-
curType = curType.getEmbeddedSchemaType();
68-
}
6954

70-
if (childModel == null) {
71-
continue;
72-
}
55+
const result = findChildModel(type);
56+
if (result) {
57+
childModel = result.childModel;
58+
type = result.type;
7359
} else {
7460
continue;
7561
}
@@ -164,3 +150,33 @@ function applyHooks(model, schema, options) {
164150
createWrapper(method, originalMethod, null, customMethodOptions);
165151
}
166152
}
153+
154+
/**
155+
* Check if there is an embedded schematype in the given schematype. Handles drilling down into primitive
156+
* arrays and maps in case of array of array of subdocs or map of subdocs.
157+
*
158+
* @param {SchemaType} curType
159+
* @returns {{ childModel: Model | typeof Subdocument, curType: SchemaType } | null}
160+
*/
161+
162+
function findChildModel(curType) {
163+
if (curType.$isSingleNested) {
164+
return { childModel: curType.caster, type: curType };
165+
}
166+
if (curType.$isMongooseDocumentArray) {
167+
return { childModel: curType.Constructor, type: curType };
168+
}
169+
if (curType.instance === 'Array') {
170+
const embedded = curType.getEmbeddedSchemaType();
171+
if (embedded) {
172+
return findChildModel(embedded);
173+
}
174+
}
175+
if (curType.instance === 'Map') {
176+
const mapType = curType.getEmbeddedSchemaType();
177+
if (mapType) {
178+
return findChildModel(mapType);
179+
}
180+
}
181+
return null;
182+
}

lib/helpers/model/applyStaticHooks.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -42,10 +42,10 @@ module.exports = function applyStaticHooks(model, hooks, statics) {
4242
const cb = typeof lastArg === 'function' ? lastArg : null;
4343
const args = Array.prototype.slice.
4444
call(arguments, 0, cb == null ? numArgs : numArgs - 1);
45-
// Special case: can't use `Kareem#wrap()` because it doesn't currently
46-
// support wrapped functions that return a promise.
4745
return promiseOrCallback(cb, callback => {
48-
hooks.execPre(key, model, args, function(err) {
46+
hooks.execPre(key, model, args).then(() => onPreComplete(null), err => onPreComplete(err));
47+
48+
function onPreComplete(err) {
4949
if (err != null) {
5050
return callback(err);
5151
}
@@ -72,7 +72,7 @@ module.exports = function applyStaticHooks(model, hooks, statics) {
7272
callback(null, res);
7373
});
7474
}
75-
});
75+
}
7676
}, model.events);
7777
};
7878
}

lib/model.js

+20-40
Original file line numberDiff line numberDiff line change
@@ -810,19 +810,16 @@ Model.prototype.deleteOne = function deleteOne(options) {
810810
}
811811
}
812812

813-
query.pre(function queryPreDeleteOne(cb) {
814-
self.constructor._middleware.execPre('deleteOne', self, [self], cb);
813+
query.pre(function queryPreDeleteOne() {
814+
return self.constructor._middleware.execPre('deleteOne', self, [self]);
815815
});
816-
query.pre(function callSubdocPreHooks(cb) {
817-
each(self.$getAllSubdocs(), (subdoc, cb) => {
818-
subdoc.constructor._middleware.execPre('deleteOne', subdoc, [subdoc], cb);
819-
}, cb);
816+
query.pre(function callSubdocPreHooks() {
817+
return Promise.all(self.$getAllSubdocs().map(subdoc => subdoc.constructor._middleware.execPre('deleteOne', subdoc, [subdoc])));
820818
});
821-
query.pre(function skipIfAlreadyDeleted(cb) {
819+
query.pre(function skipIfAlreadyDeleted() {
822820
if (self.$__.isDeleted) {
823-
return cb(Kareem.skipWrappedFunction());
821+
throw new Kareem.skipWrappedFunction();
824822
}
825-
return cb();
826823
});
827824
query.post(function callSubdocPostHooks(cb) {
828825
each(self.$getAllSubdocs(), (subdoc, cb) => {
@@ -1185,16 +1182,11 @@ Model.createCollection = async function createCollection(options) {
11851182
throw new MongooseError('Model.createCollection() no longer accepts a callback');
11861183
}
11871184

1188-
const shouldSkip = await new Promise((resolve, reject) => {
1189-
this.hooks.execPre('createCollection', this, [options], (err) => {
1190-
if (err != null) {
1191-
if (err instanceof Kareem.skipWrappedFunction) {
1192-
return resolve(true);
1193-
}
1194-
return reject(err);
1195-
}
1196-
resolve();
1197-
});
1185+
const shouldSkip = await this.hooks.execPre('createCollection', this, [options]).catch(err => {
1186+
if (err instanceof Kareem.skipWrappedFunction) {
1187+
return true;
1188+
}
1189+
throw err;
11981190
});
11991191

12001192
const collectionOptions = this &&
@@ -3380,17 +3372,13 @@ Model.bulkWrite = async function bulkWrite(ops, options) {
33803372
}
33813373
options = options || {};
33823374

3383-
const shouldSkip = await new Promise((resolve, reject) => {
3384-
this.hooks.execPre('bulkWrite', this, [ops, options], (err) => {
3385-
if (err != null) {
3386-
if (err instanceof Kareem.skipWrappedFunction) {
3387-
return resolve(err);
3388-
}
3389-
return reject(err);
3390-
}
3391-
resolve();
3392-
});
3393-
});
3375+
const shouldSkip = await this.hooks.execPre('bulkWrite', this, [ops, options]).catch(err => {
3376+
if (err instanceof Kareem.skipWrappedFunction) {
3377+
return err;
3378+
}
3379+
throw err;
3380+
}
3381+
);
33943382

33953383
if (shouldSkip) {
33963384
return shouldSkip.args[0];
@@ -3621,16 +3609,8 @@ Model.bulkSave = async function bulkSave(documents, options) {
36213609
return bulkWriteResult;
36223610
};
36233611

3624-
function buildPreSavePromise(document, options) {
3625-
return new Promise((resolve, reject) => {
3626-
document.schema.s.hooks.execPre('save', document, [options], (err) => {
3627-
if (err) {
3628-
reject(err);
3629-
return;
3630-
}
3631-
resolve();
3632-
});
3633-
});
3612+
async function buildPreSavePromise(document, options) {
3613+
return document.schema.s.hooks.execPre('save', document, [options]);
36343614
}
36353615

36363616
function handleSuccessfulWrite(document) {

0 commit comments

Comments
 (0)