2023-01-13 00:24:59 +00:00
|
|
|
// Copyright 2023 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import type { MessageAttributesType } from '../model-types.d';
|
|
|
|
import type { MessageModel } from '../models/messages';
|
|
|
|
import * as log from '../logging/log';
|
|
|
|
import { isMoreRecentThan } from './timestamp';
|
|
|
|
|
|
|
|
const MAX_ATTACHMENT_DOWNLOAD_AGE = 3600 * 72 * 1000;
|
|
|
|
const MAX_ATTACHMENT_MSGS_TO_DOWNLOAD = 250;
|
|
|
|
|
|
|
|
let isEnabled = true;
|
|
|
|
let attachmentDownloadQueue: Array<MessageModel> | undefined = [];
|
2023-10-19 17:10:08 +00:00
|
|
|
const queueEmptyCallbacks: Set<() => void> = new Set();
|
2023-01-13 00:24:59 +00:00
|
|
|
|
|
|
|
export function shouldUseAttachmentDownloadQueue(): boolean {
|
|
|
|
return isEnabled;
|
|
|
|
}
|
|
|
|
|
2023-10-19 17:10:08 +00:00
|
|
|
export function isAttachmentDownloadQueueEmpty(): boolean {
|
|
|
|
return !(attachmentDownloadQueue ?? []).length;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function registerQueueEmptyCallback(callback: () => void): void {
|
|
|
|
queueEmptyCallbacks.add(callback);
|
|
|
|
}
|
|
|
|
|
2023-01-13 00:24:59 +00:00
|
|
|
export function addToAttachmentDownloadQueue(
|
|
|
|
idLog: string,
|
|
|
|
message: MessageModel
|
|
|
|
): void {
|
|
|
|
if (!attachmentDownloadQueue) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
attachmentDownloadQueue.unshift(message);
|
|
|
|
|
|
|
|
log.info(
|
|
|
|
`${idLog}: Adding to attachmentDownloadQueue`,
|
|
|
|
message.get('sent_at')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function flushAttachmentDownloadQueue(): Promise<void> {
|
|
|
|
if (!attachmentDownloadQueue) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: ts/models/messages.ts expects this global to become undefined
|
|
|
|
// once we stop processing the queue.
|
|
|
|
isEnabled = false;
|
|
|
|
|
|
|
|
const attachmentsToDownload = attachmentDownloadQueue.filter(
|
|
|
|
(message, index) =>
|
|
|
|
index <= MAX_ATTACHMENT_MSGS_TO_DOWNLOAD ||
|
|
|
|
isMoreRecentThan(message.getReceivedAt(), MAX_ATTACHMENT_DOWNLOAD_AGE) ||
|
|
|
|
// Stickers and long text attachments has to be downloaded for UI
|
|
|
|
// to display the message properly.
|
|
|
|
message.hasRequiredAttachmentDownloads()
|
|
|
|
);
|
|
|
|
|
|
|
|
log.info(
|
|
|
|
'Downloading recent attachments of total attachments',
|
|
|
|
attachmentsToDownload.length,
|
|
|
|
attachmentDownloadQueue.length
|
|
|
|
);
|
|
|
|
|
|
|
|
const messagesWithDownloads = await Promise.all(
|
2024-05-01 19:22:06 +00:00
|
|
|
attachmentsToDownload.map(message => {
|
|
|
|
const updatedMessage =
|
|
|
|
window.MessageCache.__DEPRECATED$getById(message.id) ?? message;
|
|
|
|
return updatedMessage.queueAttachmentDownloads();
|
|
|
|
})
|
2023-01-13 00:24:59 +00:00
|
|
|
);
|
|
|
|
const messagesToSave: Array<MessageAttributesType> = [];
|
|
|
|
messagesWithDownloads.forEach((shouldSave, messageKey) => {
|
|
|
|
if (shouldSave) {
|
|
|
|
const message = attachmentsToDownload[messageKey];
|
|
|
|
messagesToSave.push(message.attributes);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await window.Signal.Data.saveMessages(messagesToSave, {
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci: window.storage.user.getCheckedAci(),
|
2023-01-13 00:24:59 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
attachmentDownloadQueue = undefined;
|
2023-10-19 17:10:08 +00:00
|
|
|
queueEmptyCallbacks.forEach(callback => callback());
|
|
|
|
queueEmptyCallbacks.clear();
|
2023-01-13 00:24:59 +00:00
|
|
|
}
|