Я думаю, проблема в том, что модель должна быть эталоном модели, а не названием модели. Таким образом, это должно быть model: Blog
вместо model: "Blog"
.
Также я предлагаю вам перепроектировать свои схемы, потому что у вас есть много ссылок между моделями. И вам нужно будет делать вызовы по 2 дБ при добавлении, вставке или удалении блога или комментария.
Я бы удалил поля блогов из пользовательской схемы, удалил поле комментариев из схемы блога и настроил виртуальное заполнение следующим образом:
пользовательская схема:
const userSchema = new Schema(
{
email: {
type: String,
required: true,
index: {
unique: true
}
},
password: {
type: String,
required: true
},
name: {
type: String,
required: true
},
website: {
type: String
},
bio: {
type: String
}
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
},
toJSON: {
virtuals: true
}
}
);
userSchema.virtual("blogs", {
ref: "Blog",
foreignField: "author",
localField: "_id"
});
blog схема:
const blogSchema = new Schema(
{
title: {
type: String,
required: true
},
body: {
type: String,
required: true
},
author: {
type: Schema.Types.ObjectId,
ref: "User"
},
likesCount: {
type: Number
}
},
{
timestamps: {
createdAt: "created_at",
updatedAt: "updated_at"
},
toJSON: { virtuals: true }
}
);
blogSchema.virtual("comments", {
ref: "Comment",
foreignField: "blog",
localField: "_id"
});
Обратите внимание, что я добавил опцию toJSON: { virtuals: true }
к обеим схемам.
Теперь вы можете получать блоги пользователей с комментариями по следующему запросу:
const user = await User.findById(req.user._id)
.populate({
path: "blogs",
populate: "comments"
})
.select("-password")
.exec();
Тест:
Допустим, у вас есть эти образцы документов.
db={
"users": [
{
"_id": "5e53b1726f41c765fc4def9c",
"email": "user1@gmail.com",
"password": "$2a$10$.heEhkN2BhxZiw8upgjGQe.r3Gt78JVfuAqLqf6lHwireaKJSrrTO",
"name": "User1"
},
{
"_id": "5e53b1906f41c765fc4def9d",
"email": "user2@gmail.com",
"password": "$2a$10$tEaXpoeH5iXVzqmzozAFOOu.Nxb32Ssy1XS5CAqad7qqanHQkrqjK",
"name": "User2"
},
{
"_id": "5e53b1996f41c765fc4def9e",
"email": "user3@gmail.com",
"password": "$2a$10$4s34RLnSd75WeG8K.gzxxeixkruplzW0vpb7PJR/aL1d3Ia31wj.W",
"name": "User3"
}
],
"blogs": [
{
"_id": "5e53b26c6f41c765fc4def9f",
"title": "Blog1 Title",
"body": "Blog1 Body",
"author": "5e53b1726f41c765fc4def9c"
},
{
"_id": "5e53b2896f41c765fc4defa1",
"title": "Blog2 Title",
"body": "Blog2 Body",
"author": "5e53b1726f41c765fc4def9c"
}
],
"comments": [
{
"_id": "5e53b2f86f41c765fc4defa3",
"body": "Comment1 (user2 on user1's blog1)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b3246f41c765fc4defa4",
"body": "Comment2 (user3 on user1's blog1)",
"user": "5e53b1996f41c765fc4def9e",
"blog": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b34c6f41c765fc4defa5",
"body": "Comment3 (user2 on user1's blog2)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b2896f41c765fc4defa1"
}
]
}
Результат будет таким для пользователя с _id: "5e53b1726f41c765fc4def9c"
:
{
"_id": "5e53b1726f41c765fc4def9c",
"email": "user1@gmail.com",
"name": "User1",
"created_at": "2020-02-24T11:20:18.343Z",
"updated_at": "2020-02-24T11:20:18.343Z",
"__v": 0,
"blogs": [
{
"_id": "5e53b26c6f41c765fc4def9f",
"title": "Blog1 Title",
"body": "Blog1 Body",
"author": "5e53b1726f41c765fc4def9c",
"created_at": "2020-02-24T11:24:28.895Z",
"updated_at": "2020-02-24T11:24:28.895Z",
"__v": 0,
"comments": [
{
"_id": "5e53b2f86f41c765fc4defa3",
"body": "Comment1 (user2 on user1's blog1)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b26c6f41c765fc4def9f",
"created_at": "2020-02-24T11:26:48.506Z",
"updated_at": "2020-02-24T11:26:48.506Z",
"__v": 0
},
{
"_id": "5e53b3246f41c765fc4defa4",
"body": "Comment2 (user3 on user1's blog1)",
"user": "5e53b1996f41c765fc4def9e",
"blog": "5e53b26c6f41c765fc4def9f",
"created_at": "2020-02-24T11:27:32.305Z",
"updated_at": "2020-02-24T11:27:32.305Z",
"__v": 0
}
],
"id": "5e53b26c6f41c765fc4def9f"
},
{
"_id": "5e53b2896f41c765fc4defa1",
"title": "Blog2 Title",
"body": "Blog2 Body",
"author": "5e53b1726f41c765fc4def9c",
"created_at": "2020-02-24T11:24:57.078Z",
"updated_at": "2020-02-24T11:24:57.078Z",
"__v": 0,
"comments": [
{
"_id": "5e53b34c6f41c765fc4defa5",
"body": "Comment3 (user2 on user1's blog2)",
"user": "5e53b1906f41c765fc4def9d",
"blog": "5e53b2896f41c765fc4defa1",
"created_at": "2020-02-24T11:28:12.551Z",
"updated_at": "2020-02-24T11:28:12.551Z",
"__v": 0
}
],
"id": "5e53b2896f41c765fc4defa1"
}
],
"id": "5e53b1726f41c765fc4def9c"
}
Другой вариант использует платформу агрегации MongoDB. С помощью этой опции вы можете удалить опции, которые я добавил для виртуального заполнения.
Детская площадка
const users = await User.aggregate([
{
$match: {
_id: req.user._id
}
},
{
$project: {
password: 0
}
},
{
$lookup: {
from: "blogs",
let: {
userId: "$_id"
},
pipeline: [
{
$match: {
$expr: {
$eq: ["$$userId", "$author"]
}
}
},
{
$lookup: {
from: "comments",
let: {
blogId: "$_id"
},
pipeline: [
{
$match: {
$expr: {
$eq: ["$blog", "$$blogId"]
}
}
}
],
as: "comments"
}
}
],
as: "blogs"
}
}
]);
users[0]
даст вам тот же результат, что и опция 1.