optimize privileges and assorted fixes.

* new methods
privileges.categories.filter
privileges.topics.filter
privileges.posts.filter

they take a list of ids and a privilege, and return the filtered list of
ids, faster than doing  async.filter and calling the db for each id.

* remove event listeners on recent page before adding
* group.exists works for both single group names and arrays
* helpers.allowedTo works for both a single cid and an array of cids
* moved filter:topic.post hook right before topic creation.
* moved filter:topic.reply hook right before topic reply.
v1.18.x
barisusakli 11 years ago
parent eeb4c9f487
commit fed8cc6d53

@ -56,7 +56,7 @@ define('forum/recent', ['forum/infinitescroll'], function(infinitescroll) {
Recent.watchForNewPosts = function () { Recent.watchForNewPosts = function () {
newPostCount = 0; newPostCount = 0;
newTopicCount = 0; newTopicCount = 0;
Recent.removeListeners();
socket.on('event:new_topic', onNewTopic); socket.on('event:new_topic', onNewTopic);
socket.on('event:new_post', onNewPost); socket.on('event:new_post', onNewPost);
}; };

@ -198,19 +198,21 @@ var db = require('./database'),
return callback(null, []); return callback(null, []);
} }
Categories.getCategories(cids, uid, function(err, categories) { privileges.categories.filter('find', cids, uid, function(err, cids) {
if (err) { if (err) {
return callback(err); return callback(err);
} }
async.filter(categories, function (category, next) {
if (category.disabled) { Categories.getCategories(cids, uid, function(err, categories) {
return next(false); if (err) {
return callback(err);
} }
privileges.categories.can('find', category.cid, uid, function(err, findable) {
next(!err && findable); categories = categories.filter(function(category) {
return !category.disabled;
}); });
}, function(visibleCategories) {
callback(null, visibleCategories); callback(null, categories);
}); });
}); });
}); });
@ -218,18 +220,15 @@ var db = require('./database'),
Categories.getModerators = function(cid, callback) { Categories.getModerators = function(cid, callback) {
Groups.get('cid:' + cid + ':privileges:mods', {}, function(err, groupObj) { Groups.get('cid:' + cid + ':privileges:mods', {}, function(err, groupObj) {
if (!err) { if (err) {
if (groupObj.members && groupObj.members.length) { return callback(err);
user.getMultipleUserFields(groupObj.members, ['uid', 'username', 'userslug', 'picture'], function(err, moderators) {
callback(err, moderators);
});
} else {
callback(null, []);
}
} else {
// Probably no mods
callback(null, []);
} }
if (!Array.isArray(groupObj) || !groupObj.members.length) {
return callback(null, []);
}
user.getMultipleUserFields(groupObj.members, ['uid', 'username', 'userslug', 'picture'], callback);
}); });
}; };
@ -309,11 +308,14 @@ var db = require('./database'),
}; };
Categories.getCategories = function(cids, uid, callback) { Categories.getCategories = function(cids, uid, callback) {
if (!Array.isArray(cids)) {
if (!Array.isArray(cids) || cids.length === 0) {
return callback(new Error('[[error:invalid-cid]]')); return callback(new Error('[[error:invalid-cid]]'));
} }
if (!cids.length) {
return callback(null, []);
}
async.parallel({ async.parallel({
categories: function(next) { categories: function(next) {
Categories.getCategoriesData(cids, next); Categories.getCategoriesData(cids, next);

@ -183,7 +183,11 @@
}; };
Groups.exists = function(name, callback) { Groups.exists = function(name, callback) {
db.isSetMember('groups', name, callback); if (Array.isArray(name)) {
db.isSetMembers('groups', name, callback);
} else {
db.isSetMember('groups', name, callback);
}
}; };
Groups.create = function(name, description, callback) { Groups.create = function(name, description, callback) {

@ -178,11 +178,14 @@ var async = require('async'),
return callback(err); return callback(err);
} }
async.filter(pids, function(pid, next) { if (!Array.isArray(pids) || !pids.length) {
privileges.posts.can('read', pid, uid, function(err, canRead) { return callback(null, []);
next(!err && canRead); }
});
}, function(pids) { privileges.posts.filter('read', pids, uid, function(err, pids) {
if (err) {
return callback(err);
}
Posts.getPostSummaryByPids(pids, {stripTags: true}, callback); Posts.getPostSummaryByPids(pids, {stripTags: true}, callback);
}); });
}); });
@ -487,11 +490,10 @@ var async = require('async'),
return callback(err); return callback(err);
} }
async.filter(pids, function(pid, next) { privileges.posts.filter('read', pids, callerUid, function(err, pids) {
privileges.posts.can('read', pid, callerUid, function(err, canRead) { if (err) {
next(!err && canRead); return callback(err);
}); }
}, function(pids) {
getPostsFromSet('uid:' + uid + ':posts', pids, callback); getPostsFromSet('uid:' + uid + ':posts', pids, callback);
}); });
}); });

