在一个平均堆栈应用程序中,我在用户架构中创建一个新用户,然后在Doc Schema中创建一个新文档
var UserSchema = new Schema({
username: String,
password: String,
docs: [{
type: mongoose.Schema.Types.ObjectId,
ref: 'Doc'
}],
)
}
var DocSchema = new Schema({…)
}
UserSchema.pre('save', function(next) {
if (this.password) {
this.salt = new Buffer(crypto.randomBytes(16).toString('base64'), 'base64');
this.password = this.hashPassword(this.password);
}
next();
});
以下代码部分是护照注册,最后我遇到newUser.save();如果我不保存用户,则在推送方法之后,文档ID不会显示在用户文档中。但是保存用户似乎也改变了哈希密码。如果我评论newUser.save();登录工作正常,否则我收到错误的密码
passport.use('local-signup', new LocalStrategy({
usernameField: 'username',
passwordField: 'password',
passReqToCallback: true
},
function(req, username, password, done) {
process.nextTick(function() {
User.findOne({
username: username
}, function(err, user) {
// if there are any errors, return the error
if (err)
return done(err);
if (user) {
return done(null, false, req.flash('signupMessage', 'Username is already taken.'));
} else {
var newUser = new User();
newUser.username = username;
newUser.password = password;
newUser.email = req.body.email;
newUser.save(function(err) {
if (err)
throw err;
var doc = new Doc({
user: newUser.username,
docTitle: newUser.username
});
doc.save(function(err) { // create doc
if (err) {
return next(err);
} else {
newUser.docs.push(doc); // push doc'id in docs field in user
newUser.save(); // save user after doc'id has been push
};
return done(null, newUser);
});
});
任何帮助将不胜感激
答案 0 :(得分:1)
您的mongoose预存中间件中的逻辑表示如果正在保存的文档上有密码,则生成一个盐并哈希密码'。因此,如果已经加盐和散列的文档上存在密码,则当中间件运行时,它将再次对此预先存在的密码进行加密和散列。这就是您无法再次登录的原因;每次保存文档时,密码都会改变。
我猜你希望mongoose pre save中间件只在你第一次保存文件时才能运行。每次保存文档时都会运行预保存中间件。您可以使用预保存中间件中的文档中提供this.isNew
属性。这将确保密码仅在您第一次保存文档时生成。
UserSchema.pre('save', function(next) {
if (this.password && this.isNew) {
this.salt = new Buffer(crypto.randomBytes(16).toString('base64'), 'base64');
this.password = this.hashPassword(this.password);
}
next();
});