// Copyright 2020 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only /* eslint-disable max-classes-per-file */ import { Collection, Model } from 'backbone'; import type { MessageModel } from '../models/messages'; import { getContactId } from '../messages/helpers'; import * as log from '../logging/log'; import * as Errors from '../types/errors'; import { deleteForEveryone } from '../util/deleteForEveryone'; import { drop } from '../util/drop'; import { getMessageSentTimestampSet } from '../util/getMessageSentTimestampSet'; export type DeleteAttributesType = { targetSentTimestamp: number; serverTimestamp: number; fromId: string; }; export class DeleteModel extends Model {} let singleton: Deletes | undefined; export class Deletes extends Collection { static getSingleton(): Deletes { if (!singleton) { singleton = new Deletes(); } return singleton; } forMessage(message: MessageModel): Array { const sentTimestamps = getMessageSentTimestampSet(message.attributes); const matchingDeletes = this.filter(item => { return ( item.get('fromId') === getContactId(message.attributes) && sentTimestamps.has(item.get('targetSentTimestamp')) ); }); if (matchingDeletes.length > 0) { log.info('Found early DOE for message'); this.remove(matchingDeletes); return matchingDeletes; } return []; } async onDelete(del: DeleteModel): Promise { try { // The conversation the deleted message was in; we have to find it in the database // to to figure that out. const targetConversation = await window.ConversationController.getConversationForTargetMessage( del.get('fromId'), del.get('targetSentTimestamp') ); if (!targetConversation) { log.info( 'No target conversation for DOE', del.get('fromId'), del.get('targetSentTimestamp') ); return; } // Do not await, since this can deadlock the queue drop( targetConversation.queueJob('Deletes.onDelete', async () => { log.info('Handling DOE for', del.get('targetSentTimestamp')); const messages = await window.Signal.Data.getMessagesBySentAt( del.get('targetSentTimestamp') ); const targetMessage = messages.find( m => del.get('fromId') === getContactId(m) && !m.deletedForEveryone ); if (!targetMessage) { log.info( 'No message for DOE', del.get('fromId'), del.get('targetSentTimestamp') ); return; } const message = window.MessageController.register( targetMessage.id, targetMessage ); await deleteForEveryone(message, del); this.remove(del); }) ); } catch (error) { log.error('Deletes.onDelete error:', Errors.toLogFormat(error)); } } }