Несколько ссылок схемы в едином массиве схем - mongoose

Можно ли заполнить массив в схеме mongoose со ссылками на несколько разных параметров схемы?

Чтобы немного прояснить вопрос, скажем, у меня есть следующие схемы:

var scenarioSchema = Schema({
  _id     : Number,
  name    : String,
  guns : []
});

var ak47 = Schema({
  _id     : Number
  //Bunch of AK specific parameters
});

var m16 = Schema({
  _id     : Number
  //Bunch of M16 specific parameters
});

Можно ли заполнить массив орудий связкой ak47 ИЛИ m16? Можно ли помещать BOTH в тот же массив оружия? Или требуется ли заполнять ref в массиве активов, как это, что ограничивает его одним конкретным типом?

guns: [{ type: Schema.Types.ObjectId, ref: 'm16' }]

Я знаю, что у меня могут быть только отдельные массивы для разных типов оружия, но это создаст безумное количество дополнительных полей в схеме, поскольку масштаб проекта, большинство из которых останется пустым в зависимости от загруженного сценария.

var scenarioSchema = Schema({
  _id     : Number,
  name    : String,
  ak47s : [{ type: Schema.Types.ObjectId, ref: 'ak47' }],
  m16s: [{ type: Schema.Types.ObjectId, ref: 'm16' }]
});

Итак, вернемся к вопросу, могу ли я использовать несколько ссылок схемы в одном массиве?

Ответ 1

Здесь вы найдете метод mongoose .discriminator(). Это в основном позволяет хранить объекты разных типов в одной и той же коллекции, но иметь их как отличные объекты первого класса.

Обратите внимание, что принцип "того же набора" здесь важен для того, как работает .populate() и определение ссылки в содержащейся модели. Так как вы действительно можете указывать только на "одну" модель для ссылки, но есть и другая магия, которая может сделать одну модель такой, как многие.

Пример:

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

mongoose.connect('mongodb://localhost/gunshow');

//mongoose.set("debug",true);

var scenarioSchema = new Schema({
  "name": String,
  "guns": [{ "type": Schema.Types.ObjectId, "ref": "Gun" }]
});

function BaseSchema() {
  Schema.apply(this, arguments);

  // Common Gun stuff
  this.add({
    "createdAt": { "type": Date, "default": Date.now }
  });
}

util.inherits(BaseSchema, Schema);

var gunSchema = new BaseSchema();

var ak47Schema = new BaseSchema({
  // Ak74 stuff
});

ak47Schema.methods.shoot = function() {
  return "Crack!Crack";
};

var m16Schema = new BaseSchema({
  // M16 Stuff
});

m16Schema.methods.shoot = function() {
  return "Blam!!"
};


var Scenario = mongoose.model("Scenario", scenarioSchema);

var Gun = mongoose.model("Gun", gunSchema );
var Ak47 = Gun.discriminator("Ak47", ak47Schema );
var M16 = Gun.discriminator("M16", m16Schema );


async.series(
  [
    // Cleanup
    function(callback) {
      async.each([Scenario,Gun],function(model,callback) {
        model.remove({},callback);
      },callback);
    },

    // Add some guns and add to scenario
    function(callback) {
      async.waterfall(
        [
          function(callback) {
            async.map([Ak47,M16],function(gun,callback) {
              gun.create({},callback);
            },callback);
          },
          function(guns,callback) {
            Scenario.create({
              "name": "Test",
              "guns": guns
            },callback);
          }
        ],
        callback
      );
    },

    // Get populated scenario
    function(callback) {
      Scenario.findOne().populate("guns").exec(function(err,data) {

        console.log("Populated:\n%s",JSON.stringify(data,undefined,2));

        // Shoot each gun for fun!
        data.guns.forEach(function(gun) {
          console.log("%s says %s",gun.__t,gun.shoot());
        });

        callback(err);
      });
    },

    // Show the Guns collection
    function(callback) {
      Gun.find().exec(function(err,guns) {
        console.log("Guns:\n%s", JSON.stringify(guns,undefined,2));
        callback(err);
      });
    },

    // Show magic filtering
    function(callback) {
      Ak47.find().exec(function(err,ak47) {
        console.log("Magic!:\n%s", JSON.stringify(ak47,undefined,2));
        callback(err);
      });
    }
  ],
  function(err) {
    if (err) throw err;
    mongoose.disconnect();
  }
);

И вывод

Populated:
{
  "_id": "56c508069d16fab84ead921d",
  "name": "Test",
  "__v": 0,
  "guns": [
    {
      "_id": "56c508069d16fab84ead921b",
      "__v": 0,
      "__t": "Ak47",
      "createdAt": "2016-02-17T23:53:42.853Z"
    },
    {
      "_id": "56c508069d16fab84ead921c",
      "__v": 0,
      "__t": "M16",
      "createdAt": "2016-02-17T23:53:42.862Z"
    }
  ]
}
Ak47 says Crack!Crack
M16 says Blam!!
Guns:
[
  {
    "_id": "56c508069d16fab84ead921b",
    "__v": 0,
    "__t": "Ak47",
    "createdAt": "2016-02-17T23:53:42.853Z"
  },
  {
    "_id": "56c508069d16fab84ead921c",
    "__v": 0,
    "__t": "M16",
    "createdAt": "2016-02-17T23:53:42.862Z"
  }
]
Magic!:
[
  {
    "_id": "56c508069d16fab84ead921b",
    "__v": 0,
    "__t": "Ak47",
    "createdAt": "2016-02-17T23:53:42.853Z"
  }
]

Вы также можете раскомментировать строку mongoose.set("debug",true) в списке, чтобы увидеть, как мангуст фактически создает вызовы.

Итак, это демонстрирует, что вы можете применять разные схемы к различным объектам первого класса и даже к различным приложенным к ним методам, как к реальным объектам. Mongoose хранит все это в коллекции "guns" с прикрепленной моделью и будет содержать все "типы", на которые ссылается дискриминатор:

var Gun = mongoose.model("Gun", gunSchema );
var Ak47 = Gun.discriminator("Ak47", ak47Schema );
var M16 = Gun.discriminator("M16", m16Schema );

Но и каждый другой "тип" ссылается на свою собственную модель особым образом. Таким образом, вы видите, что когда mongoose хранит и читает объект, есть специальное поле __t, которое сообщает ему, какая "модель" применяется, и, следовательно, прикрепленная схема.

В качестве одного примера мы называем метод .shoot(), который по-разному определяется для каждой модели/схемы. А также вы можете использовать их как модель самостоятельно для запросов или других операций, так как Ak47 будет автоматически применять значение __t во всех запросах/версиях.

Таким образом, хотя хранилище находится в одной коллекции, оно может показаться множеством коллекций, но также имеет преимущество для их совместного хранения для других полезных операций. Вот как вы можете применить вид "полиморфизма", который вы ищете.