Skip to content

Commit 61c09b8

Browse files
committedDec 26, 2022
Merge branch 'master' into vkarpov15/gh-12621
2 parents 97e5a0e + 2bf15d5 commit 61c09b8

14 files changed

+175
-51
lines changed
 

‎docs/queries.md

+83
Original file line numberDiff line numberDiff line change
@@ -241,6 +241,89 @@ const queryRes = await Person.findOne({ _id: idString });
241241
const aggRes = await Person.aggregate([{ $match: { _id: idString } }])
242242
```
243243

244+
<h3 id="sorting"><a href="#sorting">Sorting</a></h3>
245+
246+
[Sorting](/docs/api.html#query_Query-sort) is how you can ensure you query results come back in the desired order.
247+
248+
```javascript
249+
const personSchema = new mongoose.Schema({
250+
age: Number
251+
});
252+
253+
const Person = mongoose.model('Person', personSchema);
254+
for (let i = 0; i < 10; i++) {
255+
await Person.create({ age: i });
256+
}
257+
258+
await Person.find().sort({ age: -1 }); // returns age starting from 10 as the first entry
259+
await Person.find().sort({ age: 1 }); // returns age starting from 0 as the first entry
260+
```
261+
262+
When sorting with mutiple fields, the order of the sort keys determines what key MongoDB server sorts by first.
263+
264+
```javascript
265+
const personSchema = new mongoose.Schema({
266+
age: Number,
267+
name: String,
268+
weight: Number
269+
});
270+
271+
const Person = mongoose.model('Person', personSchema);
272+
const iterations = 5;
273+
for (let i = 0; i < iterations; i++) {
274+
await Person.create({
275+
age: Math.abs(2-i),
276+
name: 'Test'+i,
277+
weight: Math.floor(Math.random() * 100) + 1
278+
});
279+
}
280+
281+
await Person.find().sort({ age: 1, weight: -1 }); // returns age starting from 0, but while keeping that order will then sort by weight.
282+
```
283+
284+
You can view the output of a single run of this block below.
285+
As you can see, age is sorted from 0 to 2 but when age is equal, sorts by weight.
286+
287+
```javascript
288+
[
289+
{
290+
_id: new ObjectId("63a335a6b9b6a7bfc186cb37"),
291+
age: 0,
292+
name: 'Test2',
293+
weight: 67,
294+
__v: 0
295+
},
296+
{
297+
_id: new ObjectId("63a335a6b9b6a7bfc186cb35"),
298+
age: 1,
299+
name: 'Test1',
300+
weight: 99,
301+
__v: 0
302+
},
303+
{
304+
_id: new ObjectId("63a335a6b9b6a7bfc186cb39"),
305+
age: 1,
306+
name: 'Test3',
307+
weight: 73,
308+
__v: 0
309+
},
310+
{
311+
_id: new ObjectId("63a335a6b9b6a7bfc186cb33"),
312+
age: 2,
313+
name: 'Test0',
314+
weight: 65,
315+
__v: 0
316+
},
317+
{
318+
_id: new ObjectId("63a335a6b9b6a7bfc186cb3b"),
319+
age: 2,
320+
name: 'Test4',
321+
weight: 62,
322+
__v: 0
323+
}
324+
]
325+
```
326+
244327
<h3 id="next"><a href="#next">Next Up</a></h3>
245328

246329
Now that we've covered `Queries`, let's take a look at [Validation](validation.html).

‎docs/search.js

+11-4
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ const filemap = require('./source');
66
const fs = require('fs');
77
const pug = require('pug');
88
const mongoose = require('../');
9+
let { version } = require('../package.json');
910

1011
const { marked: markdown } = require('marked');
1112
const highlight = require('highlight.js');
@@ -15,10 +16,14 @@ markdown.setOptions({
1516
}
1617
});
1718

19+
// 5.13.5 -> 5.x, 6.8.2 -> 6.x, etc.
20+
version = version.slice(0, version.indexOf('.')) + '.x';
21+
1822
const contentSchema = new mongoose.Schema({
1923
title: { type: String, required: true },
2024
body: { type: String, required: true },
21-
url: { type: String, required: true }
25+
url: { type: String, required: true },
26+
version: { type: String, required: true, default: version }
2227
});
2328
contentSchema.index({ title: 'text', body: 'text' });
2429
const Content = mongoose.model('Content', contentSchema, 'Content');
@@ -28,7 +33,6 @@ const files = Object.keys(filemap);
2833

2934
for (const filename of files) {
3035
const file = filemap[filename];
31-
console.log(file)
3236
if (file.api) {
3337
// API docs are special, raw content is in the `docs` property
3438
for (const _class of file.docs) {
@@ -115,13 +119,16 @@ run().catch(error => console.error(error.stack));
115119
async function run() {
116120
await mongoose.connect(config.uri, { dbName: 'mongoose' });
117121

118-
await Content.deleteMany({});
122+
await Content.deleteMany({ version });
119123
for (const content of contents) {
124+
if (version !== '6.x') {
125+
content.url = `/docs/${version}/docs${content.url}`;
126+
}
120127
await content.save();
121128
}
122129

123130
const results = await Content.
124-
find({ $text: { $search: 'validate' } }, { score: { $meta: 'textScore' } }).
131+
find({ $text: { $search: 'validate' }, version }, { score: { $meta: 'textScore' } }).
125132
sort({ score: { $meta: 'textScore' } }).
126133
limit(10);
127134

‎lib/helpers/model/discriminator.js

+3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
'use strict';
22

33
const Mixed = require('../../schema/mixed');
4+
const applyBuiltinPlugins = require('../schema/applyBuiltinPlugins');
45
const defineKey = require('../document/compile').defineKey;
56
const get = require('../get');
67
const utils = require('../../utils');
@@ -40,6 +41,8 @@ module.exports = function discriminator(model, name, schema, tiedValue, applyPlu
4041
model.base._applyPlugins(schema, {
4142
skipTopLevel: !applyPluginsToDiscriminators
4243
});
44+
} else if (!mergeHooks) {
45+
applyBuiltinPlugins(schema);
4346
}
4447

4548
const key = model.schema.options.discriminatorKey;

‎lib/helpers/query/castUpdate.js

+5-3
Original file line numberDiff line numberDiff line change
@@ -85,9 +85,11 @@ module.exports = function castUpdate(schema, obj, options, context, filter) {
8585
val = ret[op];
8686
hasDollarKey = hasDollarKey || op.startsWith('$');
8787
const toUnset = {};
88-
for (const key of Object.keys(val)) {
89-
if (val[key] === undefined) {
90-
toUnset[key] = 1;
88+
if (val != null) {
89+
for (const key of Object.keys(val)) {
90+
if (val[key] === undefined) {
91+
toUnset[key] = 1;
92+
}
9193
}
9294
}
9395

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
'use strict';
2+
3+
const builtinPlugins = require('../../plugins');
4+
5+
module.exports = function applyBuiltinPlugins(schema) {
6+
for (const plugin of Object.values(builtinPlugins)) {
7+
plugin(schema, { deduplicate: true });
8+
}
9+
schema.plugins = Object.values(builtinPlugins).
10+
map(fn => ({ fn, opts: { deduplicate: true } })).
11+
concat(schema.plugins);
12+
};

‎lib/index.js

+2-12
Original file line numberDiff line numberDiff line change
@@ -19,21 +19,17 @@ const Types = require('./types');
1919
const Query = require('./query');
2020
const Model = require('./model');
2121
const applyPlugins = require('./helpers/schema/applyPlugins');
22+
const builtinPlugins = require('./plugins');
2223
const driver = require('./driver');
2324
const promiseOrCallback = require('./helpers/promiseOrCallback');
2425
const legacyPluralize = require('./helpers/pluralize');
2526
const utils = require('./utils');
2627
const pkg = require('../package.json');
2728
const cast = require('./cast');
28-
const removeSubdocs = require('./plugins/removeSubdocs');
29-
const saveSubdocs = require('./plugins/saveSubdocs');
30-
const trackTransaction = require('./plugins/trackTransaction');
31-
const validateBeforeSave = require('./plugins/validateBeforeSave');
3229

3330
const Aggregate = require('./aggregate');
3431
const PromiseProvider = require('./promise_provider');
3532
const printStrictQueryWarning = require('./helpers/printStrictQueryWarning');
36-
const shardingPlugin = require('./plugins/sharding');
3733
const trusted = require('./helpers/query/trusted').trusted;
3834
const sanitizeFilter = require('./helpers/query/sanitizeFilter');
3935
const isBsonType = require('./helpers/isBsonType');
@@ -108,13 +104,7 @@ function Mongoose(options) {
108104
configurable: false,
109105
enumerable: true,
110106
writable: false,
111-
value: [
112-
[saveSubdocs, { deduplicate: true }],
113-
[validateBeforeSave, { deduplicate: true }],
114-
[shardingPlugin, { deduplicate: true }],
115-
[removeSubdocs, { deduplicate: true }],
116-
[trackTransaction, { deduplicate: true }]
117-
]
107+
value: Object.values(builtinPlugins).map(plugin => ([plugin, { deduplicate: true }]))
118108
});
119109
}
120110

‎lib/plugins/clearValidating.js

-28
This file was deleted.

‎lib/plugins/index.js

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
'use strict';
2+
3+
exports.removeSubdocs = require('./removeSubdocs');
4+
exports.saveSubdocs = require('./saveSubdocs');
5+
exports.sharding = require('./sharding');
6+
exports.trackTransaction = require('./trackTransaction');
7+
exports.validateBeforeSave = require('./validateBeforeSave');

‎lib/schema.js

+3
Original file line numberDiff line numberDiff line change
@@ -1250,6 +1250,9 @@ Schema.prototype.interpretAsType = function(path, obj, options) {
12501250
if (options.hasOwnProperty('strict')) {
12511251
childSchemaOptions.strict = options.strict;
12521252
}
1253+
if (options.hasOwnProperty('strictQuery')) {
1254+
childSchemaOptions.strictQuery = options.strictQuery;
1255+
}
12531256

12541257
if (this._userProvidedOptions.hasOwnProperty('_id')) {
12551258
childSchemaOptions._id = this._userProvidedOptions._id;

‎test/model.discriminator.test.js

+24
Original file line numberDiff line numberDiff line change
@@ -2077,4 +2077,28 @@ describe('model', function() {
20772077
schema.pre('save', function testHook12604() {});
20782078
}
20792079
});
2080+
2081+
it('applies built-in plugins if mergePlugins and mergeHooks disabled (gh-12696) (gh-12604)', async function() {
2082+
const shapeDef = { name: String };
2083+
const shapeSchema = Schema(shapeDef, { discriminatorKey: 'kind' });
2084+
2085+
const Shape = db.model('Test', shapeSchema);
2086+
2087+
let subdocSaveCalls = 0;
2088+
const nestedSchema = Schema({ test: String });
2089+
nestedSchema.pre('save', function() {
2090+
++subdocSaveCalls;
2091+
});
2092+
2093+
const squareSchema = Schema({ ...shapeDef, nested: nestedSchema });
2094+
const Square = Shape.discriminator(
2095+
'Square',
2096+
squareSchema,
2097+
{ mergeHooks: false, mergePlugins: false }
2098+
);
2099+
2100+
assert.equal(subdocSaveCalls, 0);
2101+
await Square.create({ nested: { test: 'foo' } });
2102+
assert.equal(subdocSaveCalls, 1);
2103+
});
20802104
});

‎test/model.test.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -6916,10 +6916,12 @@ describe('Model', function() {
69166916
granularity: 'hours'
69176917
},
69186918
autoCreate: false,
6919+
autoIndex: false,
69196920
expireAfterSeconds: 86400
69206921
});
69216922

6922-
const Test = db.model('Test', schema);
6923+
const Test = db.model('Test', schema, 'Test');
6924+
await Test.init();
69236925

69246926
await Test.collection.drop().catch(() => {});
69256927
await Test.createCollection();

‎test/query.test.js

+12-2
Original file line numberDiff line numberDiff line change
@@ -1564,7 +1564,7 @@ describe('Query', function() {
15641564
const Product = db.model('Product', productSchema);
15651565
Product.create(
15661566
{ numbers: [3, 4, 5] },
1567-
{ strings: 'hi there'.split(' ') }, function(err, doc1, doc2) {
1567+
{ strings: 'hi there'.split(' '), w: 'majority' }, function(err, doc1, doc2) {
15681568
assert.ifError(err);
15691569
Product.find().setOptions({ limit: 1, sort: { _id: -1 }, read: 'n' }).exec(function(err, docs) {
15701570
assert.ifError(err);
@@ -4314,7 +4314,7 @@ describe('Query', function() {
43144314
assert.strictEqual(found[0].title, 'burrito bowl');
43154315
});
43164316

4317-
it('update operation should remove fields set to undefined (gh-12794)', async() => {
4317+
it('update operation should remove fields set to undefined (gh-12794) (gh-12821)', async function() {
43184318
const m = new mongoose.Mongoose();
43194319

43204320
await m.connect(start.uri);
@@ -4338,5 +4338,15 @@ describe('Query', function() {
43384338
).lean();
43394339

43404340
assert.ok('title' in updatedDoc === false);
4341+
4342+
const replacedDoc = await Test.findOneAndReplace(
4343+
{
4344+
_id: doc._id
4345+
},
4346+
{ title: undefined },
4347+
{ returnOriginal: false }
4348+
).lean();
4349+
4350+
assert.ok('title' in replacedDoc === false);
43414351
});
43424352
});

‎test/schema.documentarray.test.js

+9
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,15 @@ describe('schema.documentarray', function() {
7575
done();
7676
});
7777

78+
it('propagates strictQuery to implicitly created schemas (gh-12796)', function() {
79+
const schema = new Schema({
80+
arr: [{ name: String }]
81+
}, { strictQuery: 'throw' });
82+
83+
assert.equal(schema.childSchemas.length, 1);
84+
assert.equal(schema.childSchemas[0].schema.options.strictQuery, 'throw');
85+
});
86+
7887
it('supports set with array of document arrays (gh-7799)', function() {
7988
const subSchema = new Schema({
8089
title: String

‎test/schema.test.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -2833,7 +2833,7 @@ describe('schema', function() {
28332833
assert.equal(entry instanceof mongoose.Document, false);
28342834
});
28352835

2836-
it('disallows setting special properties with `add()` or constructor (gh-12085)', async function() {
2836+
it('disallows setting special properties with `add()` or constructor (gh-12085)', function() {
28372837
const maliciousPayload = '{"__proto__.toString": "Number"}';
28382838

28392839
assert.throws(() => {

0 commit comments

Comments
 (0)
Please sign in to comment.