使用具有意外结果的Mongoose Dynamic Populate

时间:2017-07-31 03:48:44

标签: node.js mongoose mongoose-schema mongoose-populate

目前有一个幻灯片概念和不同类型的幻灯片。我正在尝试在幻灯片放映对象上填充slides数组,并将动态参考设置为幻灯片类型,以获取具有不同文档的已填充的幻灯片数组。

在“动态参考”标题下的http://mongoosejs.com/docs/populate.html处匹配文档,我的代码如下所示。

// Abreviated slideshow model
var slideshowSchema = mongoose.Schema({
    slides: [{
        slideType: String,
        item: {
            type: Schema.Types.ObjectId,
            refPath: 'slides.slideType'
        }
    }]
}]


// controller get route
app.get('/slideshow/:id', function(req, res, next) {
    var id = req.params.id;
    if(id){
        Slideshow
            .findById(id)
            .populate('slides.item')
            .exec(function(err, slideshow) {
                res.writeHead( 200, { "Content-Type": "application/json" } );
                res.end(JSON.stringify(slideshow));
            });
    }
    else{
        return next();
    }
});


// type of slide trying to be populated
var currentWeatherSchema = mongoose.Schema({

  slideType: {
      type: String,
      required: true
  },

  slideshowId: {
      type: Schema.Types.ObjectId,
      required: true
  }

  // other unique props

});

当对硬编码的slideType进行直接填充时,幻灯片按预期填充,并且在运行上面的代码时,我确实得到了响应,并且幻灯片数组填充了正确数量的对象,但它们看起来像这样预期文件:

"slides": [
    {
        "_bsontype": "ObjectID",
        "id": {
            "type": "Buffer",
            "data": [
                89,
                126,
                152,
                150,
                243,
                157,
                179,
                147,
                165,
                23,
                247,
                56
            ]
        }
    }
]

我已经无数次阅读过这些例子,无法弄清楚我哪里出错了。有没有人有这方面的经验,或者看到我可能缺少什么?

2 个答案:

答案 0 :(得分:0)

看起来你的populate正在返回存储的ObjectId,因为找不到集合引用(如果找到它且对象不匹配,它将返回null)。因此,集合文档中的“slideType”值没有有效的集合引用。

  • 为每个架构创建一个模型。
  • 使用创建模型的名称填充slideType。小心大箱。
  • 确保ObjectId与右参考相同。

答案 1 :(得分:0)

您可以使用aggregate和(展开/查找)执行此操作:

流程:

在每张幻灯片中展开您的项目数组

解析数组中的每个项目

将解决方案项目分组到每张幻灯片的字段中

类似的东西:

 db.slideshowSchema.aggregate([
    {
        $match: {_id: id} // find condition
    },
    {
       $unwind: "$slides" // explode slide array of each document
    },
    {
        $lookup: { // resolve dependencies, take object in mycrosscollections where  mycrosscollections._id == $slides.item
            from:  "mycrosscollections",
            "localField": "$slides.item",
            "foreignField": "_id",
            "as": "resolveSlides"
        }
    },
    {
        $group : { // push back resolve items in slides fields, and regroup by document id
            _id: "$_id",
            slides: {$push: "$resolveSlides"}
        }
     }
 ])