Skip to content

Commit c53f5a2

Browse files
authored
Merge branch 'master' into vkarpov15/gh-15301
2 parents b33698e + 63b5d76 commit c53f5a2

File tree

4 files changed

+56
-2
lines changed

4 files changed

+56
-2
lines changed

lib/helpers/populate/assignRawDocsToIdStructure.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,12 @@ function assignRawDocsToIdStructure(rawIds, resultDocs, resultOrder, options, re
7878
continue;
7979
}
8080

81-
sid = String(id);
81+
if (id?.constructor?.name === 'Binary' && id.sub_type === 4 && typeof id.toUUID === 'function') {
82+
// Workaround for gh-15315 because Mongoose UUIDs don't use BSON UUIDs yet.
83+
sid = String(id.toUUID());
84+
} else {
85+
sid = String(id);
86+
}
8287
doc = resultDocs[sid];
8388
// If user wants separate copies of same doc, use this option
8489
if (options.clone && doc != null) {

test/model.populate.test.js

+40
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
const start = require('./common');
88

99
const assert = require('assert');
10+
const { randomUUID } = require('crypto');
1011
const utils = require('../lib/utils');
1112
const util = require('./util');
1213
const MongooseError = require('../lib/error/mongooseError');
@@ -11423,4 +11424,43 @@ describe('model: populate:', function() {
1142311424

1142411425
await m.disconnect();
1142511426
});
11427+
11428+
it('handles populating UUID fields (gh-15315)', async function() {
11429+
const categorySchema = new Schema({
11430+
_id: { type: 'UUID', default: () => randomUUID() },
11431+
name: { type: String, required: true },
11432+
desc: { type: String, required: true }
11433+
});
11434+
11435+
categorySchema.virtual('announcements', {
11436+
ref: 'Announcement',
11437+
localField: '_id',
11438+
foreignField: 'categories'
11439+
});
11440+
11441+
const announcementSchema = new Schema({
11442+
_id: { type: 'UUID', default: () => randomUUID() },
11443+
title: { type: String, required: true },
11444+
content: { type: String, required: true },
11445+
validUntil: { type: Date, required: true },
11446+
important: { type: Boolean, default: false },
11447+
categories: [{ type: 'UUID', ref: 'Category' }]
11448+
});
11449+
11450+
const Category = db.model('Category', categorySchema);
11451+
const Announcement = db.model('Announcement', announcementSchema);
11452+
11453+
const category = await Category.create({ name: 'Tech', desc: 'Technology News' });
11454+
11455+
await Announcement.create({
11456+
title: 'New Tech Release',
11457+
content: 'Details about the new tech release',
11458+
validUntil: new Date(),
11459+
categories: [category._id]
11460+
});
11461+
11462+
const populatedCategory = await Category.findOne({ _id: category._id }).populate('announcements');
11463+
assert.strictEqual(populatedCategory.announcements.length, 1);
11464+
assert.strictEqual(populatedCategory.announcements[0].title, 'New Tech Release');
11465+
});
1142611466
});

test/types/schema.test.ts

+9
Original file line numberDiff line numberDiff line change
@@ -1777,3 +1777,12 @@ function gh15301() {
17771777
}
17781778
});
17791779
}
1780+
1781+
function defaultReturnsUndefined() {
1782+
const schema = new Schema<{ arr: number[] }>({
1783+
arr: {
1784+
type: [Number],
1785+
default: () => void 0
1786+
}
1787+
});
1788+
}

types/schematypes.d.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,7 @@ declare module 'mongoose' {
9797
* The default value for this path. If a function, Mongoose executes the function
9898
* and uses the return value as the default.
9999
*/
100-
default?: DefaultType<T> | ((this: EnforcedDocType, doc: any) => DefaultType<T>) | null;
100+
default?: DefaultType<T> | ((this: EnforcedDocType, doc: any) => DefaultType<T> | null | undefined) | null;
101101

102102
/**
103103
* The model that `populate()` should use if populating this path.

0 commit comments

Comments
 (0)