我正在使用mongoose使用Node.js,Express和MongoDB构建CRUD样式的REST服务。该服务将允许已有的Android应用程序的用户在线上传/同步其各个数据库的内容。
现有应用程序的数据模型使用UUID(用Java生成),它与较短的单调MongoDB样式_id
字段冲突。因为数据模型已经存在并且填充了来自许多用户的数据,所以我无法将源数据转换为单调MongoDB风格的_id
。这给我留下了两个我能想到的选项:1)使Mongo / Mongoose(或其他一些ODM)与完整的UUID完美匹配,而不是单调_id
或2)将uuid字段添加到除_id
字段之外的猫鼬模型,并对抗这种方法的陷阱。我正在尝试选择#1选项并遇到ObjectID引用问题。
我最初偶然发现mongoose-uuid,但不幸的是,这对我的用例没有用,因为它在创建新的Mongoose对象时覆盖了我明确设置的_id
值。深入了解插件代码,它假设一个对象是新的(通过调用检查Mongoose的.isNew
值),从而用新的uuid覆盖_id
。因为我需要在Mongo中创建新文档时保留原始uuid,所以此插件对我不起作用。
接下来,我发现了猫鼬的创造者Aaron Heckmann对类似主题的post。这有帮助,但我现在遇到的问题是我不能让我的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);
引用架构:
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 ObjectID更长。如果没有ObjectID引用,我不相信我能够populate()
引用其他集合中的对象。我想我可能根本不能在我的模式定义中引用其他嵌套对象,但是我不喜欢这种方法,因为我觉得我将失去利用ODM的许多好处。还有其他想法吗?
答案 0 :(得分:18)
除了ObjectID之外,您仍然可以将populate()
与_id
类型的值一起使用,但您需要在参考定义中使用相同的类型。
因此,您的trackPassSchema
需要更改为:
var trackPassSchema = new Schema({
_id: { type: String, default: function genUUID() {
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'}
]
});
答案 1 :(得分:0)
JohnnyHK 和 Adam C 的答案都是正确的。但是如果你在 schema 中为对象数组使用 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 是唯一字段,但是当您使用非唯一字段时可能会发生这种情况。