// Copyright 2024 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import LRU from 'lru-cache'; import type { AddressableAttachmentType, LocalAttachmentV2Type, } from '../types/Attachment'; import * as log from '../logging/log'; import { AttachmentDisposition } from './getLocalAttachmentUrl'; let setCheck = false; const lru = new LRU>({ max: 1000, }); export type EncryptLegacyAttachmentOptionsType = Readonly<{ logId: string; disposition?: AttachmentDisposition; readAttachmentData: ( attachment: Partial ) => Promise; writeNewAttachmentData: (data: Uint8Array) => Promise; }>; export async function encryptLegacyAttachment< T extends Partial >(attachment: T, options: EncryptLegacyAttachmentOptionsType): Promise { // Not downloaded if (!attachment.path) { return attachment; } // Already upgraded if (attachment.version === 2) { return attachment; } const { disposition = AttachmentDisposition.Attachment } = options; const cacheKey = `${disposition}:${attachment.path}`; let promise = lru.get(cacheKey); if (!promise) { promise = doEncrypt(attachment, options); lru.set(cacheKey, promise); } try { const modern = await promise; return { ...attachment, ...modern, }; } catch (error) { const { logId } = options; log.error(`${logId}: migration failed, falling back to original`, error); return attachment; } } async function doEncrypt>( attachment: T, { readAttachmentData, writeNewAttachmentData, }: EncryptLegacyAttachmentOptionsType ): Promise { const data = await readAttachmentData(attachment); const result = await writeNewAttachmentData(data); // Remove fully migrated attachments without references on next startup. if (!setCheck) { setCheck = true; await window.storage.put('needOrphanedAttachmentCheck', true); } return result; }