UNPKG

converse.js

Version:
161 lines (153 loc) 6.67 kB
/** * @copyright The Converse.js contributors * @license Mozilla Public License (MPLv2) * @module:muc-affiliations-utils * @typedef {module:plugin-muc-parsers.MemberListItem} MemberListItem * @typedef {module:plugin-muc-affiliations-api.User} User * @typedef {import('@converse/skeletor').Model} Model * @typedef {import('../constants').AFFILIATIONS} AFFILIATIONS */ import _converse from '../../../shared/_converse.js'; import api from '../../../shared/api/index.js'; import converse from '../../../shared/api/public.js'; import log from "@converse/log"; import { parseMemberListIQ } from '../parsers.js'; const { Strophe, $iq, u, stx } = converse.env; /** * Sends an IQ stanza to the server, asking it for the relevant affiliation list . * Returns an array of {@link MemberListItem} objects, representing occupants * that have the given affiliation. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * @param {import('../types').NonOutcastAffiliation} affiliation * @param {string} muc_jid - The JID of the MUC for which the affiliation list should be fetched * @returns {Promise<MemberListItem[]|Error>} */ export async function getAffiliationList (affiliation, muc_jid) { const iq = stx` <iq xmlns="jabber:client" to="${muc_jid}" type="get"> <query xmlns="${Strophe.NS.MUC_ADMIN}"> <item affiliation="${affiliation}"/> </query> </iq>`; const result = await api.sendIQ(iq, null, false); if (result === null) { const err_msg = `Error: timeout while fetching ${affiliation} list for MUC ${muc_jid}`; const err = new Error(err_msg); log.warn(err_msg); return err; } if (u.isErrorStanza(result)) { const err_msg = `Error: not allowed to fetch ${affiliation} list for MUC ${muc_jid}`; const err = new Error(err_msg); log.warn(err_msg); log.warn(result); return err; } return parseMemberListIQ(result) .filter(p => p) .sort((a, b) => (a.nick < b.nick ? -1 : a.nick > b.nick ? 1 : 0)); } /** * Send IQ stanzas to the server to modify affiliations for users in this groupchat. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * @param {String|Array<String>} muc_jid - The JID(s) of the MUCs in which the * @param {Array<User>} users * @returns {Promise} */ export function setAffiliations (muc_jid, users) { const affiliations = [...new Set(users.map(u => u.affiliation))]; return Promise.all(affiliations.map(a => setAffiliation(a, muc_jid, users))); } /** * Send IQ stanzas to the server to set an affiliation for * the provided JIDs. * See: https://xmpp.org/extensions/xep-0045.html#modifymember * * Prosody doesn't accept multiple JIDs' affiliations * being set in one IQ stanza, so as a workaround we send * a separate stanza for each JID. * Related ticket: https://issues.prosody.im/345 * * @param {AFFILIATIONS[number]} affiliation - The affiliation to be set * @param {String|Array<String>} muc_jids - The JID(s) of the MUCs in which the * affiliations need to be set. * @param {object} members - A map of jids, affiliations and * optionally reasons. Only those entries with the * same affiliation as being currently set will be considered. * @returns {Promise} A promise which resolves and fails depending on the XMPP server response. */ export function setAffiliation (affiliation, muc_jids, members) { if (!Array.isArray(muc_jids)) { muc_jids = [muc_jids]; } members = members.filter(m => [undefined, affiliation].includes(m.affiliation)); return Promise.all( muc_jids.reduce((acc, jid) => [...acc, ...members.map(m => sendAffiliationIQ(affiliation, jid, m))], []) ); } /** * Send an IQ stanza specifying an affiliation change. * @param {AFFILIATIONS[number]} affiliation - Affiliation (could also be stored on the member object). * @param {string} muc_jid - The JID of the MUC in which the affiliation should be set. * @param {object} member - Map containing the member's jid and optionally a reason and affiliation. */ function sendAffiliationIQ (affiliation, muc_jid, member) { affiliation = member.affiliation || affiliation; const iq = $iq({ to: muc_jid, type: 'set' }) .c('query', { xmlns: Strophe.NS.MUC_ADMIN }) .c('item', { affiliation, ...(affiliation === 'outcast' ? {} : {nick: member.nick }), jid: member.jid }); if (member.reason !== undefined) { iq.c('reason', member.reason); } return api.sendIQ(iq); } /** * Given two lists of objects with 'jid', 'affiliation' and * 'reason' properties, return a new list containing * those objects that are new, changed or removed * (depending on the 'remove_absentees' boolean). * * The affiliations for new and changed members stay the * same, for removed members, the affiliation is set to 'none'. * * The 'reason' property is not taken into account when * comparing whether affiliations have been changed. * @param {boolean} exclude_existing - Indicates whether JIDs from * the new list which are also in the old list * (regardless of affiliation) should be excluded * from the delta. One reason to do this * would be when you want to add a JID only if it * doesn't have *any* existing affiliation at all. * @param {boolean} remove_absentees - Indicates whether JIDs * from the old list which are not in the new list * should be considered removed and therefore be * included in the delta with affiliation set * to 'none'. * @param {array} new_list - Array containing the new affiliations * @param {array} old_list - Array containing the old affiliations * @returns {array} */ export function computeAffiliationsDelta (exclude_existing, remove_absentees, new_list, old_list) { const new_jids = new_list.map(o => o.jid); const old_jids = old_list.map(o => o.jid); // Get the new affiliations let delta = new_jids.filter(jid => !old_jids.includes(jid)).map(jid => new_list[new_jids.indexOf(jid)]); if (!exclude_existing) { // Get the changed affiliations delta = delta.concat( new_list.filter(item => { const idx = old_jids.indexOf(item.jid); return idx >= 0 ? item.affiliation !== old_list[idx].affiliation : false; }) ); } if (remove_absentees) { // Get the removed affiliations delta = delta.concat(old_jids.filter(jid => !new_jids.includes(jid)).map(jid => ({ 'jid': jid, 'affiliation': 'none' }))); } return delta; }