Skip to content

Commit 69405b2

Browse files
committed
Merge branch '6.x' into vkarpov15/gh-13191-2
2 parents f143485 + e9eb8ab commit 69405b2

14 files changed

+95
-28
lines changed

.github/workflows/test.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,7 @@ jobs:
9696
- name: Setup Deno
9797
uses: denoland/setup-deno@v1
9898
with:
99-
deno-version: v1.33.x
99+
deno-version: v1.34.x
100100
- run: deno --version
101101
- run: npm install
102102
- name: Run Deno tests

CHANGELOG.md

+9
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,12 @@
1+
6.11.3 / 2023-07-11
2+
===================
3+
* fix: avoid prototype pollution on init
4+
* fix(schema): correctly handle uuids with populate() #13317 #13595
5+
6+
6.11.2 / 2023-06-08
7+
===================
8+
* fix(cursor): allow find middleware to modify query cursor options #13476 #13453 #13435
9+
110
6.11.1 / 2023-05-08
211
===================
312
* fix(query): apply schema-level paths before calculating projection for findOneAndUpdate() #13348 #13340

docs/layout.pug

+1-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,7 @@ html(lang='en')
143143
li.pure-menu-item.sub-item
144144
a.pure-menu-link(href=`${versions.versionedPath}/docs/api/virtualtype.html`, class=outputUrl === `${versions.versionedPath}/docs/api/virtualtype.html` ? 'selected' : '') VirtualType
145145
li.pure-menu-item
146-
a.pure-menu-link(href=`${versions.versionedPath}/docs/migrating_to_7.html`, class=outputUrl === `${versions.versionedPath}/docs/migrating_to_7.html` ? 'selected' : '') Migration Guide
146+
a.pure-menu-link(href=`${versions.versionedPath}/docs/migrating_to_6.html`, class=outputUrl === `${versions.versionedPath}/docs/migrating_to_7.html` ? 'selected' : '') Migration Guide
147147
li.pure-menu-item
148148
a.pure-menu-link(href=`${versions.versionedPath}/docs/compatibility.html`, class=outputUrl === `${versions.versionedPath}/docs/compatibility.html` ? 'selected' : '') Version Compatibility
149149
li.pure-menu-item

lib/cursor/QueryCursor.js

