// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { orderBy } from 'lodash'; import type { AttachmentType } from '../types/Attachment'; import { isVoiceMessage, isDownloaded } from '../types/Attachment'; import type { LinkPreviewType, LinkPreviewWithHydratedData, } from '../types/message/LinkPreviews'; import type { MessageAttributesType, QuotedMessageType } from '../model-types'; import * as log from '../logging/log'; import { SafetyNumberChangeSource } from '../components/SafetyNumberChangeDialog'; import { blockSendUntilConversationsAreVerified } from './blockSendUntilConversationsAreVerified'; import { getMessageIdForLogging, getConversationIdForLogging, } from './idForLogging'; import { isNotNil } from './isNotNil'; import { resetLinkPreview } from '../services/LinkPreview'; import { getRecipientsByConversation } from './getRecipientsByConversation'; import type { EmbeddedContactWithHydratedAvatar } from '../types/EmbeddedContact'; import type { DraftBodyRanges, HydratedBodyRangesType, } from '../types/BodyRange'; import type { StickerWithHydratedData } from '../types/Stickers'; import { drop } from './drop'; import { toLogFormat } from '../types/errors'; export type MessageForwardDraft = Readonly<{ attachments?: ReadonlyArray<AttachmentType>; bodyRanges?: HydratedBodyRangesType; hasContact: boolean; isSticker: boolean; messageBody?: string; originalMessageId: string; previews: ReadonlyArray<LinkPreviewType>; }>; export type ForwardMessageData = Readonly<{ originalMessage: MessageAttributesType; draft: MessageForwardDraft; }>; export function isDraftEditable(draft: MessageForwardDraft): boolean { if (draft.isSticker) { return false; } if (draft.hasContact) { return false; } const hasVoiceMessage = draft.attachments?.some(isVoiceMessage) ?? false; if (hasVoiceMessage) { return false; } return true; } function isDraftEmpty(draft: MessageForwardDraft) { const { messageBody, attachments, isSticker, hasContact } = draft; if (isSticker || hasContact) { return false; } if (attachments != null && attachments.length > 0) { return false; } if (messageBody != null && messageBody.length > 0) { return false; } return true; } export function isDraftForwardable(draft: MessageForwardDraft): boolean { const { attachments } = draft; if (isDraftEmpty(draft)) { return false; } if (attachments != null && attachments.length > 0) { if (!attachments.every(isDownloaded)) { return false; } } return true; } export function sortByMessageOrder<T>( items: ReadonlyArray<T>, getMesssage: ( item: T ) => Pick<MessageAttributesType, 'sent_at' | 'received_at'> ): Array<T> { return orderBy( items, [item => getMesssage(item).received_at, item => getMesssage(item).sent_at], ['ASC', 'ASC'] ); } export async function maybeForwardMessages( messages: Array<ForwardMessageData>, conversationIds: ReadonlyArray<string> ): Promise<boolean> { log.info( `maybeForwardMessage: Attempting to forward ${messages.length} messages...` ); const conversations = conversationIds .map(id => window.ConversationController.get(id)) .filter(isNotNil); const cannotSend = conversations.some( conversation => conversation?.get('announcementsOnly') && !conversation.areWeAdmin() ); if (cannotSend) { throw new Error('Cannot send to group'); } const recipientsByConversation = getRecipientsByConversation( conversations.map(x => x.attributes) ); // Verify that all contacts that we're forwarding // to are verified and trusted. // If there are any unverified or untrusted contacts, show the // SendAnywayDialog and if we're fine with sending then mark all as // verified and trusted and continue the send. const canSend = await blockSendUntilConversationsAreVerified( recipientsByConversation, SafetyNumberChangeSource.MessageSend ); if (!canSend) { return false; } const sendMessageOptions = { dontClearDraft: true }; const baseTimestamp = Date.now(); const { loadAttachmentData, loadContactData, loadPreviewData, loadStickerData, } = window.Signal.Migrations; let timestampOffset = 0; // load any sticker data, attachments, or link previews that we need to // send along with the message and do the send to each conversation. const preparedMessages = await Promise.all( messages.map(async message => { const { draft, originalMessage } = message; const { sticker, contact } = originalMessage; const { attachments, bodyRanges, messageBody, previews } = draft; const idForLogging = getMessageIdForLogging(originalMessage); log.info(`maybeForwardMessage: Forwarding ${idForLogging}`); const attachmentLookup = new Set(); if (attachments) { attachments.forEach(attachment => { attachmentLookup.add( `${attachment.fileName}/${attachment.contentType}` ); }); } let enqueuedMessage: { attachments: Array<AttachmentType>; body: string | undefined; bodyRanges?: DraftBodyRanges; contact?: Array<EmbeddedContactWithHydratedAvatar>; preview?: Array<LinkPreviewWithHydratedData>; quote?: QuotedMessageType; sticker?: StickerWithHydratedData; }; if (sticker) { const stickerWithData = await loadStickerData(sticker); const stickerNoPath = stickerWithData ? { ...stickerWithData, data: { ...stickerWithData.data, path: undefined, }, } : undefined; enqueuedMessage = { body: undefined, attachments: [], sticker: stickerNoPath, }; } else if (contact?.length) { const contactWithHydratedAvatar = await loadContactData(contact); enqueuedMessage = { body: undefined, attachments: [], contact: contactWithHydratedAvatar, }; } else { const preview = await loadPreviewData([...previews]); const attachmentsWithData = await Promise.all( (attachments || []).map(async item => ({ ...(await loadAttachmentData(item)), path: undefined, })) ); const attachmentsToSend = attachmentsWithData.filter( (attachment: Partial<AttachmentType>) => attachmentLookup.has( `${attachment.fileName}/${attachment.contentType}` ) ); enqueuedMessage = { body: messageBody || undefined, bodyRanges, attachments: attachmentsToSend, preview, }; } return { originalMessage, enqueuedMessage }; }) ); const sortedMessages = sortByMessageOrder( preparedMessages, message => message.originalMessage ); // Actually send the messages conversations.forEach(conversation => { if (conversation == null) { return; } sortedMessages.forEach(entry => { const timestamp = baseTimestamp + timestampOffset; timestampOffset += 1; const { enqueuedMessage, originalMessage } = entry; drop( conversation .enqueueMessageForSend(enqueuedMessage, { ...sendMessageOptions, timestamp, }) .catch(error => { log.error( 'maybeForwardMessage: message send error', getConversationIdForLogging(conversation.attributes), getMessageIdForLogging(originalMessage), toLogFormat(error) ); }) ); }); }); // Cancel any link still pending, even if it didn't make it into the message resetLinkPreview(); return true; }