2022-07-01 00:52:03 +00:00
|
|
|
// Copyright 2022 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import type { AttachmentType } from '../types/Attachment';
|
|
|
|
import type { LinkPreviewType } from '../types/message/LinkPreviews';
|
|
|
|
import type { MessageAttributesType } from '../model-types.d';
|
|
|
|
import * as log from '../logging/log';
|
2022-08-16 23:59:11 +00:00
|
|
|
import { SafetyNumberChangeSource } from '../components/SafetyNumberChangeDialog';
|
|
|
|
import { blockSendUntilConversationsAreVerified } from './blockSendUntilConversationsAreVerified';
|
2022-07-01 00:52:03 +00:00
|
|
|
import { getMessageIdForLogging } from './idForLogging';
|
2022-08-16 23:59:11 +00:00
|
|
|
import { isNotNil } from './isNotNil';
|
2022-07-01 00:52:03 +00:00
|
|
|
import { resetLinkPreview } from '../services/LinkPreview';
|
2022-12-08 07:43:48 +00:00
|
|
|
import { getRecipientsByConversation } from './getRecipientsByConversation';
|
2022-07-01 00:52:03 +00:00
|
|
|
|
|
|
|
export async function maybeForwardMessage(
|
|
|
|
messageAttributes: MessageAttributesType,
|
|
|
|
conversationIds: Array<string>,
|
|
|
|
messageBody?: string,
|
|
|
|
attachments?: Array<AttachmentType>,
|
|
|
|
linkPreview?: LinkPreviewType
|
|
|
|
): Promise<boolean> {
|
|
|
|
const idForLogging = getMessageIdForLogging(messageAttributes);
|
|
|
|
log.info(`maybeForwardMessage/${idForLogging}: Starting...`);
|
|
|
|
|
|
|
|
const attachmentLookup = new Set();
|
|
|
|
if (attachments) {
|
|
|
|
attachments.forEach(attachment => {
|
|
|
|
attachmentLookup.add(`${attachment.fileName}/${attachment.contentType}`);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-16 23:59:11 +00:00
|
|
|
const conversations = conversationIds
|
|
|
|
.map(id => window.ConversationController.get(id))
|
|
|
|
.filter(isNotNil);
|
2022-07-01 00:52:03 +00:00
|
|
|
|
|
|
|
const cannotSend = conversations.some(
|
|
|
|
conversation =>
|
|
|
|
conversation?.get('announcementsOnly') && !conversation.areWeAdmin()
|
|
|
|
);
|
|
|
|
if (cannotSend) {
|
|
|
|
throw new Error('Cannot send to group');
|
|
|
|
}
|
|
|
|
|
2022-12-08 07:43:48 +00:00
|
|
|
const recipientsByConversation = getRecipientsByConversation(
|
|
|
|
conversations.map(x => x.attributes)
|
|
|
|
);
|
2022-11-11 04:10:30 +00:00
|
|
|
|
2022-07-01 00:52:03 +00:00
|
|
|
// Verify that all contacts that we're forwarding
|
2022-08-16 23:59:11 +00:00
|
|
|
// to are verified and trusted.
|
2022-07-01 00:52:03 +00:00
|
|
|
// 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.
|
2022-08-16 23:59:11 +00:00
|
|
|
const canSend = await blockSendUntilConversationsAreVerified(
|
2022-11-11 04:10:30 +00:00
|
|
|
recipientsByConversation,
|
2022-08-16 23:59:11 +00:00
|
|
|
SafetyNumberChangeSource.MessageSend
|
|
|
|
);
|
|
|
|
if (!canSend) {
|
|
|
|
return false;
|
2022-07-01 00:52:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const sendMessageOptions = { dontClearDraft: true };
|
|
|
|
const baseTimestamp = Date.now();
|
|
|
|
|
|
|
|
const {
|
|
|
|
loadAttachmentData,
|
|
|
|
loadContactData,
|
|
|
|
loadPreviewData,
|
|
|
|
loadStickerData,
|
|
|
|
} = window.Signal.Migrations;
|
|
|
|
|
|
|
|
// Actually send the message
|
|
|
|
// load any sticker data, attachments, or link previews that we need to
|
|
|
|
// send along with the message and do the send to each conversation.
|
|
|
|
await Promise.all(
|
|
|
|
conversations.map(async (conversation, offset) => {
|
|
|
|
const timestamp = baseTimestamp + offset;
|
|
|
|
if (conversation) {
|
|
|
|
const { sticker, contact } = messageAttributes;
|
|
|
|
|
|
|
|
if (sticker) {
|
|
|
|
const stickerWithData = await loadStickerData(sticker);
|
|
|
|
const stickerNoPath = stickerWithData
|
|
|
|
? {
|
|
|
|
...stickerWithData,
|
|
|
|
data: {
|
|
|
|
...stickerWithData.data,
|
|
|
|
path: undefined,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
: undefined;
|
|
|
|
|
|
|
|
conversation.enqueueMessageForSend(
|
|
|
|
{
|
|
|
|
body: undefined,
|
|
|
|
attachments: [],
|
|
|
|
sticker: stickerNoPath,
|
|
|
|
},
|
|
|
|
{ ...sendMessageOptions, timestamp }
|
|
|
|
);
|
|
|
|
} else if (contact?.length) {
|
|
|
|
const contactWithHydratedAvatar = await loadContactData(contact);
|
|
|
|
conversation.enqueueMessageForSend(
|
|
|
|
{
|
|
|
|
body: undefined,
|
|
|
|
attachments: [],
|
|
|
|
contact: contactWithHydratedAvatar,
|
|
|
|
},
|
|
|
|
{ ...sendMessageOptions, timestamp }
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const preview = linkPreview
|
|
|
|
? await loadPreviewData([linkPreview])
|
|
|
|
: [];
|
|
|
|
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}`
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
conversation.enqueueMessageForSend(
|
|
|
|
{
|
|
|
|
body: messageBody || undefined,
|
|
|
|
attachments: attachmentsToSend,
|
|
|
|
preview,
|
|
|
|
},
|
|
|
|
{ ...sendMessageOptions, timestamp }
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
// Cancel any link still pending, even if it didn't make it into the message
|
|
|
|
resetLinkPreview();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|