2013-06-25 06:43:15 -05:00
|
|
|
var User,
|
|
|
|
Users,
|
|
|
|
UserRole,
|
|
|
|
// UserRoles,
|
|
|
|
_ = require('underscore'),
|
|
|
|
uuid = require('node-uuid'),
|
|
|
|
when = require('when'),
|
|
|
|
errors = require('../errorHandling'),
|
|
|
|
nodefn = require('when/node/function'),
|
|
|
|
bcrypt = require('bcrypt-nodejs'),
|
|
|
|
Posts = require('./post').Posts,
|
|
|
|
GhostBookshelf = require('./base'),
|
|
|
|
Role = require('./role').Role,
|
|
|
|
Permission = require('./permission').Permission;
|
|
|
|
|
|
|
|
UserRole = GhostBookshelf.Model.extend({
|
|
|
|
tableName: 'roles_users'
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2013-08-20 13:52:44 -05:00
|
|
|
function validatePasswordLength(password) {
|
|
|
|
try {
|
|
|
|
GhostBookshelf.validator.check(password, "Your password is not long enough. It must be at least 8 chars long.").len(8);
|
|
|
|
} catch (error) {
|
|
|
|
return when.reject(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
return when.resolve();
|
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
User = GhostBookshelf.Model.extend({
|
|
|
|
|
|
|
|
tableName: 'users',
|
|
|
|
|
|
|
|
hasTimestamps: true,
|
|
|
|
|
2013-08-25 05:49:31 -05:00
|
|
|
permittedAttributes: [
|
|
|
|
'id', 'uuid', 'full_name', 'password', 'email_address', 'profile_picture', 'cover_picture', 'bio', 'url', 'location',
|
|
|
|
'created_at', 'created_by', 'updated_at', 'updated_by'
|
|
|
|
],
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
defaults: function () {
|
|
|
|
return {
|
|
|
|
uuid: uuid.v4()
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2013-09-02 14:16:13 -05:00
|
|
|
parse: function (attrs) {
|
|
|
|
// temporary alias of name for full_name (will get changed in the schema)
|
|
|
|
if (attrs.full_name && !attrs.name) {
|
|
|
|
attrs.name = attrs.full_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
// temporary alias of website for url (will get changed in the schema)
|
|
|
|
if (attrs.url && !attrs.website) {
|
|
|
|
attrs.website = attrs.url;
|
|
|
|
}
|
|
|
|
|
2013-09-09 09:25:20 -05:00
|
|
|
// temporary alias of email for email_address (will get changed in the schema)
|
|
|
|
if (attrs.email_address && !attrs.email) {
|
|
|
|
attrs.email = attrs.email_address;
|
|
|
|
}
|
|
|
|
|
|
|
|
// temporary alias of image for profile_picture (will get changed in the schema)
|
|
|
|
if (attrs.profile_picture && !attrs.image) {
|
|
|
|
attrs.image = attrs.profile_picture;
|
|
|
|
}
|
|
|
|
|
|
|
|
// temporary alias of cover for cover_picture (will get changed in the schema)
|
|
|
|
if (attrs.cover_picture && !attrs.cover) {
|
|
|
|
attrs.cover = attrs.cover_picture;
|
|
|
|
}
|
|
|
|
|
2013-09-02 14:16:13 -05:00
|
|
|
return attrs;
|
|
|
|
},
|
|
|
|
|
2013-08-20 13:52:44 -05:00
|
|
|
initialize: function () {
|
2013-08-25 05:49:31 -05:00
|
|
|
this.on('saving', this.saving, this);
|
2013-08-20 13:52:44 -05:00
|
|
|
this.on('saving', this.validate, this);
|
|
|
|
},
|
|
|
|
|
|
|
|
validate: function () {
|
|
|
|
GhostBookshelf.validator.check(this.get('email_address'), "Please check your email address. It does not seem to be valid.").isEmail();
|
|
|
|
GhostBookshelf.validator.check(this.get('bio'), "Your bio is too long. Please keep it to 200 chars.").len(0, 200);
|
|
|
|
if (this.get('url') && this.get('url').length > 0) {
|
|
|
|
GhostBookshelf.validator.check(this.get('url'), "Your website is not a valid URL.").isUrl();
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2013-08-25 05:49:31 -05:00
|
|
|
saving: function () {
|
|
|
|
// Deal with the related data here
|
|
|
|
|
|
|
|
// Remove any properties which don't belong on the post model
|
|
|
|
this.attributes = this.pick(this.permittedAttributes);
|
|
|
|
},
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
posts: function () {
|
|
|
|
return this.hasMany(Posts, 'created_by');
|
|
|
|
},
|
|
|
|
|
|
|
|
roles: function () {
|
|
|
|
return this.belongsToMany(Role);
|
|
|
|
},
|
|
|
|
|
|
|
|
permissions: function () {
|
|
|
|
return this.belongsToMany(Permission);
|
|
|
|
}
|
|
|
|
|
|
|
|
}, {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Naive user add
|
|
|
|
* @param _user
|
|
|
|
*
|
|
|
|
* Hashes the password provided before saving to the database.
|
|
|
|
*/
|
|
|
|
add: function (_user) {
|
|
|
|
|
2013-08-15 18:22:08 -05:00
|
|
|
var self = this,
|
|
|
|
// Clone the _user so we don't expose the hashed password unnecessarily
|
|
|
|
userData = _.extend({}, _user);
|
2013-06-01 09:47:41 -05:00
|
|
|
|
|
|
|
/**
|
2013-06-25 06:43:15 -05:00
|
|
|
* This only allows one user to be added to the database, otherwise fails.
|
|
|
|
* @param {object} user
|
|
|
|
* @author javorszky
|
2013-06-01 09:47:41 -05:00
|
|
|
*/
|
2013-08-20 13:52:44 -05:00
|
|
|
return validatePasswordLength(userData.password).then(function () {
|
|
|
|
return self.forge().fetch();
|
|
|
|
}).then(function (user) {
|
2013-08-15 18:22:08 -05:00
|
|
|
// Check if user exists
|
2013-06-25 06:43:15 -05:00
|
|
|
if (user) {
|
|
|
|
return when.reject(new Error('A user is already registered. Only one user for now!'));
|
|
|
|
}
|
2013-08-20 13:52:44 -05:00
|
|
|
}).then(function () {
|
2013-08-15 18:22:08 -05:00
|
|
|
// Hash the provided password with bcrypt
|
|
|
|
return nodefn.call(bcrypt.hash, _user.password, null, null);
|
|
|
|
}).then(function (hash) {
|
|
|
|
// Assign the hashed password
|
|
|
|
userData.password = hash;
|
|
|
|
// Save the user with the hashed password
|
|
|
|
return GhostBookshelf.Model.add.call(self, userData);
|
|
|
|
}).then(function (addedUser) {
|
|
|
|
// Assign the userData to our created user so we can pass it back
|
|
|
|
userData = addedUser;
|
|
|
|
// Add this user to the admin role (assumes admin = role_id: 1)
|
|
|
|
return UserRole.add({role_id: 1, user_id: addedUser.id});
|
|
|
|
}).then(function (addedUserRole) {
|
|
|
|
// Return the added user as expected
|
2013-08-31 17:20:12 -05:00
|
|
|
|
2013-08-15 18:22:08 -05:00
|
|
|
return when.resolve(userData);
|
2013-08-18 16:50:42 -05:00
|
|
|
});
|
2013-06-01 09:47:41 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
/**
|
|
|
|
* Temporarily replacing the function below with another one that checks
|
|
|
|
* whether there's anyone registered at all. This is due to #138
|
|
|
|
* @author javorszky
|
|
|
|
*/
|
2013-06-01 09:47:41 -05:00
|
|
|
|
2013-08-08 20:22:49 -05:00
|
|
|
// return this.forge({email_address: userData.email_address}).fetch().then(function (user) {
|
|
|
|
// if (user !== null) {
|
|
|
|
// return when.reject(new Error('A user with that email address already exists.'));
|
|
|
|
// }
|
|
|
|
// return nodefn.call(bcrypt.hash, _user.password, null, null).then(function (hash) {
|
|
|
|
// userData.password = hash;
|
|
|
|
// GhostBookshelf.Model.add.call(UserRole, userRoles);
|
|
|
|
// return GhostBookshelf.Model.add.call(User, userData);
|
|
|
|
// }, errors.logAndThrowError);
|
|
|
|
// }, errors.logAndThrowError);
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
},
|
|
|
|
|
2013-08-06 14:27:56 -05:00
|
|
|
// Finds the user by email, and checks the password
|
2013-06-25 06:43:15 -05:00
|
|
|
check: function (_userdata) {
|
|
|
|
return this.forge({
|
|
|
|
email_address: _userdata.email
|
|
|
|
}).fetch({require: true}).then(function (user) {
|
|
|
|
return nodefn.call(bcrypt.compare, _userdata.pw, user.get('password')).then(function (matched) {
|
|
|
|
if (!matched) {
|
2013-08-20 13:52:44 -05:00
|
|
|
return when.reject(new Error('Your password is incorrect'));
|
2013-06-25 06:43:15 -05:00
|
|
|
}
|
|
|
|
return user;
|
2013-06-16 16:36:28 -05:00
|
|
|
}, errors.logAndThrowError);
|
2013-08-08 20:22:49 -05:00
|
|
|
}, function (error) {
|
2013-08-18 16:50:42 -05:00
|
|
|
return when.reject(new Error('There is no user with that email address.'));
|
2013-08-08 20:22:49 -05:00
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
},
|
|
|
|
|
2013-08-05 18:49:06 -05:00
|
|
|
/**
|
|
|
|
* Naive change password method
|
|
|
|
* @param {object} _userdata email, old pw, new pw, new pw2
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
changePassword: function (_userdata) {
|
2013-08-20 13:52:44 -05:00
|
|
|
var self = this,
|
|
|
|
userid = _userdata.currentUser,
|
2013-08-05 18:49:06 -05:00
|
|
|
oldPassword = _userdata.oldpw,
|
|
|
|
newPassword = _userdata.newpw,
|
2013-08-31 17:20:12 -05:00
|
|
|
ne2Password = _userdata.ne2pw,
|
|
|
|
user = null;
|
|
|
|
|
2013-08-05 18:49:06 -05:00
|
|
|
|
|
|
|
if (newPassword !== ne2Password) {
|
2013-08-20 13:52:44 -05:00
|
|
|
return when.reject(new Error('Your new passwords do not match'));
|
2013-08-05 18:49:06 -05:00
|
|
|
}
|
|
|
|
|
2013-08-20 13:52:44 -05:00
|
|
|
return validatePasswordLength(newPassword).then(function () {
|
|
|
|
return self.forge({id: userid}).fetch({require: true});
|
2013-08-31 17:20:12 -05:00
|
|
|
}).then(function (_user) {
|
|
|
|
user = _user;
|
|
|
|
return nodefn.call(bcrypt.compare, oldPassword, user.get('password'));
|
|
|
|
}).then(function (matched) {
|
|
|
|
if (!matched) {
|
|
|
|
return when.reject(new Error('Your password is incorrect'));
|
|
|
|
}
|
|
|
|
return nodefn.call(bcrypt.hash, newPassword, null, null);
|
|
|
|
}).then(function (hash) {
|
|
|
|
user.save({password: hash});
|
|
|
|
|
|
|
|
return user;
|
2013-08-05 18:49:06 -05:00
|
|
|
});
|
2013-08-31 17:20:12 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
forgottenPassword: function (email) {
|
|
|
|
var newPassword = Math.random().toString(36).slice(2, 12), // This is magick
|
|
|
|
user = null;
|
2013-08-20 13:52:44 -05:00
|
|
|
|
2013-08-31 17:20:12 -05:00
|
|
|
return this.forge({email_address: email}).fetch({require: true}).then(function (_user) {
|
|
|
|
user = _user;
|
|
|
|
return nodefn.call(bcrypt.hash, newPassword, null, null);
|
|
|
|
}).then(function (hash) {
|
|
|
|
user.save({password: hash});
|
|
|
|
return { user: user, newPassword: newPassword };
|
|
|
|
}, function (error) {
|
|
|
|
return when.reject(new Error('There is no user by that email address. Check again.'));
|
|
|
|
});
|
2013-08-05 18:49:06 -05:00
|
|
|
},
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
effectivePermissions: function (id) {
|
|
|
|
return this.read({id: id}, { withRelated: ['permissions', 'roles.permissions'] })
|
|
|
|
.then(function (foundUser) {
|
|
|
|
var seenPerms = {},
|
|
|
|
rolePerms = _.map(foundUser.related('roles').models, function (role) {
|
|
|
|
return role.related('permissions').models;
|
|
|
|
}),
|
|
|
|
allPerms = [];
|
|
|
|
|
|
|
|
rolePerms.push(foundUser.related('permissions').models);
|
|
|
|
|
|
|
|
_.each(rolePerms, function (rolePermGroup) {
|
|
|
|
_.each(rolePermGroup, function (perm) {
|
|
|
|
var key = perm.get('action_type') + '-' + perm.get('object_type') + '-' + perm.get('object_id');
|
|
|
|
|
|
|
|
// Only add perms once
|
|
|
|
if (seenPerms[key]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
allPerms.push(perm);
|
|
|
|
seenPerms[key] = true;
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
});
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
return when.resolve(allPerms);
|
|
|
|
}, errors.logAndThrowError);
|
|
|
|
}
|
2013-06-01 09:47:41 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
});
|
2013-06-01 09:47:41 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
Users = GhostBookshelf.Collection.extend({
|
|
|
|
model: User
|
|
|
|
});
|
2013-06-01 09:47:41 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
module.exports = {
|
|
|
|
User: User,
|
|
|
|
Users: Users
|
2013-08-05 18:49:06 -05:00
|
|
|
};
|