NodeBB/NodeBB

View on GitHub
src/groups/update.js

Summary

Maintainability
D
2 days
Test Coverage
'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 updateChatRooms(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) {
        const configKeys = [
            'groupsExemptFromPostQueue',
            'groupsExemptFromNewUserRestrictions',
            'groupsExemptFromMaintenanceMode',
        ];

        for (const key of configKeys) {
            if (meta.config[key] && meta.config[key].includes(oldName)) {
                meta.config[key].splice(
                    meta.config[key].indexOf(oldName), 1, newName
                );
                await meta.configs.set(key, meta.config[key]);
            }
        }
    }

    async function updateChatRooms(oldName, newName) {
        const messaging = require('../messaging');
        const roomIds = await db.getSortedSetRange('chat:rooms:public', 0, -1);
        const roomData = await messaging.getRoomsData(roomIds);
        const bulkSet = [];
        roomData.forEach((room) => {
            if (room && room.public && Array.isArray(room.groups) && room.groups.includes(oldName)) {
                room.groups.splice(room.groups.indexOf(oldName), 1, newName);
                bulkSet.push([`chat:room:${room.roomId}`, { groups: JSON.stringify(room.groups) }]);
            }
        });
        await db.setObjectBulk(bulkSet);
    }
};