feat: #7743, groups/index, invite, leave,membership
parent
d5342a40ba
commit
a39ca51e06
@ -0,0 +1,105 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const db = require('../database');
|
||||||
|
const user = require('../user');
|
||||||
|
const utils = require('../utils');
|
||||||
|
const plugins = require('../plugins');
|
||||||
|
const notifications = require('../notifications');
|
||||||
|
|
||||||
|
module.exports = function (Groups) {
|
||||||
|
Groups.requestMembership = async function (groupName, uid) {
|
||||||
|
await inviteOrRequestMembership(groupName, uid, 'request');
|
||||||
|
const username = await user.getUserField(uid, 'username');
|
||||||
|
const [notification, owners] = await Promise.all([
|
||||||
|
notifications.create({
|
||||||
|
type: 'group-request-membership',
|
||||||
|
bodyShort: '[[groups:request.notification_title, ' + username + ']]',
|
||||||
|
bodyLong: '[[groups:request.notification_text, ' + username + ', ' + groupName + ']]',
|
||||||
|
nid: 'group:' + groupName + ':uid:' + uid + ':request',
|
||||||
|
path: '/groups/' + utils.slugify(groupName),
|
||||||
|
from: uid,
|
||||||
|
}),
|
||||||
|
Groups.getOwners(groupName),
|
||||||
|
]);
|
||||||
|
|
||||||
|
await notifications.push(notification, owners);
|
||||||
|
};
|
||||||
|
|
||||||
|
Groups.acceptMembership = async function (groupName, uid) {
|
||||||
|
await db.setsRemove(['group:' + groupName + ':pending', 'group:' + groupName + ':invited'], uid);
|
||||||
|
await Groups.join(groupName, uid);
|
||||||
|
};
|
||||||
|
|
||||||
|
Groups.rejectMembership = async function (groupNames, uid) {
|
||||||
|
if (!Array.isArray(groupNames)) {
|
||||||
|
groupNames = [groupNames];
|
||||||
|
}
|
||||||
|
const sets = [];
|
||||||
|
groupNames.forEach(function (groupName) {
|
||||||
|
sets.push('group:' + groupName + ':pending', 'group:' + groupName + ':invited');
|
||||||
|
});
|
||||||
|
await db.setsRemove(sets, uid);
|
||||||
|
};
|
||||||
|
|
||||||
|
Groups.invite = async function (groupName, uid) {
|
||||||
|
await inviteOrRequestMembership(groupName, uid, 'invite');
|
||||||
|
const notification = await notifications.create({
|
||||||
|
type: 'group-invite',
|
||||||
|
bodyShort: '[[groups:invited.notification_title, ' + groupName + ']]',
|
||||||
|
bodyLong: '',
|
||||||
|
nid: 'group:' + groupName + ':uid:' + uid + ':invite',
|
||||||
|
path: '/groups/' + utils.slugify(groupName),
|
||||||
|
});
|
||||||
|
await notifications.push(notification, [uid]);
|
||||||
|
};
|
||||||
|
|
||||||
|
async function inviteOrRequestMembership(groupName, uid, type) {
|
||||||
|
if (!(parseInt(uid, 10) > 0)) {
|
||||||
|
throw new Error('[[error:not-logged-in]]');
|
||||||
|
}
|
||||||
|
|
||||||
|
const [exists, isMember, isPending, isInvited] = await Promise.all([
|
||||||
|
Groups.exists(groupName),
|
||||||
|
Groups.isMember(uid, groupName),
|
||||||
|
Groups.isPending(uid, groupName),
|
||||||
|
Groups.isInvited(uid, groupName),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (!exists) {
|
||||||
|
throw new Error('[[error:no-group]]');
|
||||||
|
} else if (isMember || (type === 'invite' && isInvited)) {
|
||||||
|
return;
|
||||||
|
} else if (type === 'request' && isPending) {
|
||||||
|
throw new Error('[[error:group-already-requested]]');
|
||||||
|
}
|
||||||
|
|
||||||
|
const set = type === 'invite' ? 'group:' + groupName + ':invited' : 'group:' + groupName + ':pending';
|
||||||
|
await db.setAdd(set, uid);
|
||||||
|
const hookName = type === 'invite' ? 'action:group.inviteMember' : 'action:group.requestMembership';
|
||||||
|
plugins.fireHook(hookName, {
|
||||||
|
groupName: groupName,
|
||||||
|
uid: uid,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Groups.isInvited = async function (uid, groupName) {
|
||||||
|
if (!(parseInt(uid, 10) > 0)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return await db.isSetMember('group:' + groupName + ':invited', uid);
|
||||||
|
};
|
||||||
|
|
||||||
|
Groups.isPending = async function (uid, groupName) {
|
||||||
|
if (!(parseInt(uid, 10) > 0)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return await db.isSetMember('group:' + groupName + ':pending', uid);
|
||||||
|
};
|
||||||
|
|
||||||
|
Groups.getPending = async function (groupName) {
|
||||||
|
if (!groupName) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return await db.getSetMembers('group:' + groupName + ':pending');
|
||||||
|
};
|
||||||
|
};
|
@ -1,123 +1,99 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
const async = require('async');
|
|
||||||
|
|
||||||
const db = require('../database');
|
const db = require('../database');
|
||||||
const user = require('../user');
|
const user = require('../user');
|
||||||
const plugins = require('../plugins');
|
const plugins = require('../plugins');
|
||||||
|
|
||||||
module.exports = function (Groups) {
|
module.exports = function (Groups) {
|
||||||
Groups.leave = function (groupNames, uid, callback) {
|
Groups.leave = async function (groupNames, uid) {
|
||||||
callback = callback || function () {};
|
|
||||||
|
|
||||||
if (Array.isArray(groupNames) && !groupNames.length) {
|
if (Array.isArray(groupNames) && !groupNames.length) {
|
||||||
return setImmediate(callback);
|
return;
|
||||||
}
|
}
|
||||||
if (!Array.isArray(groupNames)) {
|
if (!Array.isArray(groupNames)) {
|
||||||
groupNames = [groupNames];
|
groupNames = [groupNames];
|
||||||
}
|
}
|
||||||
|
|
||||||
async.waterfall([
|
const [isMembers, exists] = await Promise.all([
|
||||||
function (next) {
|
Groups.isMemberOfGroups(uid, groupNames),
|
||||||
async.parallel({
|
Groups.exists(groupNames),
|
||||||
isMembers: async.apply(Groups.isMemberOfGroups, uid, groupNames),
|
]);
|
||||||
exists: async.apply(Groups.exists, groupNames),
|
|
||||||
}, next);
|
const groupsToLeave = groupNames.filter((groupName, index) => isMembers[index] && exists[index]);
|
||||||
},
|
if (!groupsToLeave.length) {
|
||||||
function (result, next) {
|
return;
|
||||||
groupNames = groupNames.filter(function (groupName, index) {
|
}
|
||||||
return result.isMembers[index] && result.exists[index];
|
|
||||||
});
|
await Promise.all([
|
||||||
|
db.sortedSetRemove(groupsToLeave.map(groupName => 'group:' + groupName + ':members'), uid),
|
||||||
if (!groupNames.length) {
|
db.setRemove(groupsToLeave.map(groupName => 'group:' + groupName + ':owners'), uid),
|
||||||
return callback();
|
db.decrObjectField(groupsToLeave.map(groupName => 'group:' + groupName), 'memberCount'),
|
||||||
}
|
]);
|
||||||
|
|
||||||
async.parallel([
|
|
||||||
async.apply(db.sortedSetRemove, groupNames.map(groupName => 'group:' + groupName + ':members'), uid),
|
|
||||||
async.apply(db.setRemove, groupNames.map(groupName => 'group:' + groupName + ':owners'), uid),
|
|
||||||
async.apply(db.decrObjectField, groupNames.map(groupName => 'group:' + groupName), 'memberCount'),
|
|
||||||
], next);
|
|
||||||
},
|
|
||||||
function (results, next) {
|
|
||||||
Groups.clearCache(uid, groupNames);
|
|
||||||
Groups.getGroupsFields(groupNames, ['name', 'hidden', 'memberCount'], next);
|
|
||||||
},
|
|
||||||
function (groupData, next) {
|
|
||||||
if (!groupData) {
|
|
||||||
return callback();
|
|
||||||
}
|
|
||||||
var tasks = [];
|
|
||||||
|
|
||||||
var emptyPrivilegeGroups = groupData.filter(function (groupData) {
|
|
||||||
return groupData && Groups.isPrivilegeGroup(groupData.name) && groupData.memberCount === 0;
|
|
||||||
});
|
|
||||||
if (emptyPrivilegeGroups.length) {
|
|
||||||
tasks.push(async.apply(Groups.destroy, emptyPrivilegeGroups));
|
|
||||||
}
|
|
||||||
|
|
||||||
var visibleGroups = groupData.filter(groupData => groupData && !groupData.hidden);
|
|
||||||
if (visibleGroups.length) {
|
|
||||||
tasks.push(async.apply(db.sortedSetAdd, 'groups:visible:memberCount', visibleGroups.map(groupData => groupData.memberCount), visibleGroups.map(groupData => groupData.name)));
|
|
||||||
}
|
|
||||||
|
|
||||||
async.parallel(tasks, function (err) {
|
|
||||||
next(err);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
function (next) {
|
|
||||||
clearGroupTitleIfSet(groupNames, uid, next);
|
|
||||||
},
|
|
||||||
function (next) {
|
|
||||||
plugins.fireHook('action:group.leave', {
|
|
||||||
groupNames: groupNames,
|
|
||||||
uid: uid,
|
|
||||||
});
|
|
||||||
next();
|
|
||||||
},
|
|
||||||
], callback);
|
|
||||||
};
|
|
||||||
|
|
||||||
function clearGroupTitleIfSet(groupNames, uid, callback) {
|
Groups.clearCache(uid, groupsToLeave);
|
||||||
groupNames = groupNames.filter(function (groupName) {
|
|
||||||
return groupName !== 'registered-users' && !Groups.isPrivilegeGroup(groupName);
|
const groupData = await Groups.getGroupsFields(groupsToLeave, ['name', 'hidden', 'memberCount']);
|
||||||
|
if (!groupData) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const emptyPrivilegeGroups = groupData.filter(g => g && Groups.isPrivilegeGroup(g.name) && g.memberCount === 0);
|
||||||
|
const visibleGroups = groupData.filter(g => g && !g.hidden);
|
||||||
|
|
||||||
|
const promises = [];
|
||||||
|
if (emptyPrivilegeGroups.length) {
|
||||||
|
promises.push(Groups.destroy, emptyPrivilegeGroups);
|
||||||
|
}
|
||||||
|
if (visibleGroups.length) {
|
||||||
|
promises.push(db.sortedSetAdd, 'groups:visible:memberCount',
|
||||||
|
visibleGroups.map(groupData => groupData.memberCount),
|
||||||
|
visibleGroups.map(groupData => groupData.name)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all(promises);
|
||||||
|
|
||||||
|
await clearGroupTitleIfSet(groupsToLeave, uid);
|
||||||
|
|
||||||
|
plugins.fireHook('action:group.leave', {
|
||||||
|
groupNames: groupsToLeave,
|
||||||
|
uid: uid,
|
||||||
});
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
async function clearGroupTitleIfSet(groupNames, uid) {
|
||||||
|
groupNames = groupNames.filter(groupName => groupName !== 'registered-users' && !Groups.isPrivilegeGroup(groupName));
|
||||||
if (!groupNames.length) {
|
if (!groupNames.length) {
|
||||||
return callback();
|
return;
|
||||||
|
}
|
||||||
|
const userData = await user.getUserData(uid);
|
||||||
|
if (!userData) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newTitleArray = userData.groupTitleArray.filter(groupTitle => !groupNames.includes(groupTitle));
|
||||||
|
if (newTitleArray.length) {
|
||||||
|
await db.setObjectField('user:' + uid, 'groupTitle', JSON.stringify(newTitleArray));
|
||||||
|
} else {
|
||||||
|
await db.deleteObjectField('user:' + uid, 'groupTitle');
|
||||||
}
|
}
|
||||||
async.waterfall([
|
|
||||||
function (next) {
|
|
||||||
user.getUserData(uid, next);
|
|
||||||
},
|
|
||||||
function (userData, next) {
|
|
||||||
var newTitleArray = userData.groupTitleArray.filter(function (groupTitle) {
|
|
||||||
return !groupNames.includes(groupTitle);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (newTitleArray.length) {
|
|
||||||
db.setObjectField('user:' + uid, 'groupTitle', JSON.stringify(newTitleArray), next);
|
|
||||||
} else {
|
|
||||||
db.deleteObjectField('user:' + uid, 'groupTitle', next);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
], callback);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Groups.leaveAllGroups = function (uid, callback) {
|
Groups.leaveAllGroups = async function (uid) {
|
||||||
async.waterfall([
|
const groups = await db.getSortedSetRange('groups:createtime', 0, -1);
|
||||||
function (next) {
|
await Promise.all([
|
||||||
db.getSortedSetRange('groups:createtime', 0, -1, next);
|
Groups.leave(groups, uid),
|
||||||
},
|
Groups.rejectMembership(groups, uid),
|
||||||
function (groups, next) {
|
]);
|
||||||
async.parallel([
|
};
|
||||||
function (next) {
|
|
||||||
Groups.leave(groups, uid, next);
|
Groups.kick = async function (uid, groupName, isOwner) {
|
||||||
},
|
if (isOwner) {
|
||||||
function (next) {
|
// If the owners set only contains one member, error out!
|
||||||
Groups.rejectMembership(groups, uid, next);
|
const numOwners = await db.setCount('group:' + groupName + ':owners');
|
||||||
},
|
if (numOwners <= 1) {
|
||||||
], next);
|
throw new Error('[[error:group-needs-owner]]');
|
||||||
},
|
}
|
||||||
], callback);
|
}
|
||||||
|
await Groups.leave(groupName, uid);
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue