MongooseJS在预挂钩期间修改文档

时间:2013-08-14 00:40:39

标签: node.js mongodb express mongoose compoundjs

我和猫鼬有一些问题。我的目标是在预保存期间,我可以修改对象,如果需要可以分割标记,或者在另一种情况下计算子文档持续时间的总和并在主文档中更新它。

我发现如果我加载模型,然后调用doc.update传递新数据,只有schema.pre('update', ...)个触发器,并且我的中间件中对this的任何更改都不会更新。我也尝试在更新中间件中使用this.set('...', ....);无效。

似乎我改为doc.save(...),然后按预期附加thisschema.pre('save', ...)的更改。除了将发布的变量扩展到我的模型的属性和保存之外,我没有看到任何方法来为此目的利用doc.update。

我的目标: - 通过doc.update(properties, ....)更新现有文档 - 在保存期间使用中间件来修改文档,执行高级验证以及更新相关文档 - 在更新期间使用中间件来修改文档,执行高级验证以及更新相关文档 - 可互换地使用model.findByIdAndUpdate,model.save,model.findById-> doc.update,model.findById-> doc.save,并且全部使用我的保存/更新中间件。

一些任意示例代码:

function loadLocation(c) {
    var self = this;
    c.Location.findById(c.params.id, function(err, location) {
        c.respondTo(function(format) {
            if (err | !location) {
                format.json(function() {
                    c.send(err ? {
                        code: 500,
                        error: err
                    } : {
                        code: 404,
                        error: 'Location Not Found!'
                    });
                });
                format.html(function() {
                    c.redirect(c.path_to.admin_locations);
                });
            } else {
                self.location = location;
                c.next();
            }
        });
    });
}

LocationController.prototype.update = function update(c) {
    var location = this.location;
    this.title = 'Edit Location Details';

    location.update(c.body.Location, function(err) {
        c.respondTo(function(format) {
            format.json(function() {
                c.send(err ? {
                    code: 500,
                    error: location && location.errors || err
                } : {
                    code: 200,
                    location: location.toObject()
                });
            });
            format.html(function() {
                if (err) {
                    c.flash('error', JSON.stringify(err));
                } else {
                    c.flash('info', 'Location updated');
                }
                c.redirect(c.path_to.admin_location(location.id));
            });
        });
    });
};

module.exports = function(compound) {
    var schema = mongoose.Schema({
        name: String,
        address: String,
        tags: [{ type: String, index: true }],
        geo: {
            type: {
                type: String,
            default:
                "Point"
            },
            coordinates: [Number] // Longitude, Latitude
        }
    });
    schema.index({
        geo: '2dsphere'
    });
    var Location = mongoose.model('Location', schema);
    Location.modelName = 'Location';
    compound.models.Location = Location;

    schema.pre('save', function(next) {
        if(typeof this.tags === 'string') {
            this.tags = this.tags.split(',');
        }
    });
};

==== *修订样本* ====

module.exports = function(compound) {
    var schema = mongoose.Schema({
        name: String,
        bio: String
    });

    schema.pre('save', function(next) {
        console.log('Saving...');
        this.bio = "Tristique sed magna tortor?"; 
        next();
    });

    schema.pre('update', function(next) {
        console.log('Updating...');
        this.bio = "Quis ac, aenean egestas?"; 
        next();
    });

    var Author = mongoose.model('Author', schema);
    Author.modelName = 'Author';
    compound.models.Location = Author;
};

2 个答案:

答案 0 :(得分:7)

pre个钩子适用于doc.save()doc.update()。在这两种情况下,this都指的是文档本身。

请注意,在编译模型之前,需要将钩子添加到模式中。

schema.pre('save', function(next) {
    if(typeof this.tags === 'string') {
        this.tags = this.tags.split(',');
    }
});
var Location = mongoose.model('Location', schema);

答案 1 :(得分:0)

Mongoose不支持模型更新API的挂钩。但是,可以通过Monkey-patch完成更新挂钩。 Hooker NPM包是干净利落的好方法。

RESTeasy项目是节点REST API的Boilerplate,其代码演示了如何执行此操作:

var hooker = require('hooker');

var BaseSchema = new mongoose.Schema({
  sampleString: {
    type: String
  }
});

var BaseModel = mongoose.model('Base', BaseSchema);

// Utilize hooks for update operations. We do it in this way because MongooseJS
// does not natively support update hooks at the Schema level. This is a way
// to support it.
hooker.hook (BaseModel, 'update', {
  pre: function () {
    // Insert any logic you want before updating to occur here
    console.log('BaseModel pre update');
  },
  post: function () {
    // Insert any logic you want after updating to occur here
    console.log('BaseModel post update');
  }
});

// Export the Mongoose model
module.exports = BaseModel;