mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-27 22:49:56 -05:00
abda6e6338
closes #10773 - The refactoring is a substitute for `urlService.utils` used previously throughout the codebase and now extracted into the separate module in Ghost-SDK - Added url-utils stubbing utility for test suites - Some tests had to be refactored to avoid double mocks (when url's are being reset inside of rested 'describe' groups)
229 lines
8.1 KiB
JavaScript
229 lines
8.1 KiB
JavaScript
const Promise = require('bluebird'),
|
|
{omit, merge} = require('lodash'),
|
|
pipeline = require('../../lib/promise/pipeline'),
|
|
mail = require('../../services/mail'),
|
|
urlUtils = require('../../lib/url-utils'),
|
|
localUtils = require('./utils'),
|
|
models = require('../../models'),
|
|
common = require('../../lib/common'),
|
|
security = require('../../lib/security'),
|
|
mailAPI = require('./mail'),
|
|
settingsAPI = require('./settings'),
|
|
docName = 'invites',
|
|
allowedIncludes = ['created_by', 'updated_by'],
|
|
unsafeAttrs = ['role_id'];
|
|
|
|
const invites = {
|
|
browse(options) {
|
|
let tasks;
|
|
|
|
function modelQuery(options) {
|
|
return models.Invite.findPage(options)
|
|
.then(({data, meta}) => {
|
|
return {
|
|
invites: data.map(model => model.toJSON(options)),
|
|
meta: meta
|
|
};
|
|
});
|
|
}
|
|
|
|
tasks = [
|
|
localUtils.validate(docName, {opts: localUtils.browseDefaultOptions}),
|
|
localUtils.convertOptions(allowedIncludes),
|
|
localUtils.handlePublicPermissions(docName, 'browse'),
|
|
modelQuery
|
|
];
|
|
|
|
return pipeline(tasks, options);
|
|
},
|
|
|
|
read(options) {
|
|
const attrs = ['id', 'email'];
|
|
let tasks;
|
|
|
|
function modelQuery(options) {
|
|
return models.Invite.findOne(options.data, omit(options, ['data']))
|
|
.then((model) => {
|
|
if (!model) {
|
|
return Promise.reject(new common.errors.NotFoundError({
|
|
message: common.i18n.t('errors.api.invites.inviteNotFound')
|
|
}));
|
|
}
|
|
|
|
return {
|
|
invites: [model.toJSON(options)]
|
|
};
|
|
});
|
|
}
|
|
|
|
tasks = [
|
|
localUtils.validate(docName, {attrs: attrs}),
|
|
localUtils.convertOptions(allowedIncludes),
|
|
localUtils.handlePublicPermissions(docName, 'read'),
|
|
modelQuery
|
|
];
|
|
|
|
return pipeline(tasks, options);
|
|
},
|
|
|
|
destroy(options) {
|
|
let tasks;
|
|
|
|
function modelQuery(options) {
|
|
return models.Invite.findOne({id: options.id}, omit(options, ['data']))
|
|
.then((invite) => {
|
|
if (!invite) {
|
|
throw new common.errors.NotFoundError({message: common.i18n.t('errors.api.invites.inviteNotFound')});
|
|
}
|
|
|
|
return invite.destroy(options).return(null);
|
|
});
|
|
}
|
|
|
|
tasks = [
|
|
localUtils.validate(docName, {opts: localUtils.idDefaultOptions}),
|
|
localUtils.convertOptions(allowedIncludes),
|
|
localUtils.handlePermissions(docName, 'destroy'),
|
|
modelQuery
|
|
];
|
|
|
|
return pipeline(tasks, options);
|
|
},
|
|
|
|
add(object, options) {
|
|
let loggedInUser = options.context.user,
|
|
tasks,
|
|
emailData,
|
|
invite;
|
|
|
|
function addInvite(options) {
|
|
const data = options.data;
|
|
|
|
return models.Invite.add(data.invites[0], omit(options, 'data'))
|
|
.then((_invite) => {
|
|
invite = _invite;
|
|
|
|
return settingsAPI.read({key: 'title'});
|
|
})
|
|
.then((response) => {
|
|
const adminUrl = urlUtils.urlFor('admin', true);
|
|
|
|
emailData = {
|
|
blogName: response.settings[0].value,
|
|
invitedByName: loggedInUser.get('name'),
|
|
invitedByEmail: loggedInUser.get('email'),
|
|
// @TODO: resetLink sounds weird
|
|
resetLink: urlUtils.urlJoin(adminUrl, 'signup', security.url.encodeBase64(invite.get('token')), '/')
|
|
};
|
|
|
|
return mail.utils.generateContent({data: emailData, template: 'invite-user'});
|
|
}).then((emailContent) => {
|
|
const payload = {
|
|
mail: [{
|
|
message: {
|
|
to: invite.get('email'),
|
|
subject: common.i18n.t('common.api.users.mail.invitedByName', {
|
|
invitedByName: emailData.invitedByName,
|
|
blogName: emailData.blogName
|
|
}),
|
|
html: emailContent.html,
|
|
text: emailContent.text
|
|
},
|
|
options: {}
|
|
}]
|
|
};
|
|
|
|
return mailAPI.send(payload, {context: {internal: true}});
|
|
}).then(() => {
|
|
options.id = invite.id;
|
|
return models.Invite.edit({status: 'sent'}, options);
|
|
}).then(() => {
|
|
invite.set('status', 'sent');
|
|
const inviteAsJSON = invite.toJSON();
|
|
|
|
return {
|
|
invites: [inviteAsJSON]
|
|
};
|
|
}).catch((error) => {
|
|
if (error && error.errorType === 'EmailError') {
|
|
const errorMessage = common.i18n.t('errors.api.invites.errorSendingEmail.error', {
|
|
message: error.message
|
|
});
|
|
const helpText = common.i18n.t('errors.api.invites.errorSendingEmail.help');
|
|
error.message = `${errorMessage} ${helpText}`;
|
|
common.logging.warn(error.message);
|
|
}
|
|
|
|
return Promise.reject(error);
|
|
});
|
|
}
|
|
|
|
function destroyOldInvite(options) {
|
|
const data = options.data;
|
|
|
|
return models.Invite.findOne({email: data.invites[0].email}, omit(options, 'data'))
|
|
.then((invite) => {
|
|
if (!invite) {
|
|
return Promise.resolve(options);
|
|
}
|
|
|
|
return invite.destroy(options);
|
|
})
|
|
.then(() => {
|
|
return options;
|
|
});
|
|
}
|
|
|
|
function validation(options) {
|
|
if (!options.data.invites[0].email) {
|
|
return Promise.reject(new common.errors.ValidationError({message: common.i18n.t('errors.api.invites.emailIsRequired')}));
|
|
}
|
|
|
|
if (!options.data.invites[0].role_id) {
|
|
return Promise.reject(new common.errors.ValidationError({message: common.i18n.t('errors.api.invites.roleIsRequired')}));
|
|
}
|
|
|
|
return options;
|
|
}
|
|
|
|
function checkIfUserExists(options) {
|
|
return models.User.findOne({email: options.data.invites[0].email}, options)
|
|
.then((user) => {
|
|
if (user) {
|
|
return Promise.reject(new common.errors.ValidationError({
|
|
message: common.i18n.t('errors.api.users.userAlreadyRegistered')
|
|
}));
|
|
}
|
|
|
|
return options;
|
|
});
|
|
}
|
|
|
|
function fetchLoggedInUser(options) {
|
|
return models.User.findOne({id: loggedInUser}, merge({}, omit(options, 'data'), {withRelated: ['roles']}))
|
|
.then((user) => {
|
|
if (!user) {
|
|
return Promise.reject(new common.errors.NotFoundError({message: common.i18n.t('errors.api.users.userNotFound')}));
|
|
}
|
|
|
|
loggedInUser = user;
|
|
return options;
|
|
});
|
|
}
|
|
|
|
tasks = [
|
|
localUtils.validate(docName, {opts: ['email']}),
|
|
localUtils.convertOptions(allowedIncludes),
|
|
validation,
|
|
localUtils.handlePermissions(docName, 'add', unsafeAttrs),
|
|
fetchLoggedInUser,
|
|
checkIfUserExists,
|
|
destroyOldInvite,
|
|
addInvite
|
|
];
|
|
|
|
return pipeline(tasks, object, options);
|
|
}
|
|
};
|
|
|
|
module.exports = invites;
|