2013-06-25 06:43:15 -05:00
|
|
|
// canThis(someUser).edit.posts([id]|[[ids]])
|
|
|
|
// canThis(someUser).edit.post(somePost|somePostId)
|
|
|
|
|
|
|
|
var _ = require('underscore'),
|
|
|
|
when = require('when'),
|
|
|
|
Models = require('../models'),
|
|
|
|
objectTypeModelMap = require('./objectTypeModelMap'),
|
|
|
|
UserProvider = Models.User,
|
|
|
|
PermissionsProvider = Models.Permission,
|
|
|
|
init,
|
|
|
|
refresh,
|
|
|
|
canThis,
|
|
|
|
CanThisResult,
|
|
|
|
exported;
|
|
|
|
|
|
|
|
// Base class for canThis call results
|
|
|
|
CanThisResult = function () {
|
|
|
|
this.userPermissionLoad = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
CanThisResult.prototype.buildObjectTypeHandlers = function (obj_types, act_type, userId) {
|
|
|
|
var self = this,
|
|
|
|
obj_type_handlers = {};
|
|
|
|
|
|
|
|
// Iterate through the object types, i.e. ['post', 'tag', 'user']
|
|
|
|
_.each(obj_types, function (obj_type) {
|
|
|
|
var TargetModel = objectTypeModelMap[obj_type];
|
|
|
|
|
|
|
|
// Create the 'handler' for the object type;
|
|
|
|
// the '.post()' in canThis(user).edit.post()
|
|
|
|
obj_type_handlers[obj_type] = function (modelOrId) {
|
|
|
|
var modelId;
|
|
|
|
|
|
|
|
if (_.isNumber(modelOrId) || _.isString(modelOrId)) {
|
|
|
|
// It's an id already, do nothing
|
|
|
|
modelId = modelOrId;
|
|
|
|
} else if (modelOrId) {
|
|
|
|
// It's a model, get the id
|
|
|
|
modelId = modelOrId.id;
|
|
|
|
}
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Wait for the user loading to finish
|
|
|
|
return self.userPermissionLoad.then(function (userPermissions) {
|
|
|
|
// Iterate through the user permissions looking for an affirmation
|
|
|
|
var hasPermission;
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Allow for a target model to implement a "Permissable" interface
|
|
|
|
if (TargetModel && _.isFunction(TargetModel.permissable)) {
|
|
|
|
return TargetModel.permissable(modelId, userId, act_type, userPermissions);
|
|
|
|
}
|
2013-06-08 18:39:24 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Otherwise, check all the permissions for matching object id
|
|
|
|
hasPermission = _.any(userPermissions, function (userPermission) {
|
|
|
|
var permObjId;
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Look for a matching action type and object type first
|
|
|
|
if (userPermission.get('action_type') !== act_type || userPermission.get('object_type') !== obj_type) {
|
|
|
|
return false;
|
2013-06-04 22:47:11 -05:00
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Grab the object id (if specified, could be null)
|
|
|
|
permObjId = userPermission.get('object_id');
|
2013-06-08 18:39:24 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// If we didn't specify a model (any thing)
|
|
|
|
// or the permission didn't have an id scope set
|
|
|
|
// then the user has permission
|
|
|
|
if (!modelId || !permObjId) {
|
|
|
|
return true;
|
2013-06-08 18:39:24 -05:00
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Otherwise, check if the id's match
|
|
|
|
// TODO: String vs Int comparison possibility here?
|
|
|
|
return modelId === permObjId;
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
if (hasPermission) {
|
|
|
|
return when.resolve();
|
|
|
|
}
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
return when.reject();
|
|
|
|
}).otherwise(function () {
|
|
|
|
// No permissions loaded, or error loading permissions
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Still check for permissable without permissions
|
|
|
|
if (TargetModel && _.isFunction(TargetModel.permissable)) {
|
|
|
|
return TargetModel.permissable(modelId, userId, act_type, []);
|
2013-06-04 22:47:11 -05:00
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
return when.reject();
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
return obj_type_handlers;
|
|
|
|
};
|
|
|
|
|
|
|
|
CanThisResult.prototype.beginCheck = function (user) {
|
|
|
|
var self = this,
|
|
|
|
userId = user.id || user;
|
|
|
|
|
|
|
|
// TODO: Switch logic based on object type; user, role, post.
|
|
|
|
|
|
|
|
// Kick off the fetching of the user data
|
|
|
|
this.userPermissionLoad = UserProvider.effectivePermissions(userId);
|
|
|
|
|
|
|
|
// Iterate through the actions and their related object types
|
|
|
|
// We should have loaded these through a permissions.init() call previously
|
|
|
|
// TODO: Throw error if not init() yet?
|
|
|
|
_.each(exported.actionsMap, function (obj_types, act_type) {
|
|
|
|
// Build up the object type handlers;
|
|
|
|
// the '.post()' parts in canThis(user).edit.post()
|
|
|
|
var obj_type_handlers = self.buildObjectTypeHandlers(obj_types, act_type, userId);
|
|
|
|
|
|
|
|
// Define a property for the action on the result;
|
|
|
|
// the '.edit' in canThis(user).edit.post()
|
|
|
|
Object.defineProperty(self, act_type, {
|
|
|
|
writable: false,
|
|
|
|
enumerable: false,
|
|
|
|
configurable: false,
|
|
|
|
value: obj_type_handlers
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// Return this for chaining
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
canThis = function (user) {
|
|
|
|
var result = new CanThisResult();
|
|
|
|
|
|
|
|
return result.beginCheck(user);
|
|
|
|
};
|
|
|
|
|
|
|
|
init = refresh = function () {
|
|
|
|
// Load all the permissions
|
|
|
|
return PermissionsProvider.browse().then(function (perms) {
|
|
|
|
var seenActions = {};
|
|
|
|
|
|
|
|
exported.actionsMap = {};
|
|
|
|
|
|
|
|
// Build a hash map of the actions on objects, i.e
|
|
|
|
/*
|
|
|
|
{
|
|
|
|
'edit': ['post', 'tag', 'user', 'page'],
|
|
|
|
'delete': ['post', 'user'],
|
|
|
|
'create': ['post', 'user', 'page']
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
_.each(perms.models, function (perm) {
|
|
|
|
var action_type = perm.get('action_type'),
|
|
|
|
object_type = perm.get('object_type');
|
|
|
|
|
|
|
|
exported.actionsMap[action_type] = exported.actionsMap[action_type] || [];
|
|
|
|
seenActions[action_type] = seenActions[action_type] || {};
|
|
|
|
|
|
|
|
// Check if we've already seen this action -> object combo
|
|
|
|
if (seenActions[action_type][object_type]) {
|
|
|
|
return;
|
|
|
|
}
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
exported.actionsMap[action_type].push(object_type);
|
|
|
|
seenActions[action_type][object_type] = true;
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
return when(exported.actionsMap);
|
|
|
|
});
|
|
|
|
};
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
module.exports = exported = {
|
|
|
|
init: init,
|
|
|
|
refresh: refresh,
|
|
|
|
canThis: canThis,
|
|
|
|
actionsMap: {}
|
|
|
|
};
|