当前位置: 首页 > 面试题库 >

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

岳涵煦
2023-03-14
问题内容

我正在使用mongoose使用Node.js,Express和MongoDB构建CRUD风格的REST服务。此服务将允许已经存在的android应用程序的用户在线上载/同步其单个数据库的内容。

已经存在的应用程序的数据模型使用UUID(用Java生成),该UUID与较短的单调MongoDB样式_id字段冲突。由于数据模型已经存在,并且填充了许多用户的数据,因此无法将源数据转换为单调的MongoDB风格_id的。这给了我2个可以想到的选择:1)使Mongo
/ Mongoose(或其他ODM)与完整的UUID(而不是单调_ids
)一起很好地玩耍,或者2)除了向mongoose模型添加uuid字段在_id现场和打击这种方法的缺陷。我试图选择选项#1并遇到与ObjectID引用有关的问题。

我最初偶然发现了mongoose-uuid,但是不幸的是,这不适用于我的用例,因为它_id在创建新的Mongoose对象时会覆盖我的显式设置值。深入研究插件代码,它假定一个对象是新对象(通过调用检查Mongoose的.isNew值),因此_id用新的uuid
覆盖了该对象。由于在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
ObjectID。没有ObjectID引用,我相信我将无法populate()引用其他集合中的对象。我想我无法在架构定义中引用其他嵌套对象,但是我不喜欢这种方法,因为我觉得使用ODM将失去很多好处。还有其他想法吗?


问题答案:

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

因此,您trackPassSchema需要更改为:

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

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

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


 类似资料:
  • 问题内容: 我想用Mongoose 生成一个MongoDB 。有没有办法从Mongoose 访问构造函数? 这个问题是关于从头开始 产生新的 。生成的ID是全新的通用唯一ID。 另一个问题是关于从 现有的字符串表示形式 创建一个。在这种情况下,您已经具有ID的字符串表示形式(它可能是通用的也可能不是唯一的),并且正在将其解析为。 问题答案: 您可以在找到构造函数。这是一个例子: 是一个新生成的。

  • 问题内容: 我正在尝试在架构之间建立某种关系,而解决方案存在一些问题。这是我的设备架构: 这里是房间模式: 猫鼬抛出错误 类型错误:未定义的类型,在 你尝试筑巢的架构?您只能使用引用或数组进行嵌套。 如果我更改为一切正常。您能解释一下为什么会这样吗? 问题答案: 是构造函数,要在模式定义中使用的是(或)。 所以应该看起来像这样:

  • 问题内容: 我试图在猫鼬中指定我的数据库的架构。目前,我这样做: 我创建模式并尝试保存一些数据。 错误: 我根据以下代码编写了此代码:http : //mongoosejs.com/docs/populate.html#gsc.tab=0 我怎样才能解决这个问题? 问题答案: 您引用的猫鼬文档中的示例用于该字段以及其他字段。 我假设他们在示例中这样做是为了证明可以使用其中任何一个。如果未在架构中指

  • 问题内容: 我试图让MongoDB根据其索引检测重复值。我认为这在MongoDB中是可能的,但是通过Mongoose包装器,事情似乎被打破了。所以对于这样的事情: 我可以用同一封电子邮件保存2个用户。真是 在这里也表达了同样的问题:https : //github.com/LearnBoost/mongoose/issues/56,但是该线程很旧,导致无处可去。 现在,我正在手动调用数据库以查找用

  • 问题内容: 我有一个简单的工具来构建文档集合,然后自动格式化它们以进行EPUB或LaTeX渲染,该工具写在ExpressJS之上。我正在使用Coffeescript,如果那很重要(我对此表示怀疑)。 使用猫鼬,我有以下几点: Offrefs没有指定它的含义,因为我希望能够在其他活页夹中包含一些活页夹,以创建逻辑集合:“这些用于打印机”,“这些用于epub”,“这些仅用于Web, ”等等。(我已经剔

  • 我有一个具有多个查找的mongo查询,但它返回了一个错误 排序超出了104857600字节的内存限制,但未选择外部排序。正在中止操作。传递allowDiskUse:true以选择加入 我已经在聚合查询中添加了allow disk size true < code>await Service.aggregate(管道)。allowDiskUse(true)。exec();