在猫鼬中使用 UUID 作为 ObjectID 引用



我正在使用猫鼬用Node.js,Express和MongoDB构建一个CRUD风格的REST服务。 该服务将允许已经存在的Android应用程序的用户在线上传/同步其个人数据库的内容。

现有应用程序的数据模型使用UUID(在Java中生成(,它与较短,单调的MongoDB样式_id字段冲突。 由于数据模型已经存在并且填充了许多用户的数据,因此我无法将源数据转换为单调的MongoDB风格的_id。 这给我留下了我能想到的 2 个选项:1( 让 Mongo/Mongoose(或其他一些 ODM(使用完整的 UUID 而不是单调的 _id s 很好地发挥作用,或者 2( 除了_id字段之外,还向猫鼬模型添加一个 uuid 字段,并消除这种方法的陷阱。 我正在尝试选择选项 #1 并遇到 ObjectID 引用的问题。

我最初偶然发现了猫鼬-uuid,但不幸的是,这不适用于我的用例,因为它在创建新的猫鼬对象时覆盖了我显式设置的_id值。 深入研究插件代码,它假设一个对象是新的(通过调用检查 Mongoose 的.isNew值(,从而用新的 uuid 覆盖_id。 由于我在Mongo中创建新文档时需要保留原始uuid,因此此插件对我不起作用。

接下来,我找到了猫鼬的创造者亚伦·赫克曼(Aaron Heckmann(关于类似主题的帖子。 这很有帮助,但是我现在遇到了一个问题,即我无法让我的猫鼬模式通过 ObjectID 相互引用,因为它们在技术上现在使用 String '_ids 相互引用。

架构示例:

var mongoose = require('mongoose');
var uuid = require('node-uuid');
var Schema = mongoose.Schema;
var trackPassSchema = new Schema({
    _id: { type: String, default: function genUUID() {
        uuid.v1()
    }},
    //Omitting other fields in snippet for simplicity
    vehicle: [
        {type: Schema.Types.ObjectId, required: true, ref: 'Vehicle'}
    ]
});
module.exports = mongoose.model('TrackPass', trackPassSchema);

引用架构:

var mongoose = require('mongoose');
var uuid = require('node-uuid');
var Schema = mongoose.Schema;
var vehicleSchema = new Schema({
    _id: { type: String, default: function genUUID() {
        uuid.v1()
    }},
    //Omitting other fields in snippet for simplicity
    description: {type: String},
    year: {type: Number}
});
module.exports = mongoose.model('Vehicle', vehicleSchema);

当我尝试调用save()已从我的应用程序传入的 trackPass 时:

var trackPass = new TrackPass(req.body);
//Force the ID to match what was put into the request
trackPass._id = req.params.id;
trackPass.save(function (err) { ... }

我收到以下错误:

{ [CastError: Cast to ObjectId failed for value "b205ac4d-fd96-4b1e-892a-d4fab818ea2a" at path "vehicle"]
  message: 'Cast to ObjectId failed for value "b205ac4d-fd96-4b1e-892a-d4fab818ea2a" at path "vehicle"',
  name: 'CastError',
  type: 'ObjectId',
  value: ["b205ac4d-fd96-4b1e-892a-d4fab818ea2a"],
  path: 'vehicle' }

我相信这个错误是有道理的,因为我现在使用的字符串比典型的 Mongo 对象 ID 长。 如果没有 ObjectID 引用,我相信我将无法从其他集合中populate()引用的对象。 我想我不能简单地在我的架构定义中引用其他嵌套对象,但是我不喜欢这种方法,因为我觉得我会失去使用 ODM 的很多好处。 还有其他想法吗?

仍然可以将populate()与除 ObjectID 之外的_id类型的值一起使用,但确实需要在引用定义中使用相同的类型。

因此,您的trackPassSchema需要更改为:

var trackPassSchema = new Schema({
    _id: { type: String, default: function genUUID() {
        return uuid.v1()
    }},
    vehicle: [
        {type: String, required: true, ref: 'Vehicle'}
    ]
});

正如 Adam 在评论中指出的那样,您可以将default值简化为:

var trackPassSchema = new Schema({
    _id: { type: String, default: uuid.v1 },
    vehicle: [
        {type: String, required: true, ref: 'Vehicle'}
    ]
});

JohnnyHK和Adam C的答案都是正确的。但是,如果您在对象数组的架构中使用uuid,最好像这样使用它

var trackPassSchema = new Schema({
_id: { type: String, default: () => uuid.v1 },
vehicle: [
    {type: String, required: true, ref: 'Vehicle'}
]

}(;

因为,在一个这样的场景中,当我尝试像这样使用 _id: { type: String, default: () => uuid.v1 }数组的多个对象具有相同的 id。

在这种情况下是不可能的,因为_id字段是唯一的字段,但是当您使用非唯一的字段时,可能会发生这种情况。

最新更新