javascript - 带日期的 Mongoose 混合类型字段

标签 javascript node.js mongodb mongoose

我正在创建一个应用程序,用户可以在其中创建各种类型的自定义字段,我想将这些字段存储在 mongo 中。类型将包括字符串、数字和日期。我的架构如下所示:

const MetaSchema = new mongoose.Schema({
  key: String,
  value: {type: mongoose.Schema.Types.Mixed},
  type: String,
  created_at: {type: Date, default: Date.now}
});

效果很好,我可以按预期存储数据。问题是,当我想存储一个日期时,它会以 ISO 格式发送到服务器,有效负载可能如下所示:

{
  "key": "Contract Signed",
  "value": "2016-04-06T22:35:11.540Z",
  "type": "date"
}

有什么方法可以让 mongo/mongoose 像日期而不是字符串一样处理和存储它?如果我将其设置为日期类型,那么我认为它可以解决问题,但我必须保存他们可以为自定义字段提供的任何内容。非常感谢!

TLDR:是否可以根据插入的数据类型(IE 日期与字符串)区别对待 mongoose/mongo 中的混合数据类型。

最佳答案

使用 Mongoose discriminators可能是去这里的路。它们实际上在存储的文档中使用它们自己的“类型”(默认 __t 但可以被覆盖)属性,这允许 mongoose 实际上将一种“模型”应用到每个对象及其自己的附加模式。

举个简单的例子:

var async = require('async'),
    util  = require('util'),
    mongoose = require('mongoose'),
    Schema = mongoose.Schema;

mongoose.connect('mongodb://localhost/things');
mongoose.set("debug",true);

function BaseSchema() {

  Schema.apply(this,arguments);

  this.add({
    key: String,
    created_at: { type: Date, default: Date.now }
  });

}

util.inherits(BaseSchema,Schema);

var metaSchema = new BaseSchema();

var stringSchema = new BaseSchema({
  value: String
});

var numberSchema = new BaseSchema({
  value: Number
});

var dateSchema = new BaseSchema({
  value: Date
});

var MetaModel = mongoose.model('MetaModel',metaSchema),
    StringModel = MetaModel.discriminator('StringModel', stringSchema),
    NumberModel = MetaModel.discriminator('NumberModel', numberSchema),
    DateModel = MetaModel.discriminator('DateModel', dateSchema);

async.series(
  [
    function(callback) {
      MetaModel.remove({},callback);
    },
    function(callback) {
      async.each(
        [
          { "model": "StringModel", "value": "Hello" },
          { "model": "NumberModel", "value": 12 },
          { "model": "DateModel", "value": new Date() }
        ],
        function(item,callback) {
          mongoose.model(item.model).create(item,callback)
        },
        callback
      );
    },
    function(callback) {
      MetaModel.find().exec(function(err,docs) {
        console.log(docs);
        callback(err);
      });
    },
    function(callback) {
      DateModel.findOne().exec(function(err,doc) {
        console.log(doc);
        callback(err);
      });
    }
  ],
  function(err) {
    if (err) throw err;
    mongoose.disconnect();
  }
)

因此,由于这些基本上是“相关的”,因此我定义了一个具有共同元素的“基本”模式。那么当然每个“类型”都有单独的模式。对核心“模型”的实际分配发生在这些行中:

var MetaModel = mongoose.model('MetaModel',metaSchema),
    StringModel = MetaModel.discriminator('StringModel', stringSchema),
    NumberModel = MetaModel.discriminator('NumberModel', numberSchema),
    DateModel = MetaModel.discriminator('DateModel', dateSchema);

这意味着 MetaModel 实际上定义了集合和“默认”模式分配。以下几行使用该模型中的 .discriminator() 来定义将存储在同一集合中的其他“类型”文档。

打开调试输出以显示正在发生的事情, list 产生如下内容:

Mongoose: metamodels.remove({}) {}
Mongoose: metamodels.insert({ value: 'Hello', __t: 'StringModel', created_at: new Date("Thu, 07 Apr 2016 00:24:08 GMT"), _id: ObjectId("5705a8a8443c0f74491bdec0"), __v: 0 })
Mongoose: metamodels.insert({ value: 12, __t: 'NumberModel', created_at: new Date("Thu, 07 Apr 2016 00:24:08 GMT"), _id: ObjectId("5705a8a8443c0f74491bdec1"), __v: 0 })
Mongoose: metamodels.insert({ value: new Date("Thu, 07 Apr 2016 00:24:08 GMT"), __t: 'DateModel', created_at: new Date("Thu, 07 Apr 2016 00:24:08 GMT"), _id: ObjectId("5705a8a8443c0f74491bdec2"), __v: 0 })
Mongoose: metamodels.find({}) { fields: undefined }
[ { created_at: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
    __t: 'StringModel',
    __v: 0,
    value: 'Hello',
    _id: 5705a8a8443c0f74491bdec0 },
  { created_at: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
    __t: 'NumberModel',
    __v: 0,
    value: 12,
    _id: 5705a8a8443c0f74491bdec1 },
  { created_at: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
    __t: 'DateModel',
    __v: 0,
    value: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
    _id: 5705a8a8443c0f74491bdec2 } ]
Mongoose: metamodels.findOne({ __t: 'DateModel' }) { fields: undefined }
{ created_at: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
  __t: 'DateModel',
  __v: 0,
  value: Thu Apr 07 2016 10:24:08 GMT+1000 (AEST),
  _id: 5705a8a8443c0f74491bdec2 }

您可以看到所有内容都在分配给主模型的 metamodels 集合中创建,但是当引用每个“鉴别器模型”时,会自动创建一个 __t 字段其中包括型号名称。这将在稍后读取数据时使用,以便 mongoose 知道在转换对象时应用哪个模型和附加模式。

当然,由于这些都有自己的架构,因此适用标准验证规则。此外,附加到每种类型的模式的任何“实例方法”也适用,就像它适用于任何单独的模型一样。

最后,__t 字段也适用于使用“鉴别器模型”之一进行任何其他操作,例如查询或更新。如最后执行的语句所示:

      DateModel.findOne().exec(function(err,doc) {
        console.log(doc);
        callback(err);
      });

实际调用:

Mongoose: metamodels.findOne({ __t: 'DateModel' }) { fields: undefined }

自动包含该属性值以指示“类型”并提供集合数据的“虚拟 View ”,就好像它只包含该特定类型一样。

真正的力量实际上在于所有对象都在同一个集合中,以及 Mongoose 在检索数据时自动分配“类类型”的能力。

关于javascript - 带日期的 Mongoose 混合类型字段,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36463539/

相关文章:

node.js - MongoDb类型错误: Cannot read property 'insertMany' of undefined

javascript - 鼠标悬停时替换部分 URL 以获取原始尺寸图像

在 Div 中发布的 Javascript

c# - MVC4 + jQuery 淡出通知 div

node.js - 如何使用 Nestjs 中的请求范围提供程序动态更改数据库连接?

mongodb - 在 Mongo 中使用具有两级层次结构的位置运算符

mongodb - 加速 MongoDB 聚合

javascript - 如果点击背景不工作隐藏容器

node.js - Mongoose 按字段查找?

javascript - Node.js 异步函数