'use strict'; (function(Groups) { var async = require('async'), winston = require('winston'), user = require('./user'), meta = require('./meta'), db = require('./database'), posts = require('./posts'), utils = require('../public/src/utils'), ephemeralGroups = ['guests'], internals = { filterGroups: function(groups, options) { // Remove system, hidden, or deleted groups from this list if (groups && !options.showAllGroups) { return groups.filter(function (group) { if (group.deleted || (group.hidden && !group.system) || (!options.showSystemGroups && group.system)) { return false; } else if (options.removeEphemeralGroups && ephemeralGroups.indexOf(group.name) !== -1) { return false; } else { return true; } }); } else { return groups; } }, getEphemeralGroup: function(groupName, options, callback) { Groups.exists(groupName, function(err, exists) { if (!err && exists) { Groups.get.apply(null, arguments); } else { callback(null, { name: groupName, description: '', deleted: '0', hidden: '0', system: '1' }); } }); }, removeEphemeralGroups: function(groups) { var x = groups.length; while(x--) { if (ephemeralGroups.indexOf(groups[x]) !== -1) { groups.splice(x, 1); } } return groups; } }; Groups.list = function(options, callback) { db.getSetMembers('groups', function (err, groupNames) { groupNames = groupNames.concat(ephemeralGroups); async.map(groupNames, function (groupName, next) { Groups.get(groupName, options, next); }, function (err, groups) { callback(err, internals.filterGroups(groups, options)); }); }); }; Groups.get = function(groupName, options, callback) { var truncated = false, numUsers; async.parallel({ base: function (next) { if (ephemeralGroups.indexOf(groupName) === -1) { db.getObject('group:' + groupName, function(err, groupObj) { if (err) { next(err); } else if (!groupObj) { next('group-not-found'); } else { next(err, groupObj); } }); } else { internals.getEphemeralGroup(groupName, options, next); } }, users: function (next) { db.getSetMembers('group:' + groupName + ':members', function (err, uids) { if (err) { return next(err); } if (options.truncateUserList) { if (uids.length > 4) { numUsers = uids.length; uids.length = 4; truncated = true; } } if (options.expand) { async.map(uids, user.getUserData, next); } else { next(err, uids); } }); } }, function (err, results) { if (err) { return callback(err); } // User counts results.base.count = numUsers || results.users.length; results.base.members = results.users; results.base.memberCount = numUsers || results.users.length; results.base.deleted = !!parseInt(results.base.deleted, 10); results.base.hidden = !!parseInt(results.base.hidden, 10); results.base.system = !!parseInt(results.base.system, 10); results.base.deletable = !results.base.system; results.base.truncated = truncated; callback(err, results.base); }); }; Groups.search = function(query, options, callback) { if (query.length) { db.getSetMembers('groups', function(err, groups) { groups = groups.filter(function(groupName) { return groupName.match(new RegExp(utils.escapeRegexChars(query), 'i')); }); async.map(groups, function(groupName, next) { Groups.get(groupName, options, next); }, function(err, groups) { callback(err, internals.filterGroups(groups, options)); }); }); } else { callback(null, []); } }; Groups.isMember = function(uid, groupName, callback) { db.isSetMember('group:' + groupName + ':members', uid, callback); }; Groups.isMemberOfGroups = function(uid, groups, callback) { db.isMemberOfSets(groups, uid, callback); }; Groups.getMemberCount = function(groupName, callback) { db.setCount('group:' + groupName + ':members', callback); }; Groups.isMemberOfGroupList = function(uid, groupListKey, callback) { db.getSetMembers('group:' + groupListKey + ':members', function(err, groupNames) { groupNames = internals.removeEphemeralGroups(groupNames); if (groupNames.length === 0) { return callback(null, null); } async.some(groupNames, function(groupName, next) { Groups.isMember(uid, groupName, function(err, isMember) { if (!err && isMember) { next(true); } else { next(false); } }); }, function(result) { callback(null, result); }); }); }; Groups.exists = function(name, callback) { db.isSetMember('groups', name, callback); }; Groups.create = function(name, description, callback) { if (name.length === 0) { return callback(new Error('[[error:group-name-too-short]]')); } if (name === 'administrators' || name === 'registered-users') { var system = true; } meta.userOrGroupExists(name, function (err, exists) { if (err) { return callback(err); } if (exists) { return callback(new Error('[[error:group-already-exists]]')); } var groupData = { name: name, description: description, deleted: '0', hidden: '0', system: system ? '1' : '0' }; async.parallel([ function(next) { db.setAdd('groups', name, next); }, function(next) { db.setObject('group:' + name, groupData, function(err) { Groups.get(name, {}, next); }); } ], callback); }); }; Groups.hide = function(groupName, callback) { callback = callback || function() {}; db.setObjectField('group:' + groupName, 'hidden', 1, callback); }; Groups.update = function(groupName, values, callback) { callback = callback || function() {}; db.exists('group:' + groupName, function (err, exists) { if (err || !exists) { return callback(err || new Error('[[error:no-group]]')); } db.setObject('group:' + groupName, { userTitle: values.userTitle || '', description: values.description || '', icon: values.icon || '', labelColor: values.labelColor || '#000000', hidden: values.hidden || '0' }, callback); }); }; Groups.destroy = function(groupName, callback) { async.parallel([ function(next) { db.delete('group:' + groupName, next); }, function(next) { db.setRemove('groups', groupName, next); }, function(next) { db.delete('group:' + groupName + ':members', next); } ], callback); }; Groups.join = function(groupName, uid, callback) { Groups.exists(groupName, function(err, exists) { if (exists) { db.setAdd('group:' + groupName + ':members', uid, callback); } else { Groups.create(groupName, '', function(err) { if (err) { winston.error('[groups.join] Could not create new hidden group: ' + err.message); return callback(err); } Groups.hide(groupName); db.setAdd('group:' + groupName + ':members', uid, callback); }); } }); }; Groups.leave = function(groupName, uid, callback) { db.setRemove('group:' + groupName + ':members', uid, function(err) { if (err) { return callback(err); } // If this is a hidden group, and it is now empty, delete it Groups.get(groupName, {}, function(err, group) { if (err) { return callback(err); } if (group.hidden && group.memberCount === 0) { Groups.destroy(groupName, callback); } else { return callback(); } }); }); }; Groups.leaveAllGroups = function(uid, callback) { db.getSetMembers('groups', function(err, groups) { async.each(groups, function(groupName, next) { Groups.isMember(uid, groupName, function(err, isMember) { if (!err && isMember) { Groups.leave(groupName, uid, next); } else { next(); } }); }, callback); }); }; Groups.getLatestMemberPosts = function(groupName, max, callback) { Groups.get(groupName, {}, function(err, groupObj) { if (err || parseInt(groupObj.memberCount, 10) === 0) { return callback(null, []); } var keys = groupObj.members.map(function(uid) { return 'uid:' + uid + ':posts'; }); db.getSortedSetRevUnion(keys, 0, max-1, function(err, pids) { if (err) { return callback(err); } posts.getPostSummaryByPids(pids, {stripTags: false}, callback); }); }); }; Groups.getUserGroups = function(uid, callback) { var ignoredGroups = ['registered-users']; db.getSetMembers('groups', function(err, groupNames) { var groupKeys = groupNames.filter(function(groupName) { return ignoredGroups.indexOf(groupName) === -1; }).map(function(groupName) { return 'group:' + groupName; }); db.getObjectsFields(groupKeys, ['name', 'hidden', 'userTitle', 'icon', 'labelColor'], function(err, groupData) { groupData = groupData.filter(function(group) { return parseInt(group.hidden, 10) !== 1; }); var groupSets = groupData.map(function(group) { group.userTitle = group.userTitle || group.name; group.labelColor = group.labelColor || '#000000'; return 'group:' + group.name + ':members'; }); db.isMemberOfSets(groupSets, uid, function(err, isMembers) { for(var i=isMembers.length - 1; i>=0; --i) { if (!isMembers[i]) { groupData.splice(i, 1); } } callback(null, groupData); }); }); }); }; }(module.exports));