Получение атрибутов схемы из модели Mongoose

Я использую Mongoose.js для создания моделей со схемами.

У меня есть список моделей (многих), и иногда я хотел бы получить атрибуты / ключи, которые составляют конкретную модель.

Есть ли способ вытащить схемы атрибутов для любой модели?

Например,

var mySchema = module.exports = new Schema({
  SID: {
    type: Schema.Types.ObjectId
    //, required: true
  },
  teams: {
    type: [String]
  },
  hats: [{
    val: String,
    dt: Date
  }],
  shields: [{
    val: String,
    dt: Date
  }],
  shoes: [{
    val: String,
    dt: Date
  }]
}

);

Можно ли вытащить / идентифицировать атрибуты схемы [SID, hats, teams, shields, shoes]??

10 ответов

Решение

Да, это возможно.

Каждая схема имеет paths свойство, которое выглядит примерно так (это пример моего кода):

paths: {
    number: [Object],
    'name.first': [Object],
    'name.last': [Object],
    ssn: [Object],
    birthday: [Object],
    'job.company': [Object],
    'job.position': [Object],
    'address.city': [Object],
    'address.state': [Object],
    'address.country': [Object],
    'address.street': [Object],
    'address.number': [Object],
    'address.zip': [Object],
    email: [Object],
    phones: [Object],
    tags: [Object],
    createdBy: [Object],
    createdAt: [Object],
    updatedBy: [Object],
    updatedAt: [Object],
    meta: [Object],
    _id: [Object],
    __v: [Object]
}

Вы можете получить доступ к этому через модель тоже. Это под Model.schema.paths,

Не хватает представителя, чтобы комментировать, но это также выплевывает список и перебирает все типы схем.

mySchema.schema.eachPath(function(path) {
    console.log(path);
});

следует распечатать:

number
name.first
name.last
ssn
birthday
job.company
job.position
address.city
address.state
address.country
address.street
address.number
address.zip
email
phones
tags
createdBy
createdAt
updatedBy
updatedAt
meta
_id
__v

Или вы можете получить все атрибуты в виде массива следующим образом:

var props = Object.keys(mySchema.schema.paths);

В моем решении используется модель мангуста.

Атрибуты схемы

const schema = {
  email: {
    type: String,
    required: 'email is required',
  },
  password: {
    type: String,
    required: 'password is required',
  },
};

Схема

const FooSchema = new Schema(schema);

Модель

const FooModel = model('Foo', FooSchema);

Получить атрибуты из модели:

Object.keys(FooModel.schema.tree)

Результат:

[
  'email',
  'password',
  '_id',
  '__v'
] 

Решение для lodash, функция, которая выбрала все свойства схемы, кроме указанных

_.mixin({ pickSchema: function (model, excluded) {
    var fields = [];
    model.schema.eachPath(function (path) {
       _.isArray(excluded) ? excluded.indexOf(path) < 0 ? fields.push(path) : false : path === excluded ? false : fields.push(path);
    });
    return fields;
   }
});

_.pickSchema(User, '_id'); // will return all fields except _id

_.pick(req.body, _.pickSchema(User, ['_id', 'createdAt', 'hidden'])) // all except specified properties

читайте больше здесь https://gist.github.com/styopdev/95f3fed98ce3ebaedf5c

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

import Todo from './todoModel'
import { validationResult } from 'express-validator'

const buildObject = (body) => {
    const data = {};
    const keys = Object.keys(Todo.schema.obj);
    keys.map(key => { if (body.hasOwnProperty(key)) data[key] = body[key] })
    return data;
}

const create = async (req, res) => {
    try {
        const errors = validationResult(req);
        if (!errors.isEmpty()) return res.json(errors);
        let toBeSaved = buildObject(req.body);
        const todo = new Todo(toBeSaved);
        const savedTodo = await todo.save();
        if (savedTodo) return res.json(savedTodo);
        return res.json({ 'sanitized': keys })
    } catch (error) {
        res.json({ error })
    }
}

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

let { title, description } = req.body;
let toBeSaved = { title, description };

Использование сокращенных имен свойств ES6

Принятый ответ не работал для меня. Но используя Mongoose 5.4.2 я смог получить ключи, выполнив следующее:

const mySchema = new Schema({ ... });

const arrayOfKeys = Object.keys(mySchema.obj);

Я использую машинопись, однако. Это могло быть проблемой.

Если вы хотите иметь только добавленные вами атрибуты, а не методы добавления ORM, которые начинаются с '$___', вы должны превратить документ в объект, а затем получить доступ к таким атрибутам:

Object.keys(document.toObject());

Если вы хотите иметь все значения свойств (включая вложенные и заполненные свойства), просто используйте toObject() метод:

let modelAttributes = null;
SomeModel.findById('someId').populate('child.name').exec().then((result) => {
  modelAttributes = result.toObject();
  console.log(modelAttributes);
});

Выход будет:

{
  id: 'someId',
  name: 'someName',
  child: {
    name: 'someChildName'
  }
  ...
}

Просто введите имя поля, которое вы хотите получить.

let fieldName = 'birthday'
console.log(mySchema.schema.paths[fieldName].instance);

Итерация по ключам или атрибутам

      for (var key in FooModel.schema.obj) {
    //do your stuff with key
}
Другие вопросы по тегам