import { _converse, api, converse } from '@converse/headless/core.js'; import { isServerMessage, } from '@converse/headless/shared/parsers'; import { parseMessage } from './parsers.js'; import log from '@converse/headless/log.js'; const { Strophe, sizzle, u } = converse.env; export function openChat (jid) { if (!u.isValidJID(jid)) { return log.warn(`Invalid JID "${jid}" provided in URL fragment`); } api.chats.open(jid); } export async function onClearSession () { if (_converse.shouldClearCache()) { await Promise.all( _converse.chatboxes.map(c => c.messages && c.messages.clearStore({ 'silent': true })) ); const filter = o => o.get('type') !== _converse.CONTROLBOX_TYPE; _converse.chatboxes.clearStore({ 'silent': true }, filter); } } async function handleErrorMessage (stanza) { const from_jid = Strophe.getBareJidFromJid(stanza.getAttribute('from')); if (u.isSameBareJID(from_jid, _converse.bare_jid)) { return; } const chatbox = await api.chatboxes.get(from_jid); if (chatbox.get('type') === _converse.PRIVATE_CHAT_TYPE) { chatbox?.handleErrorMessageStanza(stanza); } } export function autoJoinChats () { // Automatically join private chats, based on the // "auto_join_private_chats" configuration setting. api.settings.get('auto_join_private_chats').forEach(jid => { if (_converse.chatboxes.where({ 'jid': jid }).length) { return; } if (typeof jid === 'string') { api.chats.open(jid); } else { log.error('Invalid jid criteria specified for "auto_join_private_chats"'); } }); /** * Triggered once any private chats have been automatically joined as * specified by the `auto_join_private_chats` setting. * See: https://conversejs.org/docs/html/configuration.html#auto-join-private-chats * @event _converse#privateChatsAutoJoined * @example _converse.api.listen.on('privateChatsAutoJoined', () => { ... }); * @example _converse.api.waitUntil('privateChatsAutoJoined').then(() => { ... }); */ api.trigger('privateChatsAutoJoined'); } export function registerMessageHandlers () { _converse.connection.addHandler( stanza => { if (sizzle(`message > result[xmlns="${Strophe.NS.MAM}"]`, stanza).pop()) { // MAM messages are handled in converse-mam. // We shouldn't get MAM messages here because // they shouldn't have a `type` attribute. log.warn(`Received a MAM message with type "chat".`); return true; } _converse.handleMessageStanza(stanza); return true; }, null, 'message', 'chat' ); _converse.connection.addHandler( stanza => { // Message receipts are usually without the `type` attribute. See #1353 if (stanza.getAttribute('type') !== null) { // TODO: currently Strophe has no way to register a handler // for stanzas without a `type` attribute. // We could update it to accept null to mean no attribute, // but that would be a backward-incompatible change return true; // Gets handled above. } _converse.handleMessageStanza(stanza); return true; }, Strophe.NS.RECEIPTS, 'message' ); _converse.connection.addHandler( stanza => { handleErrorMessage(stanza); return true; }, null, 'message', 'error' ); } /** * Handler method for all incoming single-user chat "message" stanzas. * @private * @param { MessageAttributes } attrs - The message attributes */ export async function handleMessageStanza (stanza) { if (isServerMessage(stanza)) { // Prosody sends headline messages with type `chat`, so we need to filter them out here. const from = stanza.getAttribute('from'); return log.info(`handleMessageStanza: Ignoring incoming server message from JID: ${from}`); } const attrs = await parseMessage(stanza, _converse); if (u.isErrorObject(attrs)) { attrs.stanza && log.error(attrs.stanza); return log.error(attrs.message); } const has_body = !!sizzle(`body, encrypted[xmlns="${Strophe.NS.OMEMO}"]`, stanza).length; const chatbox = await api.chats.get(attrs.contact_jid, { 'nickname': attrs.nick }, has_body); await chatbox?.queueMessage(attrs); /** * @typedef { Object } MessageData * An object containing the original message stanza, as well as the * parsed attributes. * @property { XMLElement } stanza * @property { MessageAttributes } stanza * @property { ChatBox } chatbox */ const data = { stanza, attrs, chatbox }; /** * Triggered when a message stanza is been received and processed. * @event _converse#message * @type { object } * @property { module:converse-chat~MessageData } data */ api.trigger('message', data); }