如何使用Mongoose的自定义_id加载文档?

zju*_*per 3 mongoose mongodb node.js

这是我的架构定义:

var DocSchema = new mongoose.Schema({
  _id: {
    name: String,
    path: String
  },
  label: String,
  ...
});
mongoose.model('Doc', DocSchema, 'doc_parse_utf8');
var Doc = mongoose.model('Doc');
Run Code Online (Sandbox Code Playgroud)

并且其他程序已将文件插入到mongodb中.然后我试着查询文件:

Doc.findOne({_id:{name:name,path:path}}, function(err, doc){
  if (err && err_handler) {
    err_handler(err);
  } else if(callback) {
    callback(doc);
  }
});
Run Code Online (Sandbox Code Playgroud)

但是,将报告一个强制转换错误:

{ message: 'Cast to ObjectId failed for value "[object Object]" at path "_id"',
  name: 'CastError',
  type: 'ObjectId',
  value: { name: 'mobile', path: 'etc/' },
  path: '_id' }
Run Code Online (Sandbox Code Playgroud)

我在mongoose的文档,google和statckoverflow.com上搜索过这个问题,然而,对我来说没有任何解决方案.请帮忙,谢谢.

Wir*_*rie 6

您需要做的就是_id通过将其设置为覆盖类型Mixed.

var UserSchema = new Schema({
    _id: Schema.Types.Mixed,
    name: String
});
Run Code Online (Sandbox Code Playgroud)

这导致Mongoose基本上忽略了对象的细节.

现在,当你使用find它时,它将起作用(几乎与预期一样).

我警告你,你需要确定_id你正在使用的对象的属性顺序必须以完全相同的顺序提供,否则_ids将不被认为是相同的.

当我尝试这样做时:

var User = mongoose.model('User', UserSchema);

var testId = { name: 'wiredprairie', group: 'abc'};
var u = new User({_id: testId , name: 'aaron'});

u.save(function(err, results) {
    User.find().where("_id", testId)
        .exec(function(err, users) {
            console.log(users.length); 
    });
});
Run Code Online (Sandbox Code Playgroud)

控制台输出是0.

我注意到MongoDB中的实际数据存储方式与我认为保存的方式不同:

{ 
  "_id" : { 
       "group" : "abc", 
       "name" : "wiredprairie" 
  }, 
  "name" : "aaron",
  "__v" : 0 
}
Run Code Online (Sandbox Code Playgroud)

正如你所看到的,它不是name那么group我会编码.(这是按字母顺序排列的,回想起来有意义).

所以,相反,我这样做了:

var User = mongoose.model('User', UserSchema);

var testId = { name: 'wiredprairie', group: 'abc'};
var u = new User({_id: testId , name: 'aaron'});

u.save(function(err, results) {
    User.find().where("_id", { group: 'abc', name: 'wiredprairie'})
        .exec(function(err, users) {
            console.log(users.length);
    });
});
Run Code Online (Sandbox Code Playgroud)

然后,控制台输出是1.