2023-01-03 11:55:46 -08:00
|
|
|
// Copyright 2019 Signal Messenger, LLC
|
2020-10-30 15:34:04 -05:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
2025-09-19 23:25:57 -05:00
|
|
|
import lodash from 'lodash';
|
2025-09-16 17:39:03 -07:00
|
|
|
import { createLogger } from '../logging/log.js';
|
|
|
|
import * as Bytes from '../Bytes.js';
|
2025-09-24 11:30:23 -05:00
|
|
|
import type { MessageAttachmentType } from '../types/AttachmentDownload.js';
|
2021-06-17 10:15:10 -07:00
|
|
|
|
2025-09-16 17:39:03 -07:00
|
|
|
import type { AttachmentType } from '../types/Attachment.js';
|
2025-06-18 13:16:29 -04:00
|
|
|
import {
|
|
|
|
doAttachmentsOnSameMessageMatch,
|
|
|
|
isDownloaded,
|
2025-09-16 17:39:03 -07:00
|
|
|
} from '../types/Attachment.js';
|
|
|
|
import { getMessageById } from '../messages/getMessageById.js';
|
|
|
|
import { trimMessageWhitespace } from '../types/BodyRange.js';
|
2025-01-10 08:18:32 +10:00
|
|
|
|
2025-09-19 23:25:57 -05:00
|
|
|
const { omit } = lodash;
|
|
|
|
|
2025-06-16 11:59:31 -07:00
|
|
|
const log = createLogger('AttachmentDownloads');
|
|
|
|
|
2025-01-10 08:18:32 +10:00
|
|
|
export async function markAttachmentAsCorrupted(
|
|
|
|
messageId: string,
|
|
|
|
attachment: AttachmentType
|
|
|
|
): Promise<void> {
|
|
|
|
const message = await getMessageById(messageId);
|
|
|
|
|
|
|
|
if (!message) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!attachment.path) {
|
|
|
|
throw new Error(
|
|
|
|
"Attachment can't be marked as corrupted because it wasn't loaded"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We intentionally don't check in quotes/stickers/contacts/... here,
|
|
|
|
// because this function should be called only for something that can
|
|
|
|
// be displayed as a generic attachment.
|
|
|
|
const attachments: ReadonlyArray<AttachmentType> =
|
|
|
|
message.get('attachments') || [];
|
|
|
|
|
|
|
|
let changed = false;
|
|
|
|
const newAttachments = attachments.map(existing => {
|
|
|
|
if (existing.path !== attachment.path) {
|
|
|
|
return existing;
|
|
|
|
}
|
|
|
|
changed = true;
|
|
|
|
|
|
|
|
return {
|
|
|
|
...existing,
|
|
|
|
isCorrupted: true,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!changed) {
|
|
|
|
throw new Error(
|
|
|
|
"Attachment can't be marked as corrupted because it wasn't found"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
log.info('markAttachmentAsCorrupted: marking an attachment as corrupted');
|
|
|
|
|
|
|
|
message.set({
|
|
|
|
attachments: newAttachments,
|
|
|
|
});
|
|
|
|
}
|
2019-01-30 12:15:07 -08:00
|
|
|
|
2024-04-15 20:11:48 -04:00
|
|
|
export async function addAttachmentToMessage(
|
2024-05-15 10:19:55 -04:00
|
|
|
messageId: string,
|
2021-06-17 10:15:10 -07:00
|
|
|
attachment: AttachmentType,
|
2024-04-19 13:09:51 -04:00
|
|
|
jobLogId: string,
|
2025-09-24 11:30:23 -05:00
|
|
|
{ type }: { type: MessageAttachmentType }
|
2021-06-17 10:15:10 -07:00
|
|
|
): Promise<void> {
|
2024-12-10 08:54:18 +10:00
|
|
|
const logPrefix = `${jobLogId}/addAttachmentToMessage`;
|
2025-01-10 08:18:32 +10:00
|
|
|
const message = await getMessageById(messageId);
|
2024-05-15 10:19:55 -04:00
|
|
|
|
2019-01-30 12:15:07 -08:00
|
|
|
if (!message) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-13 13:38:28 -07:00
|
|
|
if (type === 'long-message') {
|
2024-01-30 13:22:23 -08:00
|
|
|
let handledAnywhere = false;
|
|
|
|
let attachmentData: Uint8Array | undefined;
|
2022-05-23 16:07:41 -07:00
|
|
|
|
2019-03-13 13:38:28 -07:00
|
|
|
try {
|
2024-01-30 13:22:23 -08:00
|
|
|
if (attachment.path) {
|
2024-07-23 17:31:40 -07:00
|
|
|
const loaded =
|
|
|
|
await window.Signal.Migrations.loadAttachmentData(attachment);
|
2024-01-30 13:22:23 -08:00
|
|
|
attachmentData = loaded.data;
|
|
|
|
}
|
|
|
|
|
|
|
|
const editHistory = message.get('editHistory');
|
|
|
|
if (editHistory) {
|
|
|
|
let handledInEditHistory = false;
|
|
|
|
|
|
|
|
const newEditHistory = editHistory.map(edit => {
|
|
|
|
// We've already downloaded a bodyAttachment for this edit
|
|
|
|
if (!edit.bodyAttachment) {
|
|
|
|
return edit;
|
|
|
|
}
|
|
|
|
// This attachment isn't destined for this edit
|
|
|
|
if (
|
2025-06-18 13:16:29 -04:00
|
|
|
!doAttachmentsOnSameMessageMatch(edit.bodyAttachment, attachment)
|
2024-01-30 13:22:23 -08:00
|
|
|
) {
|
|
|
|
return edit;
|
|
|
|
}
|
|
|
|
|
|
|
|
handledInEditHistory = true;
|
|
|
|
handledAnywhere = true;
|
|
|
|
|
|
|
|
// Attachment wasn't downloaded yet.
|
|
|
|
if (!attachmentData) {
|
|
|
|
return {
|
|
|
|
...edit,
|
|
|
|
bodyAttachment: attachment,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...edit,
|
2025-02-08 02:04:29 +10:00
|
|
|
...trimMessageWhitespace({
|
|
|
|
body: Bytes.toString(attachmentData),
|
|
|
|
bodyRanges: edit.bodyRanges,
|
|
|
|
}),
|
2024-09-23 15:24:41 -04:00
|
|
|
bodyAttachment: attachment,
|
2024-01-30 13:22:23 -08:00
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (handledInEditHistory) {
|
|
|
|
message.set({ editHistory: newEditHistory });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const existingBodyAttachment = message.get('bodyAttachment');
|
|
|
|
// A bodyAttachment download might apply only to an edit, and not the top-level
|
|
|
|
if (!existingBodyAttachment) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (
|
2025-06-18 13:16:29 -04:00
|
|
|
!doAttachmentsOnSameMessageMatch(existingBodyAttachment, attachment)
|
2024-01-30 13:22:23 -08:00
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
handledAnywhere = true;
|
|
|
|
|
|
|
|
// Attachment wasn't downloaded yet.
|
|
|
|
if (!attachmentData) {
|
|
|
|
message.set({
|
|
|
|
bodyAttachment: attachment,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-13 13:38:28 -07:00
|
|
|
message.set({
|
2024-09-23 15:24:41 -04:00
|
|
|
bodyAttachment: attachment,
|
2025-02-08 02:04:29 +10:00
|
|
|
...trimMessageWhitespace({
|
|
|
|
body: Bytes.toString(attachmentData),
|
|
|
|
bodyRanges: message.get('bodyRanges'),
|
|
|
|
}),
|
2019-03-13 13:38:28 -07:00
|
|
|
});
|
|
|
|
} finally {
|
2021-06-17 10:15:10 -07:00
|
|
|
if (attachment.path) {
|
2024-01-30 13:22:23 -08:00
|
|
|
await window.Signal.Migrations.deleteAttachmentData(attachment.path);
|
|
|
|
}
|
|
|
|
if (!handledAnywhere) {
|
2024-04-15 20:11:48 -04:00
|
|
|
log.warn(
|
2024-01-30 13:22:23 -08:00
|
|
|
`${logPrefix}: Long message attachment found no matching place to apply`
|
|
|
|
);
|
2021-06-17 10:15:10 -07:00
|
|
|
}
|
2019-03-13 13:38:28 -07:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
const maybeReplaceAttachment = (existing: AttachmentType): AttachmentType => {
|
|
|
|
if (isDownloaded(existing)) {
|
|
|
|
return existing;
|
|
|
|
}
|
|
|
|
|
2025-06-18 13:16:29 -04:00
|
|
|
if (!doAttachmentsOnSameMessageMatch(existing, attachment)) {
|
2023-04-20 12:31:59 -04:00
|
|
|
return existing;
|
|
|
|
}
|
|
|
|
|
|
|
|
return attachment;
|
|
|
|
};
|
|
|
|
|
2019-01-30 12:15:07 -08:00
|
|
|
if (type === 'attachment') {
|
|
|
|
const attachments = message.get('attachments');
|
2023-03-27 19:48:57 -04:00
|
|
|
|
2024-01-30 13:22:23 -08:00
|
|
|
let handledAnywhere = false;
|
2023-03-27 19:48:57 -04:00
|
|
|
let handledInEditHistory = false;
|
|
|
|
|
|
|
|
const editHistory = message.get('editHistory');
|
|
|
|
if (editHistory) {
|
|
|
|
const newEditHistory = editHistory.map(edit => {
|
|
|
|
if (!edit.attachments) {
|
|
|
|
return edit;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...edit,
|
|
|
|
// Loop through all the attachments to find the attachment we intend
|
|
|
|
// to replace.
|
2023-04-20 12:31:59 -04:00
|
|
|
attachments: edit.attachments.map(item => {
|
|
|
|
const newItem = maybeReplaceAttachment(item);
|
|
|
|
handledInEditHistory ||= item !== newItem;
|
2024-01-30 13:22:23 -08:00
|
|
|
handledAnywhere ||= handledInEditHistory;
|
2023-04-20 12:31:59 -04:00
|
|
|
return newItem;
|
2023-03-27 19:48:57 -04:00
|
|
|
}),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (handledInEditHistory) {
|
2023-03-27 19:48:57 -04:00
|
|
|
message.set({ editHistory: newEditHistory });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (attachments) {
|
|
|
|
message.set({
|
2024-01-30 13:22:23 -08:00
|
|
|
attachments: attachments.map(item => {
|
|
|
|
const newItem = maybeReplaceAttachment(item);
|
|
|
|
handledAnywhere ||= item !== newItem;
|
|
|
|
return newItem;
|
|
|
|
}),
|
2023-04-20 12:31:59 -04:00
|
|
|
});
|
2023-03-27 19:48:57 -04:00
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2024-01-30 13:22:23 -08:00
|
|
|
if (!handledAnywhere) {
|
2024-04-15 20:11:48 -04:00
|
|
|
log.warn(
|
2024-01-30 13:22:23 -08:00
|
|
|
`${logPrefix}: 'attachment' type found no matching place to apply`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-01-30 12:15:07 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type === 'preview') {
|
|
|
|
const preview = message.get('preview');
|
2023-03-27 19:48:57 -04:00
|
|
|
|
|
|
|
let handledInEditHistory = false;
|
|
|
|
|
|
|
|
const editHistory = message.get('editHistory');
|
|
|
|
if (preview && editHistory) {
|
|
|
|
const newEditHistory = editHistory.map(edit => {
|
2023-04-20 12:31:59 -04:00
|
|
|
if (!edit.preview) {
|
2023-03-27 19:48:57 -04:00
|
|
|
return edit;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...edit,
|
2023-04-20 12:31:59 -04:00
|
|
|
preview: edit.preview.map(item => {
|
|
|
|
if (!item.image) {
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
const newImage = maybeReplaceAttachment(item.image);
|
|
|
|
handledInEditHistory ||= item.image !== newImage;
|
|
|
|
return { ...item, image: newImage };
|
|
|
|
}),
|
2023-03-27 19:48:57 -04:00
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (handledInEditHistory) {
|
2023-03-27 19:48:57 -04:00
|
|
|
message.set({ editHistory: newEditHistory });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (preview) {
|
|
|
|
message.set({
|
|
|
|
preview: preview.map(item => {
|
|
|
|
if (!item.image) {
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
...item,
|
|
|
|
image: maybeReplaceAttachment(item.image),
|
|
|
|
};
|
|
|
|
}),
|
|
|
|
});
|
2023-03-27 19:48:57 -04:00
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2019-01-30 12:15:07 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type === 'contact') {
|
2024-04-15 20:11:48 -04:00
|
|
|
const contacts = message.get('contact');
|
|
|
|
if (!contacts?.length) {
|
|
|
|
throw new Error(`${logPrefix}: no contacts, cannot add attachment!`);
|
2019-01-30 12:15:07 -08:00
|
|
|
}
|
2024-04-15 20:11:48 -04:00
|
|
|
let handled = false;
|
2023-04-20 12:31:59 -04:00
|
|
|
|
2024-04-15 20:11:48 -04:00
|
|
|
const newContacts = contacts.map(contact => {
|
|
|
|
if (!contact.avatar?.avatar) {
|
|
|
|
return contact;
|
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2024-04-15 20:11:48 -04:00
|
|
|
const existingAttachment = contact.avatar.avatar;
|
|
|
|
|
|
|
|
const newAttachment = maybeReplaceAttachment(existingAttachment);
|
|
|
|
if (existingAttachment !== newAttachment) {
|
|
|
|
handled = true;
|
|
|
|
return {
|
|
|
|
...contact,
|
|
|
|
avatar: { ...contact.avatar, avatar: newAttachment },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return contact;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!handled) {
|
|
|
|
throw new Error(
|
|
|
|
`${logPrefix}: Couldn't find matching contact with avatar attachment for message`
|
2019-01-30 12:15:07 -08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-04-15 20:11:48 -04:00
|
|
|
message.set({ contact: newContacts });
|
2019-01-30 12:15:07 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type === 'quote') {
|
|
|
|
const quote = message.get('quote');
|
2023-04-20 12:31:59 -04:00
|
|
|
const editHistory = message.get('editHistory');
|
|
|
|
let handledInEditHistory = false;
|
|
|
|
if (editHistory) {
|
|
|
|
const newEditHistory = editHistory.map(edit => {
|
|
|
|
if (!edit.quote) {
|
|
|
|
return edit;
|
|
|
|
}
|
2019-01-30 12:15:07 -08:00
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
return {
|
|
|
|
...edit,
|
|
|
|
quote: {
|
|
|
|
...edit.quote,
|
|
|
|
attachments: edit.quote.attachments.map(item => {
|
|
|
|
const { thumbnail } = item;
|
|
|
|
if (!thumbnail) {
|
2024-05-23 17:06:41 -04:00
|
|
|
return item;
|
2023-04-20 12:31:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const newThumbnail = maybeReplaceAttachment(thumbnail);
|
|
|
|
if (thumbnail !== newThumbnail) {
|
|
|
|
handledInEditHistory = true;
|
|
|
|
}
|
2025-05-22 21:09:54 -04:00
|
|
|
return { ...item, thumbnail: omit(newThumbnail, 'thumbnail') };
|
2023-04-20 12:31:59 -04:00
|
|
|
}),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (handledInEditHistory) {
|
|
|
|
message.set({ editHistory: newEditHistory });
|
|
|
|
}
|
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
if (quote) {
|
|
|
|
const newQuote = {
|
|
|
|
...quote,
|
|
|
|
attachments: quote.attachments.map(item => {
|
|
|
|
const { thumbnail } = item;
|
|
|
|
if (!thumbnail) {
|
|
|
|
return item;
|
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
return {
|
|
|
|
...item,
|
2025-05-22 21:09:54 -04:00
|
|
|
thumbnail: maybeReplaceAttachment(omit(thumbnail, 'thumbnail')),
|
2023-04-20 12:31:59 -04:00
|
|
|
};
|
|
|
|
}),
|
|
|
|
};
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2023-04-20 12:31:59 -04:00
|
|
|
message.set({ quote: newQuote });
|
|
|
|
}
|
2019-09-03 17:07:47 -07:00
|
|
|
|
2019-01-30 12:15:07 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-05-16 15:32:11 -07:00
|
|
|
if (type === 'sticker') {
|
|
|
|
const sticker = message.get('sticker');
|
|
|
|
if (!sticker) {
|
2024-01-30 13:22:23 -08:00
|
|
|
throw new Error(`${logPrefix}: sticker didn't exist`);
|
2019-05-16 15:32:11 -07:00
|
|
|
}
|
|
|
|
|
2019-08-22 15:04:14 -07:00
|
|
|
message.set({
|
|
|
|
sticker: {
|
|
|
|
...sticker,
|
2024-12-11 14:56:41 -05:00
|
|
|
data: sticker.data ? maybeReplaceAttachment(sticker.data) : attachment,
|
2019-08-22 15:04:14 -07:00
|
|
|
},
|
|
|
|
});
|
2019-01-30 12:15:07 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-01-30 13:22:23 -08:00
|
|
|
throw new Error(`${logPrefix}: Unknown job type ${type}`);
|
2019-01-30 12:15:07 -08:00
|
|
|
}
|