You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
282 lines
9.1 KiB
JavaScript
282 lines
9.1 KiB
JavaScript
'use strict';
|
|
|
|
const winston = require('winston');
|
|
|
|
const categories = require('../categories');
|
|
const plugins = require('../plugins');
|
|
const slugify = require('../slugify');
|
|
const db = require('../database');
|
|
const user = require('../user');
|
|
const batch = require('../batch');
|
|
const meta = require('../meta');
|
|
const cache = require('../cache');
|
|
|
|
|
|
module.exports = function (Groups) {
|
|
Groups.update = async function (groupName, values) {
|
|
const exists = await db.exists(`group:${groupName}`);
|
|
if (!exists) {
|
|
throw new Error('[[error:no-group]]');
|
|
}
|
|
|
|
({ values } = await plugins.hooks.fire('filter:group.update', {
|
|
groupName: groupName,
|
|
values: values,
|
|
}));
|
|
|
|
// Cast some values as bool (if not boolean already)
|
|
// 'true' and '1' = true, everything else false
|
|
['userTitleEnabled', 'private', 'hidden', 'disableJoinRequests', 'disableLeave'].forEach((prop) => {
|
|
if (values.hasOwnProperty(prop) && typeof values[prop] !== 'boolean') {
|
|
values[prop] = values[prop] === 'true' || parseInt(values[prop], 10) === 1;
|
|
}
|
|
});
|
|
|
|
const payload = {
|
|
description: values.description || '',
|
|
icon: values.icon || '',
|
|
labelColor: values.labelColor || '#000000',
|
|
textColor: values.textColor || '#ffffff',
|
|
};
|
|
|
|
if (values.hasOwnProperty('userTitle')) {
|
|
payload.userTitle = values.userTitle || '';
|
|
}
|
|
|
|
if (values.hasOwnProperty('userTitleEnabled')) {
|
|
payload.userTitleEnabled = values.userTitleEnabled ? '1' : '0';
|
|
}
|
|
|
|
if (values.hasOwnProperty('hidden')) {
|
|
payload.hidden = values.hidden ? '1' : '0';
|
|
}
|
|
|
|
if (values.hasOwnProperty('private')) {
|
|
payload.private = values.private ? '1' : '0';
|
|
}
|
|
|
|
if (values.hasOwnProperty('disableJoinRequests')) {
|
|
payload.disableJoinRequests = values.disableJoinRequests ? '1' : '0';
|
|
}
|
|
|
|
if (values.hasOwnProperty('disableLeave')) {
|
|
payload.disableLeave = values.disableLeave ? '1' : '0';
|
|
}
|
|
|
|
if (values.hasOwnProperty('name')) {
|
|
await checkNameChange(groupName, values.name);
|
|
}
|
|
|
|
if (values.hasOwnProperty('private')) {
|
|
await updatePrivacy(groupName, values.private);
|
|
}
|
|
|
|
if (values.hasOwnProperty('hidden')) {
|
|
await updateVisibility(groupName, values.hidden);
|
|
}
|
|
|
|
if (values.hasOwnProperty('memberPostCids')) {
|
|
const validCids = await categories.getCidsByPrivilege('categories:cid', groupName, 'topics:read');
|
|
const cidsArray = values.memberPostCids.split(',').map(cid => parseInt(cid.trim(), 10)).filter(Boolean);
|
|
payload.memberPostCids = cidsArray.filter(cid => validCids.includes(cid)).join(',') || '';
|
|
}
|
|
|
|
await db.setObject(`group:${groupName}`, payload);
|
|
await Groups.renameGroup(groupName, values.name);
|
|
|
|
plugins.hooks.fire('action:group.update', {
|
|
name: groupName,
|
|
values: values,
|
|
});
|
|
};
|
|
|
|
async function updateVisibility(groupName, hidden) {
|
|
if (hidden) {
|
|
await db.sortedSetRemoveBulk([
|
|
['groups:visible:createtime', groupName],
|
|
['groups:visible:memberCount', groupName],
|
|
['groups:visible:name', `${groupName.toLowerCase()}:${groupName}`],
|
|
]);
|
|
return;
|
|
}
|
|
const groupData = await db.getObjectFields(`group:${groupName}`, ['createtime', 'memberCount']);
|
|
await db.sortedSetAddBulk([
|
|
['groups:visible:createtime', groupData.createtime, groupName],
|
|
['groups:visible:memberCount', groupData.memberCount, groupName],
|
|
['groups:visible:name', 0, `${groupName.toLowerCase()}:${groupName}`],
|
|
]);
|
|
}
|
|
|
|
Groups.hide = async function (groupName) {
|
|
await showHide(groupName, 'hidden');
|
|
};
|
|
|
|
Groups.show = async function (groupName) {
|
|
await showHide(groupName, 'show');
|
|
};
|
|
|
|
async function showHide(groupName, hidden) {
|
|
hidden = hidden === 'hidden';
|
|
await Promise.all([
|
|
db.setObjectField(`group:${groupName}`, 'hidden', hidden ? 1 : 0),
|
|
updateVisibility(groupName, hidden),
|
|
]);
|
|
}
|
|
|
|
async function updatePrivacy(groupName, isPrivate) {
|
|
const groupData = await Groups.getGroupFields(groupName, ['private']);
|
|
const currentlyPrivate = groupData.private === 1;
|
|
if (!currentlyPrivate || currentlyPrivate === isPrivate) {
|
|
return;
|
|
}
|
|
const pendingUids = await db.getSetMembers(`group:${groupName}:pending`);
|
|
if (!pendingUids.length) {
|
|
return;
|
|
}
|
|
|
|
winston.verbose(`[groups.update] Group is now public, automatically adding ${pendingUids.length} new members, who were pending prior.`);
|
|
|
|
for (const uid of pendingUids) {
|
|
/* eslint-disable no-await-in-loop */
|
|
await Groups.join(groupName, uid);
|
|
}
|
|
await db.delete(`group:${groupName}:pending`);
|
|
}
|
|
|
|
async function checkNameChange(currentName, newName) {
|
|
if (Groups.isPrivilegeGroup(newName)) {
|
|
throw new Error('[[error:invalid-group-name]]');
|
|
}
|
|
const currentSlug = slugify(currentName);
|
|
const newSlug = slugify(newName);
|
|
if (currentName === newName || currentSlug === newSlug) {
|
|
return;
|
|
}
|
|
Groups.validateGroupName(newName);
|
|
const [group, exists] = await Promise.all([
|
|
Groups.getGroupData(currentName),
|
|
Groups.existsBySlug(newSlug),
|
|
]);
|
|
|
|
if (exists) {
|
|
throw new Error('[[error:group-already-exists]]');
|
|
}
|
|
|
|
if (!group) {
|
|
throw new Error('[[error:no-group]]');
|
|
}
|
|
|
|
if (group.system) {
|
|
throw new Error('[[error:not-allowed-to-rename-system-group]]');
|
|
}
|
|
}
|
|
|
|
Groups.renameGroup = async function (oldName, newName) {
|
|
if (oldName === newName || !newName || String(newName).length === 0) {
|
|
return;
|
|
}
|
|
const group = await db.getObject(`group:${oldName}`);
|
|
if (!group) {
|
|
return;
|
|
}
|
|
|
|
const exists = await Groups.exists(newName);
|
|
if (exists) {
|
|
throw new Error('[[error:group-already-exists]]');
|
|
}
|
|
|
|
await updateMemberGroupTitles(oldName, newName);
|
|
await updateNavigationItems(oldName, newName);
|
|
await updateWidgets(oldName, newName);
|
|
await updateConfig(oldName, newName);
|
|
await db.setObject(`group:${oldName}`, { name: newName, slug: slugify(newName) });
|
|
await db.deleteObjectField('groupslug:groupname', group.slug);
|
|
await db.setObjectField('groupslug:groupname', slugify(newName), newName);
|
|
|
|
const allGroups = await db.getSortedSetRange('groups:createtime', 0, -1);
|
|
const keys = allGroups.map(group => `group:${group}:members`);
|
|
await renameGroupsMember(keys, oldName, newName);
|
|
cache.del(keys);
|
|
|
|
await db.rename(`group:${oldName}`, `group:${newName}`);
|
|
await db.rename(`group:${oldName}:members`, `group:${newName}:members`);
|
|
await db.rename(`group:${oldName}:owners`, `group:${newName}:owners`);
|
|
await db.rename(`group:${oldName}:pending`, `group:${newName}:pending`);
|
|
await db.rename(`group:${oldName}:invited`, `group:${newName}:invited`);
|
|
await db.rename(`group:${oldName}:member:pids`, `group:${newName}:member:pids`);
|
|
|
|
await renameGroupsMember(['groups:createtime', 'groups:visible:createtime', 'groups:visible:memberCount'], oldName, newName);
|
|
await renameGroupsMember(['groups:visible:name'], `${oldName.toLowerCase()}:${oldName}`, `${newName.toLowerCase()}:${newName}`);
|
|
|
|
plugins.hooks.fire('action:group.rename', {
|
|
old: oldName,
|
|
new: newName,
|
|
});
|
|
Groups.cache.reset();
|
|
};
|
|
|
|
async function updateMemberGroupTitles(oldName, newName) {
|
|
await batch.processSortedSet(`group:${oldName}:members`, async (uids) => {
|
|
let usersData = await user.getUsersData(uids);
|
|
usersData = usersData.filter(userData => userData && userData.groupTitleArray.includes(oldName));
|
|
|
|
usersData.forEach((userData) => {
|
|
userData.newTitleArray = userData.groupTitleArray.map(oldTitle => (oldTitle === oldName ? newName : oldTitle));
|
|
});
|
|
|
|
await Promise.all(usersData.map(u => user.setUserField(u.uid, 'groupTitle', JSON.stringify(u.newTitleArray))));
|
|
}, {});
|
|
}
|
|
|
|
async function renameGroupsMember(keys, oldName, newName) {
|
|
const isMembers = await db.isMemberOfSortedSets(keys, oldName);
|
|
keys = keys.filter((key, index) => isMembers[index]);
|
|
if (!keys.length) {
|
|
return;
|
|
}
|
|
const scores = await db.sortedSetsScore(keys, oldName);
|
|
await db.sortedSetsRemove(keys, oldName);
|
|
await db.sortedSetsAdd(keys, scores, newName);
|
|
}
|
|
|
|
async function updateNavigationItems(oldName, newName) {
|
|
const navigation = require('../navigation/admin');
|
|
const navItems = await navigation.get();
|
|
navItems.forEach((navItem) => {
|
|
if (navItem && Array.isArray(navItem.groups) && navItem.groups.includes(oldName)) {
|
|
navItem.groups.splice(navItem.groups.indexOf(oldName), 1, newName);
|
|
}
|
|
});
|
|
navigation.unescapeFields(navItems);
|
|
await navigation.save(navItems);
|
|
}
|
|
|
|
async function updateWidgets(oldName, newName) {
|
|
const admin = require('../widgets/admin');
|
|
const widgets = require('../widgets');
|
|
|
|
const data = await admin.get();
|
|
|
|
data.areas.forEach((area) => {
|
|
area.widgets = area.data;
|
|
area.widgets.forEach((widget) => {
|
|
if (widget && widget.data && Array.isArray(widget.data.groups) && widget.data.groups.includes(oldName)) {
|
|
widget.data.groups.splice(widget.data.groups.indexOf(oldName), 1, newName);
|
|
}
|
|
});
|
|
});
|
|
for (const area of data.areas) {
|
|
if (area.data.length) {
|
|
await widgets.setArea(area);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function updateConfig(oldName, newName) {
|
|
if (meta.config.groupsExemptFromPostQueue.includes(oldName)) {
|
|
meta.config.groupsExemptFromPostQueue.splice(meta.config.groupsExemptFromPostQueue.indexOf(oldName), 1, newName);
|
|
await meta.configs.set('groupsExemptFromPostQueue', meta.config.groupsExemptFromPostQueue);
|
|
}
|
|
}
|
|
};
|