2019-07-17 13:28:16 +03:00
|
|
|
const _ = require('lodash');
|
2020-08-11 16:01:16 +03:00
|
|
|
const security = require('@tryghost/security');
|
2020-08-11 14:51:16 +03:00
|
|
|
const constants = require('@tryghost/constants');
|
2020-04-30 22:26:12 +03:00
|
|
|
const errors = require('@tryghost/errors');
|
2021-10-05 12:32:32 +03:00
|
|
|
const tpl = require('@tryghost/tpl');
|
2019-07-17 13:28:16 +03:00
|
|
|
const models = require('../../models');
|
2020-05-28 13:57:02 +03:00
|
|
|
const urlUtils = require('../../../shared/url-utils');
|
2019-07-17 13:28:16 +03:00
|
|
|
const mail = require('../mail');
|
|
|
|
|
2021-10-05 12:32:32 +03:00
|
|
|
const messages = {
|
|
|
|
userNotFound: 'User not found.',
|
|
|
|
tokenLocked: 'Token locked',
|
|
|
|
resetPassword: 'Reset Password',
|
|
|
|
expired: {
|
|
|
|
message: 'Cannot reset password.',
|
|
|
|
context: 'Password reset link expired.',
|
|
|
|
help: 'Request a new password reset via the login form.'
|
|
|
|
},
|
|
|
|
invalidToken: {
|
|
|
|
message: 'Cannot reset password.',
|
|
|
|
context: 'Password reset link has already been used.',
|
|
|
|
help: 'Request a new password reset via the login form.'
|
|
|
|
},
|
|
|
|
corruptedToken: {
|
|
|
|
message: 'Cannot reset password.',
|
|
|
|
context: 'Invalid password reset link.',
|
|
|
|
help: 'Check if password reset link has been fully copied or request new password reset via the login form.'
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-17 13:28:16 +03:00
|
|
|
const tokenSecurity = {};
|
|
|
|
|
2021-06-23 15:54:28 +03:00
|
|
|
function generateToken(email, settingsAPI, transaction) {
|
|
|
|
const options = {context: {internal: true}, transacting: transaction};
|
2020-04-29 18:44:27 +03:00
|
|
|
let dbHash;
|
|
|
|
let token;
|
2019-07-17 13:28:16 +03:00
|
|
|
|
|
|
|
return settingsAPI.read(_.merge({key: 'db_hash'}, options))
|
|
|
|
.then((response) => {
|
|
|
|
dbHash = response.settings[0].value;
|
|
|
|
|
|
|
|
return models.User.getByEmail(email, options);
|
|
|
|
})
|
|
|
|
.then((user) => {
|
|
|
|
if (!user) {
|
2021-10-05 12:32:32 +03:00
|
|
|
throw new errors.NotFoundError({message: tpl(messages.userNotFound)});
|
2019-07-17 13:28:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
token = security.tokens.resetToken.generateHash({
|
|
|
|
expires: Date.now() + constants.ONE_DAY_MS,
|
|
|
|
email: email,
|
|
|
|
dbHash: dbHash,
|
|
|
|
password: user.get('password')
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
email: email,
|
|
|
|
resetToken: token
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function extractTokenParts(options) {
|
|
|
|
options.data.passwordreset[0].token = security.url.decodeBase64(options.data.passwordreset[0].token);
|
|
|
|
|
|
|
|
const tokenParts = security.tokens.resetToken.extract({
|
|
|
|
token: options.data.passwordreset[0].token
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!tokenParts) {
|
2020-04-30 22:26:12 +03:00
|
|
|
return Promise.reject(new errors.UnauthorizedError({
|
2021-10-05 12:32:32 +03:00
|
|
|
message: tpl(messages.corruptedToken.message),
|
|
|
|
context: tpl(messages.corruptedToken.context),
|
|
|
|
help: tpl(messages.corruptedToken.help)
|
2019-07-17 13:28:16 +03:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve({options, tokenParts});
|
|
|
|
}
|
|
|
|
|
|
|
|
// @TODO: use brute force middleware (see https://github.com/TryGhost/Ghost/pull/7579)
|
|
|
|
function protectBruteForce({options, tokenParts}) {
|
|
|
|
if (tokenSecurity[`${tokenParts.email}+${tokenParts.expires}`] &&
|
|
|
|
tokenSecurity[`${tokenParts.email}+${tokenParts.expires}`].count >= 10) {
|
2020-04-30 22:26:12 +03:00
|
|
|
return Promise.reject(new errors.NoPermissionError({
|
2021-10-05 12:32:32 +03:00
|
|
|
message: tpl(messages.tokenLocked)
|
2019-07-17 13:28:16 +03:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve({options, tokenParts});
|
|
|
|
}
|
|
|
|
|
|
|
|
function doReset(options, tokenParts, settingsAPI) {
|
|
|
|
let dbHash;
|
|
|
|
|
|
|
|
const data = options.data.passwordreset[0];
|
|
|
|
const resetToken = data.token;
|
|
|
|
const oldPassword = data.oldPassword;
|
|
|
|
const newPassword = data.newPassword;
|
|
|
|
|
|
|
|
return settingsAPI.read(_.merge({key: 'db_hash'}, _.omit(options, 'data')))
|
|
|
|
.then((response) => {
|
|
|
|
dbHash = response.settings[0].value;
|
|
|
|
|
|
|
|
return models.User.getByEmail(tokenParts.email, options);
|
|
|
|
})
|
|
|
|
.then((user) => {
|
|
|
|
if (!user) {
|
2021-10-05 12:32:32 +03:00
|
|
|
throw new errors.NotFoundError({message: tpl(messages.userNotFound)});
|
2019-07-17 13:28:16 +03:00
|
|
|
}
|
|
|
|
|
2020-09-16 08:42:21 +03:00
|
|
|
let compareResult = security.tokens.resetToken.compare({
|
2019-07-17 13:28:16 +03:00
|
|
|
token: resetToken,
|
|
|
|
dbHash: dbHash,
|
|
|
|
password: user.get('password')
|
|
|
|
});
|
|
|
|
|
2020-09-16 08:42:21 +03:00
|
|
|
if (!compareResult.correct) {
|
|
|
|
let error;
|
|
|
|
if (compareResult.reason === 'expired' || compareResult.reason === 'invalid_expiry') {
|
|
|
|
error = new errors.BadRequestError({
|
2021-10-05 12:32:32 +03:00
|
|
|
message: tpl(messages.expired.message),
|
|
|
|
context: tpl(messages.expired.context),
|
|
|
|
help: tpl(messages.expired.help)
|
2020-09-16 08:42:21 +03:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
error = new errors.BadRequestError({
|
2021-10-05 12:32:32 +03:00
|
|
|
message: tpl(messages.invalidToken.message),
|
|
|
|
context: tpl(messages.invalidToken.context),
|
|
|
|
help: tpl(messages.invalidToken.help)
|
2020-09-16 08:42:21 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.reject(error);
|
2019-07-17 13:28:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return models.User.changePassword({
|
|
|
|
oldPassword: oldPassword,
|
|
|
|
newPassword: newPassword,
|
|
|
|
user_id: user.id
|
|
|
|
}, options);
|
|
|
|
})
|
|
|
|
.then((updatedUser) => {
|
|
|
|
updatedUser.set('status', 'active');
|
|
|
|
return updatedUser.save(options);
|
|
|
|
})
|
2020-04-30 22:26:12 +03:00
|
|
|
.catch(errors.ValidationError, (err) => {
|
2019-07-17 13:28:16 +03:00
|
|
|
return Promise.reject(err);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
2020-04-30 22:26:12 +03:00
|
|
|
if (errors.utils.isIgnitionError(err)) {
|
2019-07-17 13:28:16 +03:00
|
|
|
return Promise.reject(err);
|
|
|
|
}
|
2020-04-30 22:26:12 +03:00
|
|
|
return Promise.reject(new errors.UnauthorizedError({err: err}));
|
2019-07-17 13:28:16 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function sendResetNotification(data, mailAPI) {
|
|
|
|
const adminUrl = urlUtils.urlFor('admin', true);
|
2020-05-06 15:19:47 +03:00
|
|
|
const resetToken = security.url.encodeBase64(data.resetToken);
|
|
|
|
const resetUrl = urlUtils.urlJoin(adminUrl, 'reset', resetToken, '/');
|
2021-03-03 19:15:37 +03:00
|
|
|
const emailData = {
|
|
|
|
resetUrl: resetUrl,
|
|
|
|
recipientEmail: data.email
|
|
|
|
};
|
2019-07-17 13:28:16 +03:00
|
|
|
|
|
|
|
const content = await mail.utils.generateContent({
|
2021-03-03 19:15:37 +03:00
|
|
|
data: emailData,
|
2019-07-17 13:28:16 +03:00
|
|
|
template: 'reset-password'
|
|
|
|
});
|
|
|
|
|
|
|
|
const payload = {
|
|
|
|
mail: [{
|
|
|
|
message: {
|
|
|
|
to: data.email,
|
2021-10-05 12:32:32 +03:00
|
|
|
subject: tpl(messages.resetPassword),
|
2019-07-17 13:28:16 +03:00
|
|
|
html: content.html,
|
|
|
|
text: content.text
|
|
|
|
},
|
|
|
|
options: {}
|
|
|
|
}]
|
|
|
|
};
|
|
|
|
|
|
|
|
return mailAPI.send(payload, {context: {internal: true}});
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
2020-05-05 21:37:53 +03:00
|
|
|
generateToken,
|
|
|
|
extractTokenParts,
|
|
|
|
protectBruteForce,
|
|
|
|
doReset,
|
2020-05-06 15:19:47 +03:00
|
|
|
sendResetNotification
|
2019-07-17 13:28:16 +03:00
|
|
|
};
|