Поддерживает ли Mongoose метод Mongodb `findAndModify`? - PullRequest
32 голосов
/ 07 сентября 2011

Я хотел бы использовать findAndModify для атомарного увеличения поля, используя Mongoose.

Однако приведенный ниже код выдает ошибку «Ошибка типа: у объекта # нет метода 'findAndModify'":

// defining schema for the "counters" table
var tableSchema = new Schema({
    _id: String,
    next: Number        
});

// creating table object for the counters table
var counters_table = mongoose.model('counters', tableSchema);
var tableObj = new counters_table();    

// increment the "next" field on the table object
var query = {_id: 'messagetransaction'};
var update = {'$inc': {next: 1}};
var ret = tableObj.findAndModify(query, [], update, true, true, function(err) {
     if (err) { 
         throw err;
     }
     else { 
         console.log("updated!");
     }
});

Ответы [ 9 ]

61 голосов
/ 29 сентября 2011

Эта функция плохо документирована (читай: вообще), но после прочтения исходного кода я нашел следующее решение.

Создайте схему своей коллекции.

var Counters = new Schema({
  _id: String,
  next: Number     
});

Создайте статический метод в схеме, который предоставит метод findAndModify коллекции модели.

Counters.statics.findAndModify = function (query, sort, doc, options, callback) {
  return this.collection.findAndModify(query, sort, doc, options, callback);
};

Создайте свою модель.

var Counter = mongoose.model('counters', Counters);

Найдите и измените!

Counter.findAndModify({ _id: 'messagetransaction' }, [], { $inc: { next: 1 } }, {}, function (err, counter) {
  if (err) throw err;
  console.log('updated, counter is ' + counter.next);
});

Бонус

Counters.statics.increment = function (counter, callback) {
  return this.collection.findAndModify({ _id: counter }, [], { $inc: { next: 1 } }, callback);
};

Counter.increment('messagetransaction', callback);
18 голосов
/ 15 марта 2013
14 голосов
/ 22 октября 2012

Сделано приращение рабочей версии для Mongoose 3.x

var mongoose = require('mongoose');

var CounterSchema = new mongoose.Schema({
    _id: String,
    next: {type: Number, default: 1}
});

CounterSchema.statics.increment = function (counter, callback) {
    return this.findByIdAndUpdate(counter, { $inc: { next: 1 } }, {new: true, upsert: true, select: {next: 1}}, callback);
};

Используйте что-то вроде этого:

Counter.increment('photo', function (err, result) {
    if (err) {
        console.error('Counter on photo save error: ' + err); return;
    }
    photo.cid = result.next;
    photo.save();
});

Надеюсь, кому-нибудь пригодится

6 голосов
/ 02 апреля 2014

В версии 3 метод mongoose findOneAndUpdate предоставляет операцию mongodb findAndModify.Это работает так:

var query = { name: 'Sprinkls' };
var update = { name: 'Sprinkles' };
var options = { new: false };
Cat.findOneAndUpdate(query, update, options, function (err, cat) {
  if (err) ..
  render('cat', cat);
});

Больше информации здесь: http://aaronheckmann.tumblr.com/post/48943524629/mongoose-v3-part-2-findandmodify

3 голосов
/ 01 апреля 2014

много ответов, но я нахожу это простое решение.

Counter.findByIdAndUpdate(ID, {$inc: {next:1}}, function (err, data) {


});
3 голосов
/ 10 апреля 2012

Я нашел findAndModify для

  • Вставить счетчик (создать и инициализировать его, если он не существует)
  • Инкремент счетчика
  • Вызвать обратный вызов с увеличенным значением

в одну поездку БД с использованием следующего кода:

var Counters = new Schema({
  _id:String, // the schema name
  count: Number
});

Counters.statics.findAndModify = function (query, sort, doc, options, callback) {
    return this.collection.findAndModify(query, sort, doc, options, callback);
};

var Counter = mongoose.model('Counter', Counters);

/**
 * Increments the counter associated with the given schema name.
 * @param {string} schemaName The name of the schema for which to
 *   increment the associated counter.
 * @param {function(err, count)} The callback called with the updated
 *   count (a Number).
 */
function incrementCounter(schemaName, callback){
  Counter.findAndModify({ _id: schemaName }, [], 
    { $inc: { count: 1 } }, {"new":true, upsert:true}, function (err, result) {
      if (err)
        callback(err);
      else
        callback(null, result.count);
  });
}

Наслаждайтесь! - Curran

1 голос
/ 23 января 2017

Принимая вышеуказанный ответ от @ furf , это мое promised решение:

// eslint-disable-next-line func-names
localeTypesSchema.statics.findAndModify = function (query, sort, update, opts, callback) {
    const cb = callback || (() => { });
    try {
        const result = this.collection.findAndModify(query || {}, sort || [], update || {}, opts);
        cb(null, result);
        return Promise.resolve(result);
    } catch (err) {
        cb(err);
        return Promise.reject(err);
    }
};
0 голосов
/ 07 июля 2014

просто добавив к furf ответ, что если вы используете objectId в своем запросе, mongoDB не сможет найти ваш документ.Слой мангусты заботится о преобразовании идентификатора шестнадцатеричного строкового объекта, полученного из параметров маршрутизации, в правильный идентификатор объекта.

, для решения которого вам необходимо:

var ObjectID = require('mongodb').ObjectID;


var itemId = req.params.itemId;
var objectId = ObjectID.createFromHexString(itemId);
Item.findAndModify({_id: objectId},
0 голосов
/ 07 сентября 2011

Я бы предложил использовать стиль прямой команды, показанный внизу http://www.mongodb.org/display/DOCS/findAndModify+Command. Я недостаточно знаком с mongoose, чтобы знать метод запуска команды, но все драйверы предоставляют какой-то способ сделать это.Если мангуста нет, вы можете сделать это напрямую, используя стиль, описанный в верхней части http://www.mongodb.org/display/DOCS/Commands.

Тем не менее, вы должны убедиться, что вам действительно нужно findAndModify и что updateне буду делать то, что вам нужно.Чтобы увидеть, на что update способен взглянуть на http://www.mongodb.org/display/DOCS/Updating.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...