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.
nodebb/src/groups/update.js

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);
}
}
};