在mongoose中使用UUID进行ObjectID引用

Jer*_*myL 15 rest mongoose mongodb node.js express

我正在使用mongoose使用Node.js,Express和MongoDB构建CRUD样式的REST服务.该服务将允许已经存在的Android应用程序的用户在线上载/同步其各个数据库的内容.

已有应用程序的数据模型使用UUID(用Java生成),它与较短的单调MongoDB样式_id字段冲突.因为数据模型已经存在并且填充了来自许多用户的数据,所以我无法将源数据转换为单调的MongoDB样式_id.这给我留下了两个我能想到的选择:1)使Mongo/Mongoose(或其他一些ODM)与完整的UUID而不是单调_ids或2)添加一个uuid字段除了mongoose模型该_id领域并与这种方法的陷阱作斗争.我试图选择#1选项并遇到ObjectID引用问题.

我最初偶然发现了mongoose-uuid,但不幸的是,这对我的用例没有用,因为它_id在创建新的Mongoose对象时覆盖了我明确设置的值.潜入插件代码,它假定一个对象是新的(通过调用检查Mongoose的.isNew值),然后_id用新的uuid 覆盖它.因为我需要在Mongo中创建新文档时保留原始uuid,所以此插件对我不起作用.

接下来,我发现了猫鼬的创造者Aaron Heckmann关于类似话题的帖子.这有帮助,但我现在遇到的问题是我不能让我的mongoose模式通过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);
Run Code Online (Sandbox Code Playgroud)

引用模式:

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);
Run Code Online (Sandbox Code Playgroud)

当我尝试调用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) { ... }
Run Code Online (Sandbox Code Playgroud)

我收到以下错误:

{ [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' }
Run Code Online (Sandbox Code Playgroud)

我相信这个错误是有道理的,因为我现在使用的字符串比典型的Mongo ObjectID更长.没有ObjectID引用,我不相信我能够populate()引用其他集合中的对象.我想我可能根本不能在我的模式定义中引用其他嵌套对象,但我不喜欢这种方法,因为我觉得我将失去利用ODM的许多好处.还有其他想法吗?

Joh*_*yHK 19

您仍然可以使用populate()具有_id的除了对象ID类型的值,但你需要在参考定义中使用同一类型.

所以你trackPassSchema需要改为:

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

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

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

  • 你可以简化一下:`_id:{type:String,default:uuid.v1}` (10认同)
  • 不,这很好.`default:uuid.v1()`会有这个问题,但不是`default:uuid.v1`. (3认同)