我正在使用Node.js,Express和MongoDB使用mongoose构建一个CRUD风格的REST服务.这项服务将允许已经存在的
Android应用程序的用户在线上传/同步其各自数据库的内容.
@H_403_2@已经存在的应用程序的数据模型使用与较短单调的MongoDB样式_id字段冲突的UUID(以Java生成).由于数据模型已经存在并且填充了许多用户的数据,所以我无法将源数据转换为单调的MongoDB样式_ids.这让我有两个选项,我可以想到:1)使Mongo / Mongoose(或其他一些ODM)与完整的UUID而不是单调的_id,或2)添加一个uuid字段到mongoose模型除了_id字段,并打击这种方法的陷阱.我试图选择#1选项,并遇到与ObjectID引用的问题.
@H_403_2@我最初偶然发现了mongoose-uuid,但不幸的是,这并不适用于我的用例,因为它在创建新的Mongoose对象时覆盖了我明确设置的_id值.潜入插件代码,它假定一个对象是新的(通过调用检查Mongoose的.isNew值),从而用新的uuid覆盖_id.由于我需要在Mongo中创建新的文档时保留原来的uuid,所以这个插件对我来说是不行的.
@H_403_2@接下来,我发现了一个post,由Aaron Heckmann,猫头鹰的创建者,类似的话题.这是有帮助的,但是我现在遇到的问题是我无法通过ObjectID引用我的mongoose模式,因为他们在技术上现在使用String`_ids来引用它们.
@H_403_2@模式示例:
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);@H_403_2@引用模式:
var mongoose = require('mongoose'); var uuid = require('node-uuid'); var Schema = mongoose.Schema; var vehicleSchema = new Schema({ _id: { type: String,//Omitting other fields in snippet for simplicity description: {type: String},year: {type: Number} }); module.exports = mongoose.model('Vehicle',vehicleSchema);@H_403_2@当我尝试调用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) { ... }@H_403_2@我收到以下错误:
{ [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' }@H_403_2@我相信这个错误是有道理的,因为我现在正在使用比典型的Mongo ObjectID长的字符串.没有ObjectID引用,我不相信我将能够从其他集合填充()引用的对象.我想我可以简单地不引用我的模式定义中的其他嵌套对象,但是我不喜欢这种方法,因为我觉得我将会失去利用ODM的很多好处.还有其他想法吗?
解决方法
您仍然可以使用除ObjectID之外的类型的_id值的populate(),但是您需要在引用定义中使用相同的类型.
@H_403_2@所以您的trackPassSchema将需要更改为:
var trackPassSchema = new Schema({ _id: { type: String,vehicle: [ {type: String,ref: 'Vehicle'} ] });@H_403_2@正如Adam在评论中注释的那样,您可以将您的默认值简化为:
var trackPassSchema = new Schema({ _id: { type: String,default: uuid.v1 },ref: 'Vehicle'} ] });