4

我正在尝试使用猫鼬和 MongoDB 将任务保存到任务列表中。我想将它冗余地保存在任务集合和相应的列表文档中作为嵌入文档。

它工作正常,但有一件小事:列表的嵌入文档没有它们的 objectId。但我需要它们以便将它们与任务集合中的文档进行逻辑连接。

我的架构:

var TaskSchema = new Schema({
    _id: ObjectId,
    title: String,
    list: ObjectId

});

var Task = mongoose.model('task', TaskSchema);

var ListSchema = new Schema({
    _id: ObjectId,
    title: String,
    tasks: [Task.schema]
});

var List = mongoose.model('list', ListSchema);

我的控制器/路由器:

app.post('/lists/:list_id/tasks', function(req, res) {

        var listId = req.params.list_id;   

        // 1. Save the new task into the tasks-collection.

        var newTask = new Task();
        newTask.title = req.body.title;
        newTask.list = listId; 

        console.log('TaskId:' + newTask._id);  // Returns undefined on the console!!!

        newTask.save(); // Works fine!

        // 2. Add the new task to the coresponding list.

        list.findById(listId, function(err, doc){

            doc.tasks.push(newTask);

            doc.save();  // Saves the new task in the list but WITHOUT its objectId

            });

        res.redirect('/lists/' + listId)

});

我可以使用猫鼬不同的方式来实现这一目标吗?还是我必须保存任务然后在将其保存到列表中之前对其进行查询?

谢谢你的建议:-)

4

2 回答 2

9

我通过使用一个名为populate的很棒的功能解决了这个问题!

dtryon 也是对的:您不需要在模型中声明您的 _id ObjectIds,它们无论如何都会被添加。而且你必须嵌套这些东西,因为你的任务是异步保存的,所以你必须确保它在其他东西之前运行。

这是解决方案:

架构:

var TaskSchema = new Schema({
    title: String,
    list: { type: Schema.ObjectId, ref: 'list' }

});

var Task = mongoose.model('task', TaskSchema);

var ListSchema = new Schema({
    title: String,
    tasks: [{ type: Schema.ObjectId, ref: 'task' }]
});

var List = mongoose.model('list', ListSchema);

控制器/路由器:

app.post('/lists/:list_id/tasks', function(req, res) {

    var listId = req.params.list_id;   

    var newTask = new Task();
    newTask.title = req.body.title;
    newTask.list = listId; 


    // WHEN SAVING, WRAP THE REST OF THE CODE

    newTask.save(function (err){
       if (err) {
            console.log('error saving new task');
            console.log(err);
        } else {
            console.log('new task saved successfully'); 

            list.findById(listId), function(err, doc){

                doc.tasks.push(newTask);

                doc.save(function (err){
                    if (err) {
                    console.log('error adding new task to list');
                    console.log(err);
                    } else {

                    console.log('new task saved successfully'); 
                    res.redirect('/lists/' + listId);

                    }  
                });
            });
        });
    });
});

现在它正确引用并使用填充功能,您可以非常舒适地访问条目。像魅力一样工作:-)

于 2012-06-06T13:59:41.307 回答
1

我认为这可能与异步有关。list.findById 您可能看不到 ObjectId,因为您在推送newTask. res.redirect此外,您在执行异步操作之外也会遇到麻烦。

试试这个:

app.post('/lists/:list_id/tasks', function(req, res) {

        var listId = req.params.list_id;   

        // 1. Save the new task into the tasks-collection.

        var newTask = new Task();
        newTask.title = req.body.title;
        newTask.list = listId; 

        console.log('TaskId:' + newTask._id);  // Returns undefined on the console!!!

        // WHEN SAVING, WRAP THE REST OF THE CODE
        newTask.save(function (err){
            if (err) {
                console.log(err);
                    // do something
            }

            // 2. Add the new task to the coresponding list.

            list.findById(listId, function(err, doc){

                doc.tasks.push(newTask);

                doc.save();  // Saves the new task in the list but WITHOUT its objectId

                // REDIRECT AFTER SUCCESS
                res.redirect('/lists/' + listId)

            });
       });
});
于 2012-06-06T10:53:05.140 回答