103 lines
3.3 KiB
TypeScript
103 lines
3.3 KiB
TypeScript
// Copyright 2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { ConversationAttributesType } from '../model-types.d';
|
|
import type { ConversationQueueJobData } from '../jobs/conversationJobQueue';
|
|
import { DataWriter } from '../sql/Client';
|
|
import * as Errors from '../types/errors';
|
|
import { DAY } from './durations';
|
|
import * as log from '../logging/log';
|
|
import {
|
|
conversationJobQueue,
|
|
conversationQueueJobEnum,
|
|
} from '../jobs/conversationJobQueue';
|
|
import { deleteForEveryone } from './deleteForEveryone';
|
|
import {
|
|
getConversationIdForLogging,
|
|
getMessageIdForLogging,
|
|
} from './idForLogging';
|
|
import { __DEPRECATED$getMessageById } from '../messages/getMessageById';
|
|
import { getRecipientConversationIds } from './getRecipientConversationIds';
|
|
import { getRecipients } from './getRecipients';
|
|
import { repeat, zipObject } from './iterables';
|
|
import { isMe } from './whatTypeOfConversation';
|
|
|
|
export async function sendDeleteForEveryoneMessage(
|
|
conversationAttributes: ConversationAttributesType,
|
|
options: {
|
|
deleteForEveryoneDuration?: number;
|
|
id: string;
|
|
timestamp: number;
|
|
}
|
|
): Promise<void> {
|
|
const {
|
|
deleteForEveryoneDuration,
|
|
timestamp: targetTimestamp,
|
|
id: messageId,
|
|
} = options;
|
|
const message = await __DEPRECATED$getMessageById(messageId);
|
|
if (!message) {
|
|
throw new Error('sendDeleteForEveryoneMessage: Cannot find message!');
|
|
}
|
|
const idForLogging = getMessageIdForLogging(message.attributes);
|
|
|
|
// If conversation is a Note To Self, no deletion time limits apply.
|
|
if (!isMe(conversationAttributes)) {
|
|
const timestamp = Date.now();
|
|
const maxDuration = deleteForEveryoneDuration || DAY;
|
|
if (timestamp - targetTimestamp > maxDuration) {
|
|
throw new Error(
|
|
`Cannot send DOE for a message older than ${maxDuration}`
|
|
);
|
|
}
|
|
}
|
|
|
|
message.set({
|
|
deletedForEveryoneSendStatus: zipObject(
|
|
getRecipientConversationIds(conversationAttributes),
|
|
repeat(false)
|
|
),
|
|
});
|
|
|
|
const conversationIdForLogging = getConversationIdForLogging(
|
|
conversationAttributes
|
|
);
|
|
|
|
log.info(
|
|
`sendDeleteForEveryoneMessage: enqueuing DeleteForEveryone: ${idForLogging} ` +
|
|
`in conversation ${conversationIdForLogging}`
|
|
);
|
|
|
|
try {
|
|
const jobData: ConversationQueueJobData = {
|
|
type: conversationQueueJobEnum.enum.DeleteForEveryone,
|
|
conversationId: conversationAttributes.id,
|
|
messageId,
|
|
recipients: getRecipients(conversationAttributes),
|
|
revision: conversationAttributes.revision,
|
|
targetTimestamp,
|
|
};
|
|
await conversationJobQueue.add(jobData, async jobToInsert => {
|
|
log.info(
|
|
`sendDeleteForEveryoneMessage: Deleting message ${idForLogging} ` +
|
|
`in conversation ${conversationIdForLogging} with job ${jobToInsert.id}`
|
|
);
|
|
await DataWriter.saveMessage(message.attributes, {
|
|
jobToInsert,
|
|
ourAci: window.textsecure.storage.user.getCheckedAci(),
|
|
});
|
|
});
|
|
} catch (error) {
|
|
log.error(
|
|
`sendDeleteForEveryoneMessage: Failed to queue delete for everyone for message ${idForLogging}`,
|
|
Errors.toLogFormat(error)
|
|
);
|
|
throw error;
|
|
}
|
|
|
|
await deleteForEveryone(message, {
|
|
targetSentTimestamp: targetTimestamp,
|
|
serverTimestamp: Date.now(),
|
|
fromId: window.ConversationController.getOurConversationIdOrThrow(),
|
|
});
|
|
}
|