import { Meteor } from 'meteor/meteor'; import _ from 'underscore'; import { getRoomByNameOrIdWithOptionToJoin } from './getRoomByNameOrIdWithOptionToJoin'; import { sendMessage } from './sendMessage'; import { validateRoomMessagePermissionsAsync } from '../../../authorization/server/functions/canSendMessage'; import { SystemLogger } from '../../../../server/lib/logger/system'; import { trim } from '../../../../lib/utils/stringUtils'; export const processWebhookMessage = async function (messageObj, user, defaultValues = { channel: '', alias: '', avatar: '', emoji: '' }) { const sentData = []; const channels = [].concat(messageObj.channel || messageObj.roomId || defaultValues.channel); for await (const channel of channels) { const channelType = channel[0]; let channelValue = channel.substr(1); let room; switch (channelType) { case '#': room = await getRoomByNameOrIdWithOptionToJoin({ user, nameOrId: channelValue, joinChannel: true, }); break; case '@': room = await getRoomByNameOrIdWithOptionToJoin({ user, nameOrId: channelValue, type: 'd', }); break; default: channelValue = channelType + channelValue; // Try to find the room by id or name if they didn't include the prefix. room = await getRoomByNameOrIdWithOptionToJoin({ user, nameOrId: channelValue, joinChannel: true, errorOnEmpty: false, }); if (room) { break; } // We didn't get a room, let's try finding direct messages room = await getRoomByNameOrIdWithOptionToJoin({ user, nameOrId: channelValue, tryDirectByUserIdOnly: true, type: 'd', }); if (room) { break; } // No room, so throw an error throw new Meteor.Error('invalid-channel'); } if (messageObj.attachments && !Array.isArray(messageObj.attachments)) { SystemLogger.warn({ msg: 'Attachments should be Array, ignoring value', attachments: messageObj.attachments, }); messageObj.attachments = undefined; } const message = { alias: messageObj.username || messageObj.alias || defaultValues.alias, msg: trim(messageObj.text || messageObj.msg || ''), attachments: messageObj.attachments || [], parseUrls: messageObj.parseUrls !== undefined ? messageObj.parseUrls : !messageObj.attachments, bot: messageObj.bot, groupable: messageObj.groupable !== undefined ? messageObj.groupable : false, tmid: messageObj.tmid, }; if (!_.isEmpty(messageObj.icon_url) || !_.isEmpty(messageObj.avatar)) { message.avatar = messageObj.icon_url || messageObj.avatar; } else if (!_.isEmpty(messageObj.icon_emoji) || !_.isEmpty(messageObj.emoji)) { message.emoji = messageObj.icon_emoji || messageObj.emoji; } else if (!_.isEmpty(defaultValues.avatar)) { message.avatar = defaultValues.avatar; } else if (!_.isEmpty(defaultValues.emoji)) { message.emoji = defaultValues.emoji; } if (Array.isArray(message.attachments)) { for (let i = 0; i < message.attachments.length; i++) { const attachment = message.attachments[i]; if (attachment.msg) { attachment.text = trim(attachment.msg); delete attachment.msg; } } } await validateRoomMessagePermissionsAsync(room, { uid: user._id, ...user }); const messageReturn = await sendMessage(user, message, room); sentData.push({ channel, message: messageReturn }); } return sentData; };