Skip to content

Commit 63b5d76

Browse files
authored
Merge pull request #15329 from Automattic/vkarpov15/gh-15315
fix(populate): handle virtual populate on array of UUIDs
2 parents 8435b9b + ae3cc59 commit 63b5d76

File tree

2 files changed

+46
-1
lines changed

2 files changed

+46
-1
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
});

0 commit comments

Comments
 (0)