350 lines
11 KiB
TypeScript
350 lines
11 KiB
TypeScript
// Copyright 2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import { isNumber } from 'lodash';
|
|
|
|
import * as Errors from '../../types/errors';
|
|
import { getSendOptions } from '../../util/getSendOptions';
|
|
import {
|
|
isDirectConversation,
|
|
isGroupV2,
|
|
isMe,
|
|
} from '../../util/whatTypeOfConversation';
|
|
import { SignalService as Proto } from '../../protobuf';
|
|
import {
|
|
handleMultipleSendErrors,
|
|
maybeExpandErrors,
|
|
} from './handleMultipleSendErrors';
|
|
import { ourProfileKeyService } from '../../services/ourProfileKey';
|
|
import { wrapWithSyncMessageSend } from '../../util/wrapWithSyncMessageSend';
|
|
import { DataWriter } from '../../sql/Client';
|
|
|
|
import type { ConversationModel } from '../../models/conversations';
|
|
import type {
|
|
ConversationQueueJobBundle,
|
|
DeleteForEveryoneJobData,
|
|
} from '../conversationJobQueue';
|
|
import { getUntrustedConversationServiceIds } from './getUntrustedConversationServiceIds';
|
|
import { handleMessageSend } from '../../util/handleMessageSend';
|
|
import { isConversationAccepted } from '../../util/isConversationAccepted';
|
|
import { isConversationUnregistered } from '../../util/isConversationUnregistered';
|
|
import { __DEPRECATED$getMessageById } from '../../messages/getMessageById';
|
|
import { isNotNil } from '../../util/isNotNil';
|
|
import type { CallbackResultType } from '../../textsecure/Types.d';
|
|
import type { MessageModel } from '../../models/messages';
|
|
import { SendMessageProtoError } from '../../textsecure/Errors';
|
|
import { strictAssert } from '../../util/assert';
|
|
import type { LoggerType } from '../../types/Logging';
|
|
import type { ServiceIdString } from '../../types/ServiceId';
|
|
import { isStory } from '../../messages/helpers';
|
|
import { sendToGroup } from '../../util/sendToGroup';
|
|
|
|
export async function sendDeleteForEveryone(
|
|
conversation: ConversationModel,
|
|
{
|
|
isFinalAttempt,
|
|
messaging,
|
|
shouldContinue,
|
|
timestamp,
|
|
timeRemaining,
|
|
log,
|
|
}: ConversationQueueJobBundle,
|
|
data: DeleteForEveryoneJobData
|
|
): Promise<void> {
|
|
const {
|
|
messageId,
|
|
recipients: recipientsFromJob,
|
|
revision,
|
|
targetTimestamp,
|
|
} = data;
|
|
|
|
const logId = `sendDeleteForEveryone(${conversation.idForLogging()}, ${messageId})`;
|
|
|
|
const message = await __DEPRECATED$getMessageById(messageId, logId);
|
|
if (!message) {
|
|
log.error(`${logId}: Failed to fetch message. Failing job.`);
|
|
return;
|
|
}
|
|
|
|
const story = isStory(message.attributes);
|
|
if (story && !isGroupV2(conversation.attributes)) {
|
|
log.error(`${logId}: 1-on-1 Story DOE must use its own job. Failing job`);
|
|
return;
|
|
}
|
|
|
|
if (!shouldContinue) {
|
|
log.info(`${logId}: Ran out of time. Giving up on sending`);
|
|
void updateMessageWithFailure(
|
|
message,
|
|
[new Error('Ran out of time!')],
|
|
log
|
|
);
|
|
return;
|
|
}
|
|
|
|
const sendType = 'deleteForEveryone';
|
|
const { ContentHint } = Proto.UnidentifiedSenderMessage.Message;
|
|
const contentHint = ContentHint.RESENDABLE;
|
|
const messageIds = [messageId];
|
|
|
|
const deletedForEveryoneSendStatus = message.get(
|
|
'deletedForEveryoneSendStatus'
|
|
);
|
|
const recipients = deletedForEveryoneSendStatus
|
|
? getRecipients(deletedForEveryoneSendStatus)
|
|
: recipientsFromJob
|
|
.map(recipient => {
|
|
return window.ConversationController.get(recipient)?.getServiceId();
|
|
})
|
|
.filter(isNotNil);
|
|
|
|
const untrustedServiceIds = getUntrustedConversationServiceIds(recipients);
|
|
if (untrustedServiceIds.length) {
|
|
window.reduxActions.conversations.conversationStoppedByMissingVerification({
|
|
conversationId: conversation.id,
|
|
untrustedServiceIds,
|
|
});
|
|
throw new Error(
|
|
`Delete for everyone blocked because ${untrustedServiceIds.length} conversation(s) were untrusted. Failing this attempt.`
|
|
);
|
|
}
|
|
|
|
await conversation.queueJob(
|
|
'conversationQueue/sendDeleteForEveryone',
|
|
async abortSignal => {
|
|
log.info(
|
|
`${logId}: Sending deleteForEveryone with timestamp ${timestamp}` +
|
|
`for message ${targetTimestamp}, isStory=${story}`
|
|
);
|
|
|
|
let profileKey: Uint8Array | undefined;
|
|
if (conversation.get('profileSharing')) {
|
|
profileKey = await ourProfileKeyService.get();
|
|
}
|
|
|
|
const sendOptions = await getSendOptions(conversation.attributes, {
|
|
story,
|
|
});
|
|
|
|
try {
|
|
if (isMe(conversation.attributes)) {
|
|
const proto = await messaging.getContentMessage({
|
|
deletedForEveryoneTimestamp: targetTimestamp,
|
|
profileKey,
|
|
recipients: conversation.getRecipients(),
|
|
timestamp,
|
|
expireTimerVersion: undefined,
|
|
});
|
|
strictAssert(
|
|
proto.dataMessage,
|
|
'ContentMessage must have dataMessage'
|
|
);
|
|
|
|
await handleMessageSend(
|
|
messaging.sendSyncMessage({
|
|
encodedDataMessage: Proto.DataMessage.encode(
|
|
proto.dataMessage
|
|
).finish(),
|
|
destination: conversation.get('e164'),
|
|
destinationServiceId: conversation.getServiceId(),
|
|
expirationStartTimestamp: null,
|
|
options: sendOptions,
|
|
timestamp,
|
|
urgent: false,
|
|
}),
|
|
{ messageIds, sendType }
|
|
);
|
|
await updateMessageWithSuccessfulSends(message);
|
|
} else if (isDirectConversation(conversation.attributes)) {
|
|
if (!isConversationAccepted(conversation.attributes)) {
|
|
log.info(
|
|
`conversation ${conversation.idForLogging()} is not accepted; refusing to send`
|
|
);
|
|
void updateMessageWithFailure(
|
|
message,
|
|
[new Error('Message request was not accepted')],
|
|
log
|
|
);
|
|
return;
|
|
}
|
|
if (isConversationUnregistered(conversation.attributes)) {
|
|
log.info(
|
|
`conversation ${conversation.idForLogging()} is unregistered; refusing to send`
|
|
);
|
|
void updateMessageWithFailure(
|
|
message,
|
|
[new Error('Contact no longer has a Signal account')],
|
|
log
|
|
);
|
|
return;
|
|
}
|
|
if (conversation.isBlocked()) {
|
|
log.info(
|
|
`conversation ${conversation.idForLogging()} is blocked; refusing to send`
|
|
);
|
|
void updateMessageWithFailure(
|
|
message,
|
|
[new Error('Contact is blocked')],
|
|
log
|
|
);
|
|
return;
|
|
}
|
|
|
|
await wrapWithSyncMessageSend({
|
|
conversation,
|
|
logId,
|
|
messageIds,
|
|
send: async sender =>
|
|
sender.sendMessageToServiceId({
|
|
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
serviceId: conversation.getSendTarget()!,
|
|
messageText: undefined,
|
|
attachments: [],
|
|
deletedForEveryoneTimestamp: targetTimestamp,
|
|
timestamp,
|
|
expireTimer: undefined,
|
|
expireTimerVersion: undefined,
|
|
contentHint,
|
|
groupId: undefined,
|
|
profileKey,
|
|
options: sendOptions,
|
|
urgent: true,
|
|
story,
|
|
includePniSignatureMessage: true,
|
|
}),
|
|
sendType,
|
|
timestamp,
|
|
});
|
|
|
|
await updateMessageWithSuccessfulSends(message);
|
|
} else {
|
|
if (isGroupV2(conversation.attributes) && !isNumber(revision)) {
|
|
log.error('No revision provided, but conversation is GroupV2');
|
|
}
|
|
|
|
const groupV2Info = conversation.getGroupV2Info({
|
|
members: recipients,
|
|
});
|
|
if (groupV2Info && isNumber(revision)) {
|
|
groupV2Info.revision = revision;
|
|
}
|
|
|
|
await wrapWithSyncMessageSend({
|
|
conversation,
|
|
logId,
|
|
messageIds,
|
|
send: async () =>
|
|
sendToGroup({
|
|
abortSignal,
|
|
contentHint,
|
|
groupSendOptions: {
|
|
groupV2: groupV2Info,
|
|
deletedForEveryoneTimestamp: targetTimestamp,
|
|
timestamp,
|
|
profileKey,
|
|
},
|
|
messageId,
|
|
sendOptions,
|
|
sendTarget: conversation.toSenderKeyTarget(),
|
|
sendType: 'deleteForEveryone',
|
|
story,
|
|
urgent: true,
|
|
}),
|
|
sendType,
|
|
timestamp,
|
|
});
|
|
|
|
await updateMessageWithSuccessfulSends(message);
|
|
}
|
|
} catch (error: unknown) {
|
|
if (error instanceof SendMessageProtoError) {
|
|
await updateMessageWithSuccessfulSends(message, error);
|
|
}
|
|
|
|
const errors = maybeExpandErrors(error);
|
|
await handleMultipleSendErrors({
|
|
errors,
|
|
isFinalAttempt,
|
|
log,
|
|
markFailed: () => updateMessageWithFailure(message, errors, log),
|
|
timeRemaining,
|
|
toThrow: error,
|
|
});
|
|
}
|
|
}
|
|
);
|
|
}
|
|
|
|
function getRecipients(
|
|
sendStatusByConversationId: Record<string, boolean>
|
|
): Array<ServiceIdString> {
|
|
return Object.entries(sendStatusByConversationId)
|
|
.filter(([_, isSent]) => !isSent)
|
|
.map(([conversationId]) => {
|
|
const recipient = window.ConversationController.get(conversationId);
|
|
if (!recipient) {
|
|
return null;
|
|
}
|
|
if (recipient.isUnregistered()) {
|
|
return null;
|
|
}
|
|
if (recipient.isBlocked()) {
|
|
return null;
|
|
}
|
|
return recipient.getServiceId();
|
|
})
|
|
.filter(isNotNil);
|
|
}
|
|
|
|
async function updateMessageWithSuccessfulSends(
|
|
message: MessageModel,
|
|
result?: CallbackResultType | SendMessageProtoError
|
|
): Promise<void> {
|
|
if (!result) {
|
|
message.set({
|
|
deletedForEveryoneSendStatus: {},
|
|
deletedForEveryoneFailed: undefined,
|
|
});
|
|
await DataWriter.saveMessage(message.attributes, {
|
|
ourAci: window.textsecure.storage.user.getCheckedAci(),
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
const deletedForEveryoneSendStatus = {
|
|
...message.get('deletedForEveryoneSendStatus'),
|
|
};
|
|
|
|
result.successfulServiceIds?.forEach(serviceId => {
|
|
const conversation = window.ConversationController.get(serviceId);
|
|
if (!conversation) {
|
|
return;
|
|
}
|
|
deletedForEveryoneSendStatus[conversation.id] = true;
|
|
});
|
|
|
|
message.set({
|
|
deletedForEveryoneSendStatus,
|
|
deletedForEveryoneFailed: undefined,
|
|
});
|
|
await DataWriter.saveMessage(message.attributes, {
|
|
ourAci: window.textsecure.storage.user.getCheckedAci(),
|
|
});
|
|
}
|
|
|
|
async function updateMessageWithFailure(
|
|
message: MessageModel,
|
|
errors: ReadonlyArray<unknown>,
|
|
log: LoggerType
|
|
): Promise<void> {
|
|
log.error(
|
|
'updateMessageWithFailure: Setting this set of errors',
|
|
errors.map(Errors.toLogFormat)
|
|
);
|
|
|
|
message.set({ deletedForEveryoneFailed: true });
|
|
await DataWriter.saveMessage(message.attributes, {
|
|
ourAci: window.textsecure.storage.user.getCheckedAci(),
|
|
});
|
|
}
|