+1
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,7 @@ function QueryCursor(query, options) {
6666
// Max out the number of documents we'll populate in parallel at 5000.
6767
this.options._populateBatchSize = Math.min(this.options.batchSize, 5000);
6868
}
69+
Object.assign(this.options, query._optionsForExec());
6970
model.collection.find(query._conditions, this.options, (err, cursor) => {
7071
if (err != null) {
7172
_this._markError(err);

lib/document.js

+4
Original file line numberDiff line numberDiff line change
@@ -740,6 +740,10 @@ function init(self, obj, doc, opts, prefix) {
740740

741741
function _init(index) {
742742
i = keys[index];
743+
// avoid prototype pollution
744+
if (i === '__proto__' || i === 'constructor') {
745+
return;
746+
}
743747
path = prefix + i;
744748
schemaType = docSchema.path(path);
745749

lib/schema/uuid.js

+10-8
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@ const MongooseBuffer = require('../types/buffer');
88
const SchemaType = require('../schematype');
99
const CastError = SchemaType.CastError;
1010
const utils = require('../utils');
11-
const isBsonType = require('../helpers/isBsonType');
1211
const handleBitwiseOperator = require('./operators/bitwise');
1312

1413
const UUID_FORMAT = /[0-9a-f]{8}-[0-9a-f]{4}-[0-9][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/i;
@@ -86,7 +85,13 @@ function binaryToString(uuidBin) {
8685

8786
function SchemaUUID(key, options) {
8887
SchemaType.call(this, key, options, 'UUID');
89-
this.getters.push(binaryToString);
88+
this.getters.push(function(value) {
89+
// For populated
90+
if (value != null && value.$__ != null) {
91+
return value;
92+
}
93+
return binaryToString(value);
94+
});
9095
}
9196

9297
/**
@@ -110,7 +115,7 @@ SchemaUUID.prototype.constructor = SchemaUUID;
110115
*/
111116

112117
SchemaUUID._cast = function(value) {
113-
if (value === null) {
118+
if (value == null) {
114119
return value;
115120
}
116121

@@ -247,11 +252,8 @@ SchemaUUID.prototype.checkRequired = function checkRequired(value) {
247252
*/
248253

249254
SchemaUUID.prototype.cast = function(value, doc, init) {
250-
if (SchemaType._isRef(this, value, doc, init)) {
251-
if (isBsonType(value, 'UUID')) {
252-
return value;
253-
}
254-
255+
if (utils.isNonBuiltinObject(value) &&
256+
SchemaType._isRef(this, value, doc, init)) {
255257
return this._castRef(value, doc, init);
256258
}
257259

lib/schematype.js

+1
Original file line numberDiff line numberDiff line change
@@ -1521,6 +1521,7 @@ SchemaType.prototype._castRef = function _castRef(value, doc, init) {
15211521
const path = doc.$__fullPath(this.path, true);
15221522
const owner = doc.ownerDocument();
15231523
const pop = owner.$populated(path, true);
1524+
15241525
let ret = value;
15251526
if (!doc.$__.populated ||
15261527
!doc.$__.populated[path] ||

lib/utils.js

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
* Module dependencies.
55
*/
66

7+
const UUID = require('bson').UUID;
78
const ms = require('ms');
89
const mpath = require('mpath');
910
const ObjectId = require('./types/objectid');
@@ -406,6 +407,7 @@ exports.isNonBuiltinObject = function isNonBuiltinObject(val) {
406407
return typeof val === 'object' &&
407408
!exports.isNativeObject(val) &&
408409
!exports.isMongooseType(val) &&
410+
!(val instanceof UUID) &&
409411
val != null;
410412
};
411413

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
{
22
"name": "mongoose",
33
"description": "Mongoose MongoDB ODM",
4-
"version": "6.11.1",
4+
"version": "6.11.3",
55
"author": "Guillermo Rauch <[email protected]>",
66
"keywords": [
77
"mongodb",

test/collection.capped.test.js

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ describe('collections: capped:', function() {
4646
capped.set('capped', { size: 1000 });
4747
const Capped = db.model('Test', capped, 'Test');
4848
await Capped.init();
49+
await Capped.createCollection();
4950
await new Promise((resolve) => setTimeout(resolve, 100));
5051

5152
const isCapped = await Capped.collection.isCapped();

test/connection.test.js

-15
Original file line numberDiff line numberDiff line change
@@ -1011,21 +1011,6 @@ describe('connections:', function() {
10111011
assert.throws(() => m.model('Test', Schema({ name: String })), /overwrite/);
10121012
});
10131013

1014-
it('can use destructured `connect` and `disconnect` (gh-9597)', async function() {
1015-
const m = new mongoose.Mongoose();
1016-
const connect = m.connect;
1017-
const disconnect = m.disconnect;
1018-
1019-
await disconnect();
1020-
await new Promise((resolve) => setTimeout(resolve, 0));
1021-
1022-
const errorOnConnect = await connect(start.uri).then(() => null, err => err);
1023-
assert.ifError(errorOnConnect);
1024-
1025-
const errorOnDisconnect = await disconnect().then(() => null, err => err);
1026-
assert.ifError(errorOnDisconnect);
1027-
});
1028-
10291014
describe('when connecting with a secondary read preference(gh-9374)', function() {
10301015
describe('mongoose.connect', function() {
10311016
it('forces autoIndex & autoCreate to be false if read preference is secondary or secondaryPreferred', async function() {

test/document.test.js

+18
Original file line numberDiff line numberDiff line change
@@ -12278,6 +12278,24 @@ describe('document', function() {
1227812278
assert.equal(fromDb.obj.subArr.length, 1);
1227912279
assert.equal(fromDb.obj.subArr[0].str, 'subArr.test1');
1228012280
});
12281+
12282+
it('avoids prototype pollution on init', async function() {
12283+
const Example = db.model('Example', new Schema({ hello: String }));
12284+
12285+
const example = await new Example({ hello: 'world!' }).save();
12286+
await Example.findByIdAndUpdate(example._id, {
12287+
$rename: {
12288+
hello: '__proto__.polluted'
12289+
}
12290+
});
12291+
12292+
// this is what causes the pollution
12293+
await Example.find();
12294+
12295+
const test = {};
12296+
assert.strictEqual(test.polluted, undefined);
12297+
assert.strictEqual(Object.prototype.polluted, undefined);
12298+
});
1228112299
});
1228212300

1228312301
describe('Check if instance function that is supplied in schema option is availabe', function() {

test/query.cursor.test.js

+24
Original file line numberDiff line numberDiff line change
@@ -844,6 +844,30 @@ describe('QueryCursor', function() {
844844
const docs = await Example.find().sort('foo');
845845
assert.deepStrictEqual(docs.map(d => d.foo), ['example1', 'example2']);
846846
});
847+
848+
it('should allow middleware to run before applying _optionsForExec() gh-13417', async function() {
849+
const testSchema = new Schema({
850+
a: Number,
851+
b: Number,
852+
c: Number
853+
});
854+
testSchema.pre('find', function() {
855+
this.select('-c');
856+
});
857+
const Test = db.model('gh13417', testSchema);
858+
await Test.create([{ a: 1, b: 1, c: 1 }, { a: 2, b: 2, c: 2 }]);
859+
const cursorMiddleSelect = [];
860+
let r;
861+
const cursor = Test.find().select('-b').sort({ a: 1 }).cursor();
862+
// eslint-disable-next-line no-cond-assign
863+
while (r = await cursor.next()) {
864+
cursorMiddleSelect.push(r);
865+
}
866+
assert.equal(typeof cursorMiddleSelect[0].b, 'undefined');
867+
assert.equal(typeof cursorMiddleSelect[1].b, 'undefined');
868+
assert.equal(typeof cursorMiddleSelect[0].c, 'undefined');
869+
assert.equal(typeof cursorMiddleSelect[1].c, 'undefined');
870+
});
847871
});
848872

849873
async function delay(ms) {

test/schema.uuid.test.js

+22-2
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,8 @@
33
const start = require('./common');
44
const util = require('./util');
55

6-
const bson = require('bson');
7-
86
const assert = require('assert');
7+
const bson = require('bson');
98

109
const mongoose = start.mongoose;
1110
const Schema = mongoose.Schema;
@@ -129,6 +128,27 @@ describe('SchemaUUID', function() {
129128
assert.equal(organization, undefined);
130129
});
131130

131+
it('works with populate (gh-13267)', async function() {
132+
const userSchema = new mongoose.Schema({
133+
_id: { type: 'UUID', default: () => uuidv4() },
134+
name: String,
135+
createdBy: {
136+
type: 'UUID',
137+
ref: 'User'
138+
}
139+
});
140+
const User = db.model('User', userSchema);
141+
142+
const u1 = await User.create({ name: 'admin' });
143+
const { _id } = await User.create({ name: 'created', createdBy: u1._id });
144+
145+
const pop = await User.findById(_id).populate('createdBy').orFail();
146+
assert.equal(pop.createdBy.name, 'admin');
147+
assert.equal(pop.createdBy._id.toString(), u1._id.toString());
148+
149+
await pop.save();
150+
});
151+
132152
// the following are TODOs based on SchemaUUID.prototype.$conditionalHandlers which are not tested yet
133153
it('should work with $bits* operators');
134154
it('should work with $all operator');

0 commit comments

Comments
 (0)