@ -4,6 +4,7 @@
var async = require('async'), var async = require('async'),
user = require('../user'), user = require('../user'),
categories = require('../categories'),
groups = require('../groups'), groups = require('../groups'),
helpers = require('./helpers'); helpers = require('./helpers');
@ -43,17 +44,54 @@ module.exports = function(privileges) {
}; };
privileges.categories.can = function(privilege, cid, uid, callback) { privileges.categories.can = function(privilege, cid, uid, callback) {
helpers.some([ categories.getCategoryField(cid, 'disabled', function(err, disabled) {
function(next) { if (err) {
helpers.allowedTo(privilege, uid, cid, next); return callback(err);
}
if (parseInt(disabled, 10) === 1) {
return callback(null, false);
}
helpers.some([
function(next) {
helpers.allowedTo(privilege, uid, cid, next);
},
function(next) {
user.isModerator(uid, cid, next);
},
function(next) {
user.isAdministrator(uid, next);
}
], callback);
});
};
privileges.categories.filter = function(privilege, cids, uid, callback) {
async.parallel({
allowedTo: function(next) {
helpers.allowedTo(privilege, uid, cids, next);
}, },
function(next) { isModerators: function(next) {
user.isModerator(uid, cid, next); user.isModerator(uid, cids, next);
}, },
function(next) { isAdmin: function(next) {
user.isAdministrator(uid, next); user.isAdministrator(uid, next);
} }
], callback); }, function(err, results) {
if (err) {
return callback(err);
}
if (results.isAdmin) {
return callback(null, cids);
}
cids = cids.filter(function(cid, index) {
return results.allowedTo[index] || results.isModerators[index];
});
callback(null, cids);
});
}; };
privileges.categories.canMoveAllTopics = function(currentCid, targetCid, uid, callback) { privileges.categories.canMoveAllTopics = function(currentCid, targetCid, uid, callback) {
@ -78,15 +116,15 @@ module.exports = function(privileges) {
privileges.categories.userPrivileges = function(cid, uid, callback) { privileges.categories.userPrivileges = function(cid, uid, callback) {
async.parallel({ async.parallel({
find: async.apply(helpers.isMember, groups.isMember, 'cid:' + cid + ':privileges:find', uid), find: async.apply(groups.isMember, uid, 'cid:' + cid + ':privileges:find'),
read: function(next) { read: function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:read', uid, next); groups.isMember(uid, 'cid:' + cid + ':privileges:read', next);
}, },
'topics:create': function(next) { 'topics:create': function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:topics:create', uid, next); groups.isMember(uid, 'cid:' + cid + ':privileges:topics:create', next);
}, },
'topics:reply': function(next) { 'topics:reply': function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:topics:reply', uid, next); groups.isMember(uid, 'cid:' + cid + ':privileges:topics:reply', next);
}, },
mods: function(next) { mods: function(next) {
user.isModerator(uid, cid, next); user.isModerator(uid, cid, next);
@ -96,21 +134,15 @@ module.exports = function(privileges) {
privileges.categories.groupPrivileges = function(cid, groupName, callback) { privileges.categories.groupPrivileges = function(cid, groupName, callback) {
async.parallel({ async.parallel({
'groups:find': async.apply(helpers.isMember, groups.isMember, 'cid:' + cid + ':privileges:groups:find', groupName), 'groups:find': async.apply(groups.isMember, groupName, 'cid:' + cid + ':privileges:groups:find'),
'groups:read': function(next) { 'groups:read': function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:groups:read', groupName, function(err, isMember){ groups.isMember(groupName, 'cid:' + cid + ':privileges:groups:read', next);
next(err, !!isMember);
});
}, },
'groups:topics:create': function(next) { 'groups:topics:create': function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:groups:topics:create', groupName, function(err, isMember){ groups.isMember(groupName, 'cid:' + cid + ':privileges:groups:topics:create', next);
next(err, !!isMember);
});
}, },
'groups:topics:reply': function(next) { 'groups:topics:reply': function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:groups:topics:reply', groupName, function(err, isMember){ groups.isMember(groupName, 'cid:' + cid + ':privileges:groups:topics:reply', next);
next(err, !!isMember);
});
} }
}, callback); }, callback);
}; };

