mongoose模式引用vs嵌入数组

时间:2018-01-11 10:35:46

标签: javascript arrays node.js mongodb mongoose

我有快递和猫鼬的应用程序。我有两个模式,一个Blog模式和一个Comment模式。我想使用RESTful路由将注释数组从表单推送到单个博客。如果我在Blog模式中嵌入了Comment模式,那么另一方面,如果我在Blog模式中引用Comment模式的ObjectId,则代码适用于第一条注释,然后在第二条注释中,控制台会抛出验证错误:< / p>

(node:5472) UnhandledPromiseRejectionWarning: 
Unhandled promise rejection (rejection id: 1): 
ValidationError: Blog validation failed: 
comments: Cast to [undefined] failed for value 
"[{
  "_id":"5a57374da3ba43156005c881",
  "text":"test1",
  "author":"test1",
  "__v":0
}]" 
at path "comments"

记录的对象是我推送到数组的第一个注释

这是博客架构(嵌入版本)

//Embed

var mongoose = require("mongoose");

var comment = new mongoose.Schema({
    text: String,
    author: String
});

var blogSchema = new mongoose.Schema({
    name: String,
    image: String,
    description: String,
    comments: [comment]
});

module.exports = mongoose.model("Blog", blogSchema);

这是博客架构(参考版本)

// Reference

var mongoose = require("mongoose");

var blogSchema = new mongoose.Schema({
    name: String,
    image: String,
    description: String,
    comments: [{
        type: mongoose.Schema.Types.ObjectId,
        ref: "Comment"
    }]
});

module.exports = mongoose.model("Blog", blogSchema);

这是评论模式

var mongoose = require("mongoose");

var commentSchema = new mongoose.Schema({
    text: String,
    author: String
});

module.exports = mongoose.model("Comment", commentSchema);

这是评论路线我用来发布新评论:

app.post("/blogs/:id/comments", function (req, res) {
    // find the correct campground
    Blog.findById(req.params.id, function (err, blog) {
        if (err) {
            console.log(err);
            res.redirect("/blogs");
        } else {
            Comment.create(req.body.comment, function (err, comment) {
                if (err) {
                    console.log(err);
                } else {
                    blog.comments.push(comment);
                    blog.save();
                    res.redirect("/blogs/" + blog._id);
                }
            });
        }
    });
});

请注意,Route可以正常工作并重定向到“blogs::id”页面,如果我在console.log博客或che评论对象我有正确的输出

最后,这是我用来发布请求的表单:

<div class="container">
    <div class="row">
        <h1 style="text-align: center;">Add a new Comment to <%= blog.name %></h1>
        <div style="width: 30%; margin:25px auto;">
            <form action="/blog/<%= blog._id %>/comments" method="POST">
                <div class="form-group">
                    <input class="form-control" type="text" name="comment[text]" placeholder="text">
                </div>
                <div class="form-group">
                    <input class="form-control"  type="text" name="comment[author]" placeholder="author">
                </div>
                <div class="form-group">
                    <button class="btn btn-lg btn-primary btn-block">Submit!</button>
                </div>
            </form>
            <a href="/blogs">Go Back</a>
        </div>
    </div>
</div>

编辑我更喜欢只使用注释的ID而不是完整的注释架构,但我不知道如何设置应用程序来做到这一点。

在mongoose github配置文件中也报告了相同的错误:https://github.com/Automattic/mongoose/issues/5972

2 个答案:

答案 0 :(得分:0)

对于参考版本:创建评论后,您只需在博客文档中推送已创建评论文档的ID。

blog.comments.push(comment._id); 

blog.comments.push(mongoose.Types.ObjectId(comment._id)); //to ensure the pushed id will be mongoose object id.

您必须为步骤2要求猫鼬。

答案 1 :(得分:0)

我找到解决此问题的唯一方法是将我的代码重构为:

app.post("/blogs/:id/comments", function (req, res) {
    Comment.create(req.body.comment, function (err, comment) {
        if (err) {
            console.log(err);
        } else {
            Blog.findOne({"_id": req.params.id})
            .populate("comments")
            .exec(function (err, blog) {
                if (err) {
                    console.log(err);
                    res.redirect("/blogs");
                } else {
                    blog.comments.push(comment);
                    blog.save();
                    res.redirect("/blogs/" + blog._id);
                }
            });
        }
    });
});

然而,这并不能解释Schemas的行为

希望有所帮助