signal-desktop/ts/util/maybeForwardMessages.ts

269 lines
7.7 KiB
TypeScript
Raw Normal View History

2023-03-20 22:23:53 +00:00
// 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';
2023-03-20 22:23:53 +00:00
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';
2023-03-20 22:23:53 +00:00
import type { StickerWithHydratedData } from '../types/Stickers';
import { drop } from './drop';
import { toLogFormat } from '../types/errors';
export type MessageForwardDraft = Readonly<{
attachments?: ReadonlyArray<AttachmentType>;
bodyRanges?: HydratedBodyRangesType;
2023-03-20 22:23:53 +00:00
hasContact: boolean;
isSticker: boolean;
2023-03-20 22:23:53 +00:00
messageBody?: string;
originalMessageId: string;
previews: ReadonlyArray<LinkPreviewType>;
2023-03-20 22:23:53 +00:00
}>;
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;
2023-03-20 22:23:53 +00:00
}
return true;
}
function isDraftEmpty(draft: MessageForwardDraft) {
const { messageBody, attachments, isSticker, hasContact } = draft;
if (isSticker || hasContact) {
return false;
2023-03-20 22:23:53 +00:00
}
if (attachments != null && attachments.length > 0) {
return false;
2023-03-20 22:23:53 +00:00
}
if (messageBody != null && messageBody.length > 0) {
return false;
2023-03-20 22:23:53 +00:00
}
return true;
2023-03-20 22:23:53 +00:00
}
export function isDraftForwardable(draft: MessageForwardDraft): boolean {
const { attachments } = draft;
if (isDraftEmpty(draft)) {
return false;
2023-03-20 22:23:53 +00:00
}
if (attachments != null && attachments.length > 0) {
if (!attachments.every(isDownloaded)) {
return false;
}
2023-03-20 22:23:53 +00:00
}
return true;
2023-03-20 22:23:53 +00:00
}
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;
2023-03-20 22:23:53 +00:00
// 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;
2023-03-20 22:23:53 +00:00
const { sticker, contact } = originalMessage;
const { attachments, bodyRanges, messageBody, previews } = draft;
2023-03-20 22:23:53 +00:00
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>;
2023-03-20 22:23:53 +00:00
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,
2023-03-20 22:23:53 +00:00
attachments: attachmentsToSend,
preview,
};
}
return { originalMessage, enqueuedMessage };
})
);
const sortedMessages = sortByMessageOrder(
preparedMessages,
message => message.originalMessage
);
// Actually send the messages
conversations.forEach(conversation => {
2023-03-20 22:23:53 +00:00
if (conversation == null) {
return;
}
2023-03-20 22:23:53 +00:00
sortedMessages.forEach(entry => {
const timestamp = baseTimestamp + timestampOffset;
timestampOffset += 1;
2023-03-20 22:23:53 +00:00
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;
}