2014-06-03 08:05:25 -05:00
|
|
|
// # Users API
|
|
|
|
// RESTful API for the User resource
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
var when = require('when'),
|
|
|
|
_ = require('lodash'),
|
|
|
|
dataProvider = require('../models'),
|
|
|
|
settings = require('./settings'),
|
|
|
|
canThis = require('../permissions').canThis,
|
|
|
|
errors = require('../errors'),
|
|
|
|
utils = require('./utils'),
|
2014-07-02 09:22:18 -05:00
|
|
|
globalUtils = require('../utils'),
|
|
|
|
config = require('../config'),
|
|
|
|
mail = require('./mail'),
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
|
|
|
|
docName = 'users',
|
2014-07-08 11:00:59 -05:00
|
|
|
// TODO: implement created_by, updated_by
|
|
|
|
allowedIncludes = ['permissions', 'roles', 'roles.permissions'],
|
2013-12-06 03:51:35 -05:00
|
|
|
users;
|
|
|
|
|
2014-07-08 11:00:59 -05:00
|
|
|
// ## Helpers
|
|
|
|
function prepareInclude(include) {
|
|
|
|
include = _.intersection(include.split(','), allowedIncludes);
|
|
|
|
return include;
|
|
|
|
}
|
|
|
|
|
2014-06-03 08:05:25 -05:00
|
|
|
/**
|
|
|
|
* ## Posts API Methods
|
|
|
|
*
|
|
|
|
* **See:** [API Methods](index.js.html#api%20methods)
|
|
|
|
*/
|
2013-12-06 03:51:35 -05:00
|
|
|
users = {
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
/**
|
|
|
|
* ## Browse
|
|
|
|
* Fetch all users
|
2014-06-03 08:05:25 -05:00
|
|
|
* @param {{context}} options (optional)
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
* @returns {Promise(Users)} Users Collection
|
|
|
|
*/
|
2013-12-06 03:51:35 -05:00
|
|
|
browse: function browse(options) {
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
options = options || {};
|
|
|
|
return canThis(options.context).browse.user().then(function () {
|
2014-07-08 11:00:59 -05:00
|
|
|
if (options.include) {
|
|
|
|
options.include = prepareInclude(options.include);
|
|
|
|
}
|
2014-07-20 11:42:03 -05:00
|
|
|
return dataProvider.User.findPage(options);
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function (error) {
|
|
|
|
return errors.handleAPIError(error);
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-06-03 08:05:25 -05:00
|
|
|
/**
|
|
|
|
* ### Read
|
|
|
|
* @param {{id, context}} options
|
|
|
|
* @returns {Promise(User)} User
|
|
|
|
*/
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
read: function read(options) {
|
2014-07-08 17:28:31 -05:00
|
|
|
var attrs = ['id', 'slug', 'email'],
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
data = _.pick(options, attrs);
|
|
|
|
|
|
|
|
options = _.omit(options, attrs);
|
|
|
|
|
2014-07-08 11:00:59 -05:00
|
|
|
if (options.include) {
|
|
|
|
options.include = prepareInclude(options.include);
|
|
|
|
}
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
if (data.id === 'me' && options.context && options.context.user) {
|
|
|
|
data.id = options.context.user;
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
return dataProvider.User.findOne(data, options).then(function (result) {
|
2013-12-06 03:51:35 -05:00
|
|
|
if (result) {
|
2014-05-06 05:14:58 -05:00
|
|
|
return { users: [result.toJSON()] };
|
2013-12-06 03:51:35 -05:00
|
|
|
}
|
|
|
|
|
2014-05-09 05:11:29 -05:00
|
|
|
return when.reject(new errors.NotFoundError('User not found.'));
|
2013-12-06 03:51:35 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-06-03 08:05:25 -05:00
|
|
|
/**
|
|
|
|
* ### Edit
|
|
|
|
* @param {User} object the user details to edit
|
|
|
|
* @param {{id, context}} options
|
|
|
|
* @returns {Promise(User)}
|
|
|
|
*/
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
edit: function edit(object, options) {
|
2014-07-24 04:46:05 -05:00
|
|
|
var editOperation;
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
if (options.id === 'me' && options.context && options.context.user) {
|
|
|
|
options.id = options.context.user;
|
|
|
|
}
|
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
if (options.include) {
|
|
|
|
options.include = prepareInclude(options.include);
|
|
|
|
}
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 07:41:19 -05:00
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
return utils.checkObject(object, docName).then(function (data) {
|
|
|
|
// Edit operation
|
|
|
|
editOperation = function () {
|
|
|
|
return dataProvider.User.edit(data.users[0], options)
|
|
|
|
.then(function (result) {
|
|
|
|
if (result) {
|
|
|
|
return { users: [result.toJSON()]};
|
|
|
|
}
|
|
|
|
|
|
|
|
return when.reject(new errors.NotFoundError('User not found.'));
|
|
|
|
});
|
|
|
|
};
|
2014-07-08 11:00:59 -05:00
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
// Check permissions
|
|
|
|
return canThis(options.context).edit.user(options.id).then(function () {
|
|
|
|
if (data.users[0].roles) {
|
|
|
|
if (options.id === options.context.user) {
|
|
|
|
return when.reject(new errors.NoPermissionError('You cannot change your own role.'));
|
|
|
|
}
|
|
|
|
return canThis(options.context).assign.role(data.users[0].roles[0]).then(function () {
|
|
|
|
return editOperation();
|
|
|
|
});
|
2014-04-08 08:40:33 -05:00
|
|
|
}
|
2013-12-06 03:51:35 -05:00
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
return editOperation();
|
|
|
|
|
2014-06-20 04:15:01 -05:00
|
|
|
});
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function (error) {
|
|
|
|
return errors.handleAPIError(error);
|
2014-06-20 04:15:01 -05:00
|
|
|
});
|
|
|
|
},
|
2014-04-03 08:03:09 -05:00
|
|
|
|
2014-07-02 09:22:18 -05:00
|
|
|
/**
|
2014-07-11 07:17:09 -05:00
|
|
|
* ### Add user
|
|
|
|
* The newly added user is invited to join the blog via email.
|
2014-07-02 09:22:18 -05:00
|
|
|
* @param {User} object the user to create
|
|
|
|
* @returns {Promise(User}} Newly created user
|
|
|
|
*/
|
2014-07-11 07:17:09 -05:00
|
|
|
add: function add(object, options) {
|
2014-07-02 09:22:18 -05:00
|
|
|
var newUser,
|
2014-07-24 04:46:05 -05:00
|
|
|
user,
|
2014-07-29 08:35:48 -05:00
|
|
|
roleId,
|
|
|
|
emailData;
|
2014-07-02 09:22:18 -05:00
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
return canThis(options.context).add.user(object).then(function () {
|
2014-07-02 09:22:18 -05:00
|
|
|
return utils.checkObject(object, docName).then(function (checkedUserData) {
|
2014-07-08 11:00:59 -05:00
|
|
|
if (options.include) {
|
|
|
|
options.include = prepareInclude(options.include);
|
|
|
|
}
|
|
|
|
|
2014-07-02 09:22:18 -05:00
|
|
|
newUser = checkedUserData.users[0];
|
2014-07-30 10:40:30 -05:00
|
|
|
|
|
|
|
if (_.isEmpty(newUser.roles)) {
|
|
|
|
return when.reject(new errors.BadRequestError('No role provided.'));
|
|
|
|
}
|
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
roleId = parseInt(newUser.roles[0].id || newUser.roles[0], 10);
|
2014-07-02 09:22:18 -05:00
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
// Make sure user is allowed to add a user with this role
|
|
|
|
return dataProvider.Role.findOne({id: roleId}).then(function (role) {
|
|
|
|
if (role.get('name') === 'Owner') {
|
|
|
|
return when.reject(new errors.NoPermissionError('Not allowed to create an owner user.'));
|
2014-07-23 01:13:20 -05:00
|
|
|
}
|
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
return canThis(options.context).assign.role(role);
|
|
|
|
}).then(function () {
|
2014-07-23 01:13:20 -05:00
|
|
|
if (newUser.email) {
|
|
|
|
newUser.name = object.users[0].email.substring(0, newUser.email.indexOf('@'));
|
|
|
|
newUser.password = globalUtils.uid(50);
|
|
|
|
newUser.status = 'invited';
|
|
|
|
} else {
|
|
|
|
return when.reject(new errors.BadRequestError('No email provided.'));
|
|
|
|
}
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function () {
|
|
|
|
return when.reject(new errors.NoPermissionError('Not allowed to create user with that role.'));
|
2014-07-23 01:13:20 -05:00
|
|
|
});
|
2014-07-02 09:22:18 -05:00
|
|
|
}).then(function () {
|
|
|
|
return dataProvider.User.getByEmail(newUser.email);
|
|
|
|
}).then(function (foundUser) {
|
|
|
|
if (!foundUser) {
|
|
|
|
return dataProvider.User.add(newUser, options);
|
|
|
|
} else {
|
|
|
|
// only invitations for already invited users are resent
|
2014-07-13 13:36:27 -05:00
|
|
|
if (foundUser.get('status') === 'invited' || foundUser.get('status') === 'invited-pending') {
|
2014-07-02 09:22:18 -05:00
|
|
|
return foundUser;
|
|
|
|
} else {
|
|
|
|
return when.reject(new errors.BadRequestError('User is already registered.'));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}).then(function (invitedUser) {
|
|
|
|
user = invitedUser.toJSON();
|
|
|
|
return settings.read({context: {internal: true}, key: 'dbHash'});
|
|
|
|
}).then(function (response) {
|
2014-07-28 08:19:49 -05:00
|
|
|
var expires = Date.now() + (14 * globalUtils.ONE_DAY_MS),
|
2014-07-02 09:22:18 -05:00
|
|
|
dbHash = response.settings[0].value;
|
|
|
|
return dataProvider.User.generateResetToken(user.email, expires, dbHash);
|
|
|
|
}).then(function (resetToken) {
|
2014-07-29 08:35:48 -05:00
|
|
|
return when.join(users.read({'id': user.created_by}), settings.read({'key': 'title'})).then(function (values) {
|
2014-07-22 20:22:13 -05:00
|
|
|
var invitedBy = values[0].users[0],
|
|
|
|
blogTitle = values[1].settings[0].value,
|
|
|
|
baseUrl = config.forceAdminSSL ? (config.urlSSL || config.url) : config.url,
|
2014-07-29 08:35:48 -05:00
|
|
|
resetUrl = baseUrl.replace(/\/$/, '') + '/ghost/signup/' + resetToken + '/';
|
|
|
|
|
|
|
|
emailData = {
|
|
|
|
blogName: blogTitle,
|
|
|
|
invitedByName: invitedBy.name,
|
|
|
|
invitedByEmail: invitedBy.email,
|
|
|
|
resetLink: resetUrl
|
|
|
|
};
|
|
|
|
|
|
|
|
return mail.generateContent({data: emailData, template: 'invite-user'});
|
|
|
|
}).then(function (emailContent) {
|
|
|
|
var payload = {
|
|
|
|
mail: [
|
|
|
|
{
|
|
|
|
message: {
|
|
|
|
to: user.email,
|
|
|
|
subject: emailData.invitedByName + ' has invited you to join ' + emailData.blogName,
|
|
|
|
html: emailContent.html,
|
|
|
|
text: emailContent.text
|
|
|
|
},
|
|
|
|
options: {}
|
2014-07-22 20:22:13 -05:00
|
|
|
}
|
2014-07-29 08:35:48 -05:00
|
|
|
]
|
|
|
|
};
|
|
|
|
return mail.send(payload, {context: {internal: true}}).then(function () {
|
|
|
|
// If status was invited-pending and sending the invitation succeeded, set status to invited.
|
|
|
|
if (user.status === 'invited-pending') {
|
|
|
|
return dataProvider.User.edit({status: 'invited'}, {id: user.id}).then(function (editedUser) {
|
|
|
|
user = editedUser.toJSON();
|
|
|
|
});
|
|
|
|
}
|
2014-07-22 20:22:13 -05:00
|
|
|
});
|
2014-07-13 13:36:27 -05:00
|
|
|
});
|
2014-07-02 09:22:18 -05:00
|
|
|
}).then(function () {
|
|
|
|
return when.resolve({users: [user]});
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function (error) {
|
2014-07-02 09:22:18 -05:00
|
|
|
if (error && error.type === 'EmailError') {
|
|
|
|
error.message = 'Error sending email: ' + error.message + ' Please check your email settings and resend the invitation.';
|
2014-07-23 01:13:20 -05:00
|
|
|
errors.logWarn(error.message);
|
|
|
|
|
2014-07-13 13:36:27 -05:00
|
|
|
// If sending the invitation failed, set status to invited-pending
|
2014-07-23 01:13:20 -05:00
|
|
|
return dataProvider.User.edit({status: 'invited-pending'}, {id: user.id}).then(function (user) {
|
|
|
|
return dataProvider.User.findOne({ id: user.id }, options).then(function (user) {
|
|
|
|
return { users: [user] };
|
|
|
|
});
|
2014-07-13 13:36:27 -05:00
|
|
|
});
|
2014-07-02 09:22:18 -05:00
|
|
|
}
|
|
|
|
return when.reject(error);
|
|
|
|
});
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function (error) {
|
|
|
|
return errors.handleAPIError(error);
|
2014-07-02 09:22:18 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-07-24 04:46:05 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* ### Destroy
|
|
|
|
* @param {{id, context}} options
|
|
|
|
* @returns {Promise(User)}
|
|
|
|
*/
|
|
|
|
destroy: function destroy(options) {
|
|
|
|
return canThis(options.context).destroy.user(options.id).then(function () {
|
|
|
|
return users.read(options).then(function (result) {
|
|
|
|
return dataProvider.User.destroy(options).then(function () {
|
|
|
|
return result;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}).catch(function (error) {
|
|
|
|
return errors.handleAPIError(error);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
|
2014-06-20 04:15:01 -05:00
|
|
|
/**
|
|
|
|
* ### Change Password
|
2014-07-08 17:28:31 -05:00
|
|
|
* @param {password} object
|
2014-06-20 04:15:01 -05:00
|
|
|
* @param {{context}} options
|
|
|
|
* @returns {Promise(password}} success message
|
|
|
|
*/
|
|
|
|
changePassword: function changePassword(object, options) {
|
|
|
|
var oldPassword,
|
|
|
|
newPassword,
|
|
|
|
ne2Password;
|
|
|
|
return utils.checkObject(object, 'password').then(function (checkedPasswordReset) {
|
|
|
|
oldPassword = checkedPasswordReset.password[0].oldPassword;
|
|
|
|
newPassword = checkedPasswordReset.password[0].newPassword;
|
|
|
|
ne2Password = checkedPasswordReset.password[0].ne2Password;
|
|
|
|
|
|
|
|
return dataProvider.User.changePassword(oldPassword, newPassword, ne2Password, options).then(function () {
|
|
|
|
return when.resolve({password: [{message: 'Password changed successfully.'}]});
|
2014-07-24 04:46:05 -05:00
|
|
|
}).catch(function (error) {
|
2014-07-12 23:01:26 -05:00
|
|
|
return when.reject(new errors.ValidationError(error.message));
|
2014-06-20 04:15:01 -05:00
|
|
|
});
|
|
|
|
});
|
2014-07-30 10:40:30 -05:00
|
|
|
},
|
2013-12-06 03:51:35 -05:00
|
|
|
|
2014-07-30 10:40:30 -05:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
transferOwnership: function transferOwnership(object, options) {
|
|
|
|
return dataProvider.Role.findOne({name: 'Owner'}).then(function (ownerRole) {
|
|
|
|
return canThis(options.context).assign.role(ownerRole);
|
|
|
|
}).then(function () {
|
|
|
|
return utils.checkObject(object, 'owner').then(function (checkedOwnerTransfer) {
|
|
|
|
return dataProvider.User.transferOwnership(checkedOwnerTransfer.owner[0], options).then(function () {
|
|
|
|
return when.resolve({owner: [{message: 'Ownership transferred successfully.'}]});
|
|
|
|
}).catch(function (error) {
|
|
|
|
return when.reject(new errors.ValidationError(error.message));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}).catch(function (error) {
|
|
|
|
return errors.handleAPIError(error);
|
|
|
|
});
|
|
|
|
}
|
2013-12-06 03:51:35 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = users;
|