在Sequelize中预先保存钩子和实例方法?

 石蓉亮 发布于 2023-01-30 18:43

Sequelize.js中有预先保存的钩子和实例方法吗?

具体来说,我需要将此Mongoose代码转换为等效的Sequelize代码:

架构

var userSchema = new mongoose.Schema({
  username: { type: String, unique: true },
  email: { type: String, unique: true },
  password: String,
  token: String
});

预存

userSchema.pre('save', function(next) {
  var user = this;

  var hashContent = user.username + user.password + Date.now() + Math.random();
  user.token = crypto.createHash('sha1').update(hashContent).digest('hex');

  if (!user.isModified('password')) return next();
  bcrypt.genSalt(5, function(err, salt) {
    if (err) return next(err);
    bcrypt.hash(user.password, salt, function(err, hash) {
      if (err) return next(err);
      user.password = hash;
      next();
    });
  });
});

实例方法

userSchema.methods.comparePassword = function(candidatePassword, cb) {
  bcrypt.compare(candidatePassword, this.password, function(err, isMatch) {
    if(err) return cb(err);
    cb(null, isMatch);
  });
};

Dan Rocha.. 9

最好的方法是使用类或实例方法扩展模型:

var User = sequelize.define('User', {
    username: { type: Sequelize.STRING, unique: true },
    email: { type: Sequelize.STRING, unique: true },
    password: Sequelize.STRING,
    token: Sequelize.STRING
}, {
    instanceMethods: {
        comparePassword : function(candidatePassword, cb) {
            bcrypt.compare(candidatePassword, this.getDataValue('password'), function(err, isMatch) {
                if(err) return cb(err);
                cb(null, isMatch);
            });
        },
        setToken: function(){
            // bla bla bla
            // bla bla bla
        },
        getFullname: function() {
            return [this.firstname, this.lastname].join(' ');
        }
    }
})

所以,当你这样做

User.build({ firstname: 'foo', lastname: 'bar' }).getFullname(); // 'foo bar'

因此,要设置令牌,您可以这样做:

User.build({ ... }).setToken().save();

或者,使用comparePassword函数:

User.find({ ... }).success(function(user) { user.comparePassword('the password to check', function(err, isMatch) { ... } });

您可以在Sequelize文档中看到这一点

编辑

在最新版本的每个模型都有钩子,你可以查看非常简单的钩子文档,并用类或实例方法补充它们.

2 个回答
  • 最好的方法是使用类或实例方法扩展模型:

    var User = sequelize.define('User', {
        username: { type: Sequelize.STRING, unique: true },
        email: { type: Sequelize.STRING, unique: true },
        password: Sequelize.STRING,
        token: Sequelize.STRING
    }, {
        instanceMethods: {
            comparePassword : function(candidatePassword, cb) {
                bcrypt.compare(candidatePassword, this.getDataValue('password'), function(err, isMatch) {
                    if(err) return cb(err);
                    cb(null, isMatch);
                });
            },
            setToken: function(){
                // bla bla bla
                // bla bla bla
            },
            getFullname: function() {
                return [this.firstname, this.lastname].join(' ');
            }
        }
    })
    

    所以,当你这样做

    User.build({ firstname: 'foo', lastname: 'bar' }).getFullname(); // 'foo bar'
    

    因此,要设置令牌,您可以这样做:

    User.build({ ... }).setToken().save();
    

    或者,使用comparePassword函数:

    User.find({ ... }).success(function(user) { user.comparePassword('the password to check', function(err, isMatch) { ... } });
    

    您可以在Sequelize文档中看到这一点

    编辑

    在最新版本的每个模型都有钩子,你可以查看非常简单的钩子文档,并用类或实例方法补充它们.

    2023-01-30 18:45 回答
  • 我遇到了同样的问题,但是至少在2.0版本的sequelize中可以使用此功能,完整的文档可在Hooks上获得。

    以下是使用beforeValidate挂钩的示例代码:

    "use strict";
    var md5 = require('blueimp-md5').md5;
    
    module.exports = function(sequelize, DataTypes) {
      var Sms = sequelize.define("sms", {
        senderName: DataTypes.STRING,
        smsBody : {
          type : DataTypes.STRING, allowNull:false
        },
        userId : {
          type: DataTypes.INTEGER, allowNull:false
        },
        hash : {
          type:DataTypes.CHAR(32),
          unique:true,
          allowNull:false
        }
    
      });
    
    Sms.beforeValidate(function(sms){
      sms.hash = md5(sms.smsBody+sms.userId);
      return sequelize.Promise.resolve(sms)
    });
    
    
    return Sms;
    };
    

    这里的要求是,使用smsBody和userId创建一个哈希,因此我创建了一个钩子,即beforeValidate,该钩子将在Sequelize对模型执行任何验证之前执行。还有许多其他可用的挂钩,最好的部分是您在保存数据时不必编写任何其他代码,这些挂钩将解决这一问题。

    您应该在钩子和instanceMethods之间明智地选择。但是在您的情况下,我想钩子会是一个更好的选择

    2023-01-30 18:45 回答
撰写答案
今天,你开发时遇到什么问题呢?
立即提问
热门标签
PHP1.CN | 中国最专业的PHP中文社区 | PNG素材下载 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有