2013-06-25 06:43:15 -05:00
|
|
|
// canThis(someUser).edit.posts([id]|[[ids]])
|
|
|
|
// canThis(someUser).edit.post(somePost|somePostId)
|
|
|
|
|
2014-02-05 03:40:30 -05:00
|
|
|
var _ = require('lodash'),
|
2014-08-17 01:17:23 -05:00
|
|
|
Promise = require('bluebird'),
|
2013-09-24 05:46:30 -05:00
|
|
|
Models = require('../models'),
|
2014-02-11 22:40:39 -05:00
|
|
|
effectivePerms = require('./effective'),
|
2013-06-25 06:43:15 -05:00
|
|
|
init,
|
|
|
|
refresh,
|
|
|
|
canThis,
|
|
|
|
CanThisResult,
|
|
|
|
exported;
|
|
|
|
|
2013-08-15 18:22:08 -05:00
|
|
|
function hasActionsMap() {
|
|
|
|
// Just need to find one key in the actionsMap
|
|
|
|
|
|
|
|
return _.any(exported.actionsMap, function (val, key) {
|
2013-10-31 13:02:34 -05:00
|
|
|
/*jslint unparam:true*/
|
2014-05-07 21:58:01 -05:00
|
|
|
return Object.hasOwnProperty.call(exported.actionsMap, key);
|
2013-08-15 18:22:08 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
// TODO: Move this to its own file so others can use it?
|
|
|
|
function parseContext(context) {
|
|
|
|
// Parse what's passed to canThis.beginCheck for standard user and app scopes
|
|
|
|
var parsed = {
|
2014-03-20 18:48:06 -05:00
|
|
|
internal: false,
|
2014-02-11 22:40:39 -05:00
|
|
|
user: null,
|
|
|
|
app: null
|
|
|
|
};
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
|
2014-03-20 18:48:06 -05:00
|
|
|
if (context && (context === 'internal' || context.internal)) {
|
|
|
|
parsed.internal = true;
|
|
|
|
}
|
2014-02-11 22:40:39 -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
|
|
|
if (context && context.user) {
|
2014-02-11 22:40:39 -05:00
|
|
|
parsed.user = context.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
|
|
|
}
|
|
|
|
|
|
|
|
if (context && context.app) {
|
2014-02-11 22:40:39 -05:00
|
|
|
parsed.app = context.app;
|
|
|
|
}
|
|
|
|
|
|
|
|
return parsed;
|
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Base class for canThis call results
|
|
|
|
CanThisResult = function () {
|
2014-02-11 22:40:39 -05:00
|
|
|
return;
|
2013-06-25 06:43:15 -05:00
|
|
|
};
|
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
CanThisResult.prototype.buildObjectTypeHandlers = function (objTypes, actType, context, permissionLoad) {
|
2014-09-21 15:54:32 -05:00
|
|
|
var objectTypeModelMap = {
|
|
|
|
post: Models.Post,
|
|
|
|
role: Models.Role,
|
|
|
|
user: Models.User,
|
|
|
|
permission: Models.Permission,
|
|
|
|
setting: Models.Settings
|
|
|
|
};
|
2014-05-15 21:29:42 -05:00
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
// Iterate through the object types, i.e. ['post', 'tag', 'user']
|
2014-09-09 23:06:24 -05:00
|
|
|
return _.reduce(objTypes, function (objTypeHandlers, objType) {
|
2014-02-11 22:40:39 -05:00
|
|
|
// Grab the TargetModel through the objectTypeModelMap
|
2014-09-09 23:06:24 -05:00
|
|
|
var TargetModel = objectTypeModelMap[objType];
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
|
|
// Create the 'handler' for the object type;
|
|
|
|
// the '.post()' in canThis(user).edit.post()
|
2014-09-09 23:06:24 -05:00
|
|
|
objTypeHandlers[objType] = function (modelOrId) {
|
2013-06-25 06:43:15 -05:00
|
|
|
var modelId;
|
|
|
|
|
2014-03-20 18:48:06 -05:00
|
|
|
// If it's an internal request, resolve immediately
|
|
|
|
if (context.internal) {
|
2014-08-17 01:17:23 -05:00
|
|
|
return Promise.resolve();
|
2014-03-20 18:48:06 -05:00
|
|
|
}
|
|
|
|
|
2013-06-25 06:43:15 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
// Wait for the user loading to finish
|
2014-02-11 22:40:39 -05:00
|
|
|
return permissionLoad.then(function (loadedPermissions) {
|
2013-06-25 06:43:15 -05:00
|
|
|
// Iterate through the user permissions looking for an affirmation
|
2014-07-23 13:17:29 -05:00
|
|
|
var userPermissions = loadedPermissions.user ? loadedPermissions.user.permissions : null,
|
|
|
|
appPermissions = loadedPermissions.app ? loadedPermissions.app.permissions : null,
|
2014-02-11 22:40:39 -05:00
|
|
|
hasUserPermission,
|
|
|
|
hasAppPermission,
|
|
|
|
checkPermission = function (perm) {
|
|
|
|
var permObjId;
|
|
|
|
|
|
|
|
// Look for a matching action type and object type first
|
2014-09-09 23:06:24 -05:00
|
|
|
if (perm.get('action_type') !== actType || perm.get('object_type') !== objType) {
|
2014-02-11 22:40:39 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grab the object id (if specified, could be null)
|
|
|
|
permObjId = perm.get('object_id');
|
|
|
|
|
|
|
|
// If we didn't specify a model (any thing)
|
|
|
|
// or the permission didn't have an id scope set
|
|
|
|
// then the "thing" has permission
|
|
|
|
if (!modelId || !permObjId) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, check if the id's match
|
|
|
|
// TODO: String vs Int comparison possibility here?
|
|
|
|
return modelId === permObjId;
|
|
|
|
};
|
|
|
|
// Check user permissions for matching action, object and id.
|
2014-07-23 13:17:29 -05:00
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
if (_.any(loadedPermissions.user.roles, {name: 'Owner'})) {
|
2014-07-23 13:17:29 -05:00
|
|
|
hasUserPermission = true;
|
|
|
|
} else if (!_.isEmpty(userPermissions)) {
|
|
|
|
hasUserPermission = _.any(userPermissions, checkPermission);
|
2014-02-11 22:40:39 -05:00
|
|
|
}
|
2013-06-08 18:39:24 -05:00
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
// Check app permissions if they were passed
|
|
|
|
hasAppPermission = true;
|
|
|
|
if (!_.isNull(appPermissions)) {
|
|
|
|
hasAppPermission = _.any(appPermissions, checkPermission);
|
|
|
|
}
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2014-05-13 20:49:07 -05:00
|
|
|
// Offer a chance for the TargetModel to override the results
|
2014-07-23 13:17:29 -05:00
|
|
|
if (TargetModel && _.isFunction(TargetModel.permissible)) {
|
|
|
|
return TargetModel.permissible(
|
2014-09-09 23:06:24 -05:00
|
|
|
modelId, actType, context, loadedPermissions, hasUserPermission, hasAppPermission
|
2014-07-23 13:17:29 -05:00
|
|
|
);
|
2013-06-04 22:47:11 -05:00
|
|
|
}
|
|
|
|
|
2014-05-13 20:49:07 -05:00
|
|
|
if (hasUserPermission && hasAppPermission) {
|
2014-08-17 01:17:23 -05:00
|
|
|
return;
|
2014-05-13 20:49:07 -05:00
|
|
|
}
|
2014-08-17 01:17:23 -05:00
|
|
|
|
|
|
|
return Promise.reject();
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
};
|
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
return objTypeHandlers;
|
2014-02-11 22:40:39 -05:00
|
|
|
}, {});
|
2013-06-25 06:43:15 -05:00
|
|
|
};
|
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
CanThisResult.prototype.beginCheck = function (context) {
|
2013-06-25 06:43:15 -05:00
|
|
|
var self = this,
|
2014-02-11 22:40:39 -05:00
|
|
|
userPermissionLoad,
|
|
|
|
appPermissionLoad,
|
|
|
|
permissionsLoad;
|
|
|
|
|
|
|
|
// Get context.user and context.app
|
|
|
|
context = parseContext(context);
|
2013-06-25 06:43:15 -05:00
|
|
|
|
2013-08-15 18:22:08 -05:00
|
|
|
if (!hasActionsMap()) {
|
2014-09-09 23:06:24 -05:00
|
|
|
throw new Error('No actions map found, please call permissions.init() before use.');
|
2013-08-15 18:22:08 -05:00
|
|
|
}
|
|
|
|
|
2014-03-20 18:48:06 -05:00
|
|
|
// Kick off loading of effective user permissions if necessary
|
|
|
|
if (context.user) {
|
|
|
|
userPermissionLoad = effectivePerms.user(context.user);
|
|
|
|
} else {
|
|
|
|
// Resolve null if no context.user to prevent db call
|
2014-08-17 01:17:23 -05:00
|
|
|
userPermissionLoad = Promise.resolve(null);
|
2014-03-20 18:48:06 -05:00
|
|
|
}
|
Consistency in model method naming
- The API has the BREAD naming for methods
- The model now has findAll, findOne, findPage (where needed), edit, add and destroy, meaning it is similar but with a bit more flexibility
- browse, read, update, create, and delete, which were effectively just aliases, have all been removed.
- added jsDoc for the model methods
2014-05-05 10:18:38 -05:00
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
// Kick off loading of effective app permissions if necessary
|
|
|
|
if (context.app) {
|
|
|
|
appPermissionLoad = effectivePerms.app(context.app);
|
|
|
|
} else {
|
|
|
|
// Resolve null if no context.app
|
2014-08-17 01:17:23 -05:00
|
|
|
appPermissionLoad = Promise.resolve(null);
|
2014-02-11 22:40:39 -05:00
|
|
|
}
|
2013-06-25 06:43:15 -05:00
|
|
|
|
2014-03-20 18:48:06 -05:00
|
|
|
// Wait for both user and app permissions to load
|
2014-08-17 01:17:23 -05:00
|
|
|
permissionsLoad = Promise.all([userPermissionLoad, appPermissionLoad]).then(function (result) {
|
2014-02-11 22:40:39 -05:00
|
|
|
return {
|
|
|
|
user: result[0],
|
|
|
|
app: result[1]
|
|
|
|
};
|
|
|
|
});
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
|
|
// Iterate through the actions and their related object types
|
2014-09-09 23:06:24 -05:00
|
|
|
_.each(exported.actionsMap, function (objTypes, actType) {
|
2013-06-25 06:43:15 -05:00
|
|
|
// Build up the object type handlers;
|
|
|
|
// the '.post()' parts in canThis(user).edit.post()
|
2014-09-09 23:06:24 -05:00
|
|
|
var objTypeHandlers = self.buildObjectTypeHandlers(objTypes, actType, context, permissionsLoad);
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
|
|
// Define a property for the action on the result;
|
|
|
|
// the '.edit' in canThis(user).edit.post()
|
2014-09-09 23:06:24 -05:00
|
|
|
Object.defineProperty(self, actType, {
|
2013-06-25 06:43:15 -05:00
|
|
|
writable: false,
|
|
|
|
enumerable: false,
|
|
|
|
configurable: false,
|
2014-09-09 23:06:24 -05:00
|
|
|
value: objTypeHandlers
|
2013-06-25 06:43:15 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// Return this for chaining
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
canThis = function (context) {
|
2013-06-25 06:43:15 -05:00
|
|
|
var result = new CanThisResult();
|
|
|
|
|
2014-02-11 22:40:39 -05:00
|
|
|
return result.beginCheck(context);
|
2013-06-25 06:43:15 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
init = refresh = function () {
|
|
|
|
// Load all the permissions
|
2014-05-15 21:29:42 -05:00
|
|
|
return Models.Permission.findAll().then(function (perms) {
|
2013-06-25 06:43:15 -05:00
|
|
|
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) {
|
2014-09-09 23:06:24 -05:00
|
|
|
var actionType = perm.get('action_type'),
|
|
|
|
objectType = perm.get('object_type');
|
2013-06-25 06:43:15 -05:00
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
exported.actionsMap[actionType] = exported.actionsMap[actionType] || [];
|
|
|
|
seenActions[actionType] = seenActions[actionType] || {};
|
2013-06-25 06:43:15 -05:00
|
|
|
|
|
|
|
// Check if we've already seen this action -> object combo
|
2014-09-09 23:06:24 -05:00
|
|
|
if (seenActions[actionType][objectType]) {
|
2013-06-25 06:43:15 -05:00
|
|
|
return;
|
|
|
|
}
|
2013-06-04 22:47:11 -05:00
|
|
|
|
2014-09-09 23:06:24 -05:00
|
|
|
exported.actionsMap[actionType].push(objectType);
|
|
|
|
seenActions[actionType][objectType] = true;
|
2013-06-04 22:47:11 -05:00
|
|
|
});
|
|
|
|
|
2014-08-17 01:17:23 -05:00
|
|
|
return exported.actionsMap;
|
2013-06-25 06:43:15 -05:00
|
|
|
});
|
|
|
|
};
|
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: {}
|
2013-10-31 13:02:34 -05:00
|
|
|
};
|