@ -2,7 +2,7 @@
'use strict'; 'use strict';
var async = require('async'), var async = require('async'),
db = require('../database'),
meta = require('../meta'), meta = require('../meta'),
user = require('../user'), user = require('../user'),
groups = require('../groups'), groups = require('../groups'),
@ -20,67 +20,87 @@ helpers.some = function(tasks, callback) {
}); });
}; };
helpers.allowedTo = function(privilege, uid, cid, callback) { helpers.allowedTo = function(privilege, uid, cids, callback) {
categories.getCategoryField(cid, 'disabled', function(err, disabled) {
if (!Array.isArray(cids)) {
cids = [cids];
}
if (parseInt(uid, 10) === 0) {
return isGuestAllowedTo(privilege, cids, callback);
}
var userKeys = [], groupKeys = [];
for (var i=0; i<cids.length; ++i) {
userKeys.push('cid:' + cids[i] + ':privileges:' + privilege);
groupKeys.push('cid:' + cids[i] + ':privileges:groups:' + privilege);
}
async.parallel({
userPrivilegeExists: function(next) {
groups.exists(userKeys, next);
},
groupPrivilegeExists: function(next) {
groups.exists(groupKeys, next);
},
hasUserPrivilege: function(next) {
groups.isMemberOfGroups(uid, userKeys, next);
},
hasGroupPrivilege: function(next) {
async.map(groupKeys, function(groupKey, next) {
groups.isMemberOfGroupList(uid, groupKey, next);
}, next);
}
}, function(err, results) {
if (err) { if (err) {
return callback(err); return callback(err);
} }
if (parseInt(disabled, 10) === 1) { var result = [];
return callback(null, false); for (var i=0; i<cids.length; ++i) {
result.push((!results.userPrivilegeExists[i] && !results.groupPrivilegeExists[i]) || results.hasUserPrivilege[i] || results.hasGroupPrivilege[i]);
} }
if (parseInt(uid, 10) === 0) { if (result.length === 1) {
return isGuestAllowedTo(privilege, cid, callback); result = result[0];
} }
async.parallel({ callback(null, result);
hasUserPrivilege: function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:' + privilege, uid, next);
},
hasGroupPrivilege: function(next) {
helpers.isMember(groups.isMemberOfGroupList, 'cid:' + cid + ':privileges:groups:' + privilege, uid, next);
}
}, function(err, results) {
if (err) {
return callback(err);
}
callback(null, (results.hasUserPrivilege === null && results.hasGroupPrivilege === null) || results.hasUserPrivilege || results.hasGroupPrivilege);
});
}); });
}; };
function isGuestAllowedTo(privilege, cid, callback) { function isGuestAllowedTo(privilege, cids, callback) {
async.parallel([ var userKeys = [], groupKeys = [];
function(next) { for (var i=0; i<cids.length; ++i) {
groups.exists('cid:' + cid + ':privileges:' + privilege, function(err, exists) { userKeys.push('cid:' + cids[i] + ':privileges:' + privilege);
next(err, !err ? !exists : false); groupKeys.push('cid:' + cids[i] + ':privileges:groups:' + privilege);
}); }
async.parallel({
userPrivilegeExists: function(next) {
groups.exists(userKeys, next);
}, },
function(next) { hasGroupPrivilege: function(next) {
helpers.isMember(groups.isMember, 'cid:' + cid + ':privileges:groups:' + privilege, 'guests', function(err, isMember) { groups.isMemberOfGroups('guests', groupKeys, next);
next(err, privilege !== 'find' && privilege !== 'read' ? isMember === true : isMember !== false);
});
} }
], function(err, results) { }, function(err, results) {
callback(err, results[0] && (results[1] || results[1] === null));
});
}
helpers.isMember = function(method, group, uid, callback) {
groups.exists(group, function(err, exists) {
if (err) { if (err) {
return callback(err); return callback(err);
} }
if (!exists) { var result = [];
return callback(null, null); for (var i = 0; i<cids.length; ++i) {
var groupPriv = privilege !== 'find' && privilege !== 'read' ? results.hasGroupPrivilege[i] === true : results.hasGroupPrivilege[i] !== false;
result.push(!results.userPrivilegeExists[i] && groupPriv);
}
if (result.length === 1) {
result = result[0];
} }
method(uid, group, callback); callback(null, result);
}); });
}; }
helpers.hasEnoughReputationFor = function(privilege, uid, callback) { helpers.hasEnoughReputationFor = function(privilege, uid, callback) {
if (parseInt(meta.config['privileges:disabled'], 10)) { if (parseInt(meta.config['privileges:disabled'], 10)) {

@ -76,6 +76,31 @@ module.exports = function(privileges) {
}); });
}; };
privileges.posts.filter = function(privilege, pids, uid, callback) {
posts.getCidsByPids(pids, function(err, cids) {
if (err) {
return callback(err);
}
pids = pids.map(function(pid, index) {
return {pid: pid, cid: cids[index]};
});
privileges.categories.filter(privilege, cids, uid, function(err, cids) {
if (err) {
return callback(err);
}
pids = pids.filter(function(post) {
return cids.indexOf(post.cid) !== -1;
}).map(function(post) {
return post.pid;
});
callback(null, pids);
});
});
};
privileges.posts.canEdit = function(pid, uid, callback) { privileges.posts.canEdit = function(pid, uid, callback) {
helpers.some([ helpers.some([
function(next) { function(next) {

@ -3,6 +3,7 @@
var async = require('async'), var async = require('async'),
db = require('../database'),
topics = require('../topics'), topics = require('../topics'),
user = require('../user'), user = require('../user'),
helpers = require('./helpers'), helpers = require('./helpers'),
@ -69,6 +70,35 @@ module.exports = function(privileges) {
}); });
}; };
privileges.topics.filter = function(privilege, tids, uid, callback) {
var keys = tids.map(function(tid) {
return 'topic:' + tid;
});
db.getObjectsFields(keys, ['tid', 'cid'], function(err, topics) {
if (err) {
return callback(err);
}
var cids = topics.map(function(topic) {
return topic.cid;
});
privileges.categories.filter(privilege, cids, uid, function(err, cids) {
if (err) {
return callback(err);
}
tids = topics.filter(function(topic) {
return cids.indexOf(topic.cid) !== -1;
}).map(function(topic) {
return topic.tid;
});
callback(null, tids);
});
});
};
privileges.topics.canEdit = function(tid, uid, callback) { privileges.topics.canEdit = function(tid, uid, callback) {
helpers.some([ helpers.some([
function(next) { function(next) {

@ -45,11 +45,11 @@ search.search = function(term, uid, callback) {
} }
}); });
async.filter(mainPids, function(pid, next) { privileges.posts.filter('read', mainPids, uid, function(err, pids) {
privileges.posts.can('read', pid, uid, function(err, canRead) { if (err) {
next(!err && canRead); return callback(err);
}); }
}, function(pids) {
posts.getPostSummaryByPids(pids, {stripTags: true, parse: false}, function(err, posts) { posts.getPostSummaryByPids(pids, {stripTags: true, parse: false}, function(err, posts) {
if (err) { if (err) {
return callback(err); return callback(err);

@ -132,11 +132,11 @@ var async = require('async'),
return callback(null, returnTopics); return callback(null, returnTopics);
} }
async.filter(tids, function(tid, next) { privileges.topics.filter('read', tids, uid, function(err, tids) {
privileges.topics.can('read', tid, uid, function(err, canRead) { if (err) {
next(!err && canRead); return callback(err);
}); }
}, function(tids) {
Topics.getTopicsByTids(tids, uid, function(err, topicData) { Topics.getTopicsByTids(tids, uid, function(err, topicData) {
if(err) { if(err) {
return callback(err); return callback(err);
@ -185,6 +185,9 @@ var async = require('async'),
} }
function isTopicVisible(topicData, topicInfo) { function isTopicVisible(topicData, topicInfo) {
if (parseInt(topicInfo.categoryData.disabled, 10) === 1) {
return false;
}
var deleted = parseInt(topicData.deleted, 10) !== 0; var deleted = parseInt(topicData.deleted, 10) !== 0;
return !deleted || (deleted && topicInfo.privileges.view_deleted) || parseInt(topicData.uid, 10) === parseInt(uid, 10); return !deleted || (deleted && topicInfo.privileges.view_deleted) || parseInt(topicData.uid, 10) === parseInt(uid, 10);
} }
@ -206,7 +209,7 @@ var async = require('async'),
if (categoryCache[topicData.cid]) { if (categoryCache[topicData.cid]) {
return next(null, categoryCache[topicData.cid]); return next(null, categoryCache[topicData.cid]);
} }
categories.getCategoryFields(topicData.cid, ['name', 'slug', 'icon', 'bgColor', 'color'], next); categories.getCategoryFields(topicData.cid, ['name', 'slug', 'icon', 'bgColor', 'color', 'disabled'], next);
}, },
user: function(next) { user: function(next) {
if (userCache[topicData.uid]) { if (userCache[topicData.uid]) {

@ -92,16 +92,6 @@ module.exports = function(Topics) {
} }
async.waterfall([ async.waterfall([
function(next) {
plugins.fireHook('filter:topic.post', data, function(err, filteredData) {
if (err) {
return next(err);
}
content = filteredData.content || data.content;
next();
});
},
function(next) { function(next) {
categories.exists(cid, next); categories.exists(cid, next);
}, },
@ -120,6 +110,13 @@ module.exports = function(Topics) {
function(next) { function(next) {
user.isReadyToPost(uid, next); user.isReadyToPost(uid, next);
}, },
function(next) {
plugins.fireHook('filter:topic.post', data, next);
},
function(filteredData, next) {
content = filteredData.content || data.content;
next();
},
function(next) { function(next) {
Topics.create({uid: uid, title: title, cid: cid, thumb: data.thumb, tags: data.tags}, next); Topics.create({uid: uid, title: title, cid: cid, thumb: data.thumb, tags: data.tags}, next);
}, },
@ -161,10 +158,6 @@ module.exports = function(Topics) {
async.waterfall([ async.waterfall([
function(next) { function(next) {
plugins.fireHook('filter:topic.reply', data, next);
},
function(filteredData, next) {
content = filteredData.content || data.content;
threadTools.exists(tid, next); threadTools.exists(tid, next);
}, },
function(topicExists, next) { function(topicExists, next) {
@ -191,6 +184,10 @@ module.exports = function(Topics) {
user.isReadyToPost(uid, next); user.isReadyToPost(uid, next);
}, },
function(next) { function(next) {
plugins.fireHook('filter:topic.reply', data, next);
},
function(filteredData, next) {
content = filteredData.content || data.content;
if (content) { if (content) {
content = content.trim(); content = content.trim();
} }

@ -44,15 +44,16 @@ module.exports = function(Topics) {
if(err) { if(err) {
return callback(err); return callback(err);
} }
var newtids = tids.filter(function(tid, index, self) { var newtids = tids.filter(function(tid, index, self) {
return !read[index]; return !read[index];
}); });
async.filter(newtids, function(tid, next) { privileges.topics.filter('read', newtids, uid, function(err, newTids) {
privileges.topics.can('read', tid, uid, function(err, canRead) { if(err) {
next(!err && canRead); return callback(err);
}); }
}, function(newtids) {
unreadTids.push.apply(unreadTids, newtids); unreadTids.push.apply(unreadTids, newtids);
start = stop + 1; start = stop + 1;

@ -17,8 +17,6 @@ var async = require('async'),
(function(UserNotifications) { (function(UserNotifications) {
UserNotifications.get = function(uid, callback) { UserNotifications.get = function(uid, callback) {
function getNotifications(set, start, stop, iterator, done) { function getNotifications(set, start, stop, iterator, done) {
db.getSortedSetRevRange(set, start, stop, function(err, uniqueIds) { db.getSortedSetRevRange(set, start, stop, function(err, uniqueIds) {
if(err) { if(err) {

Loading…
Cancel
Save