2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
2021-07-15 23:48:09 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import {
|
|
|
|
DecryptionErrorMessage,
|
|
|
|
PlaintextContent,
|
2022-03-24 21:47:21 +00:00
|
|
|
} from '@signalapp/libsignal-client';
|
2024-09-30 22:23:32 +00:00
|
|
|
import { isNumber, random } from 'lodash';
|
|
|
|
import type PQueue from 'p-queue';
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2022-02-08 17:30:42 +00:00
|
|
|
import * as Bytes from '../Bytes';
|
2024-07-22 18:16:33 +00:00
|
|
|
import { DataReader, DataWriter } from '../sql/Client';
|
2021-08-06 21:21:01 +00:00
|
|
|
import { isProduction } from './version';
|
2021-08-03 00:12:49 +00:00
|
|
|
import { strictAssert } from './assert';
|
2021-07-15 23:48:09 +00:00
|
|
|
import { isGroupV2 } from './whatTypeOfConversation';
|
|
|
|
import { isOlderThan } from './timestamp';
|
|
|
|
import { parseIntOrThrow } from './parseIntOrThrow';
|
|
|
|
import * as RemoteConfig from '../RemoteConfig';
|
2021-09-10 02:38:11 +00:00
|
|
|
import { Address } from '../types/Address';
|
|
|
|
import { QualifiedAddress } from '../types/QualifiedAddress';
|
2023-08-10 16:43:33 +00:00
|
|
|
import type { AciString, ServiceIdString } from '../types/ServiceId';
|
2024-01-29 20:09:54 +00:00
|
|
|
import { ToastType } from '../types/Toast';
|
2022-01-14 21:34:52 +00:00
|
|
|
import * as Errors from '../types/errors';
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ConversationModel } from '../models/conversations';
|
|
|
|
import type {
|
2021-07-15 23:48:09 +00:00
|
|
|
DecryptionErrorEvent,
|
|
|
|
DecryptionErrorEventData,
|
2023-05-03 20:53:28 +00:00
|
|
|
InvalidPlaintextEvent,
|
2021-07-15 23:48:09 +00:00
|
|
|
RetryRequestEvent,
|
|
|
|
RetryRequestEventData,
|
2024-09-30 22:23:32 +00:00
|
|
|
SuccessfulDecryptEvent,
|
2021-07-15 23:48:09 +00:00
|
|
|
} from '../textsecure/messageReceiverEvents';
|
|
|
|
|
|
|
|
import { SignalService as Proto } from '../protobuf';
|
2021-09-17 18:27:53 +00:00
|
|
|
import * as log from '../logging/log';
|
2023-03-14 20:25:05 +00:00
|
|
|
import type MessageSender from '../textsecure/SendMessage';
|
2022-10-07 17:02:08 +00:00
|
|
|
import type { StoryDistributionListDataType } from '../state/ducks/storyDistributionLists';
|
2022-12-21 18:41:48 +00:00
|
|
|
import { drop } from './drop';
|
2023-03-14 20:25:05 +00:00
|
|
|
import { conversationJobQueue } from '../jobs/conversationJobQueue';
|
2023-04-11 03:54:43 +00:00
|
|
|
import { incrementMessageCounter } from './incrementMessageCounter';
|
2024-09-30 22:23:32 +00:00
|
|
|
import { SECOND } from './durations';
|
|
|
|
import { sleep } from './sleep';
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-11-18 22:22:24 +00:00
|
|
|
const RETRY_LIMIT = 5;
|
|
|
|
|
2023-10-11 18:38:03 +00:00
|
|
|
type RetryKeyType = `${AciString}.${number}:${number}`;
|
|
|
|
const retryRecord = new Map<RetryKeyType, number>();
|
2021-11-18 22:22:24 +00:00
|
|
|
|
2024-09-30 22:23:32 +00:00
|
|
|
const DELAY_UNIT = window.SignalCI ? 100 : SECOND;
|
|
|
|
|
|
|
|
// Entrypoints
|
|
|
|
|
|
|
|
export function onSuccessfulDecrypt(event: SuccessfulDecryptEvent): void {
|
|
|
|
const key = getRetryKey(event.data);
|
|
|
|
unregisterError(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getOnDecryptionError(getDecryptionErrorQueue: () => PQueue) {
|
|
|
|
return (event: DecryptionErrorEvent): void => {
|
|
|
|
const key = getRetryKey(event.decryptionError);
|
|
|
|
const logId = `decryption-error(${key})`;
|
|
|
|
if (isErrorRegistered(key)) {
|
|
|
|
log.warn(`${logId}: key registered before queueing job; dropping.`);
|
|
|
|
event.confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const needsDelay = !getDecryptionErrorQueue().isPaused;
|
|
|
|
|
|
|
|
registerError(key);
|
|
|
|
drop(
|
|
|
|
getDecryptionErrorQueue().add(async () => {
|
|
|
|
if (needsDelay) {
|
|
|
|
const jitter = random(5) * DELAY_UNIT;
|
|
|
|
const delay = DELAY_UNIT + jitter;
|
|
|
|
log.warn(`${logId}: delay needed; sleeping for ${delay}ms`);
|
|
|
|
await sleep(delay);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isErrorRegistered(key)) {
|
|
|
|
log.warn(`${logId}: key unregistered before job ran; dropping.`);
|
|
|
|
event.confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
await handleDecryptionError(event);
|
|
|
|
} finally {
|
|
|
|
unregisterError(key);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getRetryKey({
|
|
|
|
senderAci,
|
|
|
|
senderDevice,
|
|
|
|
timestamp,
|
|
|
|
}: {
|
|
|
|
senderAci: AciString;
|
|
|
|
senderDevice: number;
|
|
|
|
timestamp: number;
|
|
|
|
}): RetryKeyType {
|
|
|
|
return `${senderAci}.${senderDevice}:${timestamp}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
const registeredErrors = new Set<RetryKeyType>();
|
|
|
|
export function registerError(key: RetryKeyType): void {
|
|
|
|
registeredErrors.add(key);
|
|
|
|
}
|
|
|
|
export function isErrorRegistered(key: RetryKeyType): boolean {
|
|
|
|
return registeredErrors.has(key);
|
|
|
|
}
|
|
|
|
export function unregisterError(key: RetryKeyType): void {
|
|
|
|
registeredErrors.delete(key);
|
|
|
|
}
|
|
|
|
|
2023-10-11 18:38:03 +00:00
|
|
|
export function _getRetryRecord(): Map<string, number> {
|
2021-11-18 22:22:24 +00:00
|
|
|
return retryRecord;
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
export async function onRetryRequest(event: RetryRequestEvent): Promise<void> {
|
2021-10-20 21:50:00 +00:00
|
|
|
const { confirm, retryRequest } = event;
|
2021-07-15 23:48:09 +00:00
|
|
|
const {
|
|
|
|
groupId: requestGroupId,
|
|
|
|
requesterDevice,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
senderDevice,
|
|
|
|
sentAt,
|
|
|
|
} = retryRequest;
|
2023-08-10 16:43:33 +00:00
|
|
|
const logId = `${requesterAci}.${requesterDevice} ${sentAt}.${senderDevice}`;
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`onRetryRequest/${logId}: Starting...`);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-08-04 01:02:35 +00:00
|
|
|
if (!RemoteConfig.isEnabled('desktop.senderKey.retry')) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-08-04 01:02:35 +00:00
|
|
|
`onRetryRequest/${logId}: Feature flag disabled, returning early.`
|
|
|
|
);
|
2021-10-20 21:50:00 +00:00
|
|
|
confirm();
|
2021-08-04 01:02:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-09-30 22:23:32 +00:00
|
|
|
const retryKey = getRetryKey({
|
|
|
|
senderAci: requesterAci,
|
|
|
|
senderDevice: requesterDevice,
|
|
|
|
timestamp: sentAt,
|
|
|
|
});
|
2023-10-11 18:38:03 +00:00
|
|
|
const retryCount = (retryRecord.get(retryKey) || 0) + 1;
|
|
|
|
retryRecord.set(retryKey, retryCount);
|
2021-11-18 22:22:24 +00:00
|
|
|
if (retryCount > RETRY_LIMIT) {
|
|
|
|
log.warn(
|
|
|
|
`onRetryRequest/${logId}: retryCount is ${retryCount}; returning early.`
|
|
|
|
);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
if (window.RETRY_DELAY) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(`onRetryRequest/${logId}: Delaying because RETRY_DELAY is set...`);
|
2021-07-15 23:48:09 +00:00
|
|
|
await new Promise(resolve => setTimeout(resolve, 5000));
|
|
|
|
}
|
|
|
|
|
|
|
|
const HOUR = 60 * 60 * 1000;
|
2021-08-06 00:17:23 +00:00
|
|
|
const DAY = 24 * HOUR;
|
|
|
|
let retryRespondMaxAge = 14 * DAY;
|
2021-07-15 23:48:09 +00:00
|
|
|
try {
|
|
|
|
retryRespondMaxAge = parseIntOrThrow(
|
|
|
|
RemoteConfig.getValue('desktop.retryRespondMaxAge'),
|
|
|
|
'retryRespondMaxAge'
|
|
|
|
);
|
|
|
|
} catch (error) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-07-15 23:48:09 +00:00
|
|
|
`onRetryRequest/${logId}: Failed to parse integer from desktop.retryRespondMaxAge feature flag`,
|
2022-11-22 18:43:43 +00:00
|
|
|
Errors.toLogFormat(error)
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-18 22:22:24 +00:00
|
|
|
const didArchive = await archiveSessionOnMatch(retryRequest);
|
2021-09-20 18:51:30 +00:00
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
if (isOlderThan(sentAt, retryRespondMaxAge)) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-07-15 23:48:09 +00:00
|
|
|
`onRetryRequest/${logId}: Message is too old, refusing to send again.`
|
|
|
|
);
|
2021-11-18 22:22:24 +00:00
|
|
|
await sendDistributionMessageOrNullMessage(logId, retryRequest, didArchive);
|
2021-10-20 21:50:00 +00:00
|
|
|
confirm();
|
2021-07-15 23:48:09 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-22 18:16:33 +00:00
|
|
|
const sentProto = await DataWriter.getSentProtoByRecipient({
|
2021-07-15 23:48:09 +00:00
|
|
|
now: Date.now(),
|
2023-08-10 16:43:33 +00:00
|
|
|
recipientServiceId: requesterAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
timestamp: sentAt,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!sentProto) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`onRetryRequest/${logId}: Did not find sent proto`);
|
2021-11-18 22:22:24 +00:00
|
|
|
await sendDistributionMessageOrNullMessage(logId, retryRequest, didArchive);
|
2021-10-20 21:50:00 +00:00
|
|
|
confirm();
|
2021-07-15 23:48:09 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`onRetryRequest/${logId}: Resending message`);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2022-06-13 21:39:35 +00:00
|
|
|
const { messaging } = window.textsecure;
|
|
|
|
if (!messaging) {
|
|
|
|
throw new Error(`onRetryRequest/${logId}: messaging is not available!`);
|
|
|
|
}
|
|
|
|
|
2022-07-01 16:55:13 +00:00
|
|
|
const { contentHint, messageIds, proto, timestamp, urgent } = sentProto;
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2022-10-07 17:02:08 +00:00
|
|
|
// Only applies to sender key sends in groups. See below for story distribution lists.
|
|
|
|
const addSenderKeyResult = await maybeAddSenderKeyDistributionMessage({
|
2021-07-15 23:48:09 +00:00
|
|
|
contentProto: Proto.Content.decode(proto),
|
|
|
|
logId,
|
|
|
|
messageIds,
|
|
|
|
requestGroupId,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2022-02-08 17:30:42 +00:00
|
|
|
timestamp,
|
2021-07-15 23:48:09 +00:00
|
|
|
});
|
2022-10-07 17:02:08 +00:00
|
|
|
// eslint-disable-next-line prefer-destructuring
|
|
|
|
let contentProto: Proto.IContent | undefined =
|
|
|
|
addSenderKeyResult.contentProto;
|
|
|
|
const { groupId } = addSenderKeyResult;
|
|
|
|
|
|
|
|
// Assert that the requesting UUID is still part of a story distribution list that
|
|
|
|
// the message was sent to, and add its sender key distribution message (SKDM).
|
|
|
|
if (contentProto.storyMessage && !groupId) {
|
|
|
|
contentProto = await checkDistributionListAndAddSKDM({
|
|
|
|
confirm,
|
|
|
|
contentProto,
|
|
|
|
logId,
|
|
|
|
messaging,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2022-10-07 17:02:08 +00:00
|
|
|
timestamp,
|
2022-08-23 16:37:16 +00:00
|
|
|
});
|
2022-10-07 17:02:08 +00:00
|
|
|
if (!contentProto) {
|
2022-08-23 16:37:16 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2022-10-07 17:02:08 +00:00
|
|
|
const story = Boolean(contentProto.storyMessage);
|
2022-08-23 16:37:16 +00:00
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
const recipientConversation = window.ConversationController.getOrCreate(
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
'private'
|
|
|
|
);
|
2023-03-14 20:25:05 +00:00
|
|
|
const protoToSend = new Proto.Content(contentProto);
|
|
|
|
|
|
|
|
await conversationJobQueue.add({
|
|
|
|
type: 'SavedProto',
|
|
|
|
conversationId: recipientConversation.id,
|
2021-07-15 23:48:09 +00:00
|
|
|
contentHint,
|
|
|
|
groupId,
|
2023-03-14 20:25:05 +00:00
|
|
|
protoBase64: Bytes.toBase64(Proto.Content.encode(protoToSend).finish()),
|
|
|
|
story,
|
2022-07-01 16:55:13 +00:00
|
|
|
timestamp,
|
|
|
|
urgent,
|
2021-07-15 23:48:09 +00:00
|
|
|
});
|
2021-10-20 21:50:00 +00:00
|
|
|
|
|
|
|
confirm();
|
|
|
|
log.info(`onRetryRequest/${logId}: Resend complete.`);
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
|
|
|
|
2021-12-07 00:21:30 +00:00
|
|
|
function maybeShowDecryptionToast(
|
|
|
|
logId: string,
|
|
|
|
name: string,
|
|
|
|
deviceId: number
|
|
|
|
) {
|
2021-08-06 21:21:01 +00:00
|
|
|
if (isProduction(window.getVersion())) {
|
2021-07-15 23:48:09 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`maybeShowDecryptionToast/${logId}: Showing decryption error toast`);
|
2024-01-29 20:09:54 +00:00
|
|
|
window.reduxActions.toast.showToast({
|
|
|
|
toastType: ToastType.DecryptionError,
|
|
|
|
parameters: {
|
|
|
|
deviceId,
|
|
|
|
name,
|
|
|
|
},
|
2021-09-22 20:59:54 +00:00
|
|
|
});
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
|
|
|
|
2023-05-03 20:53:28 +00:00
|
|
|
export function onInvalidPlaintextMessage({
|
|
|
|
data,
|
|
|
|
}: InvalidPlaintextEvent): void {
|
2023-08-10 16:43:33 +00:00
|
|
|
const { senderAci, senderDevice, timestamp } = data;
|
|
|
|
const logId = `${senderAci}.${senderDevice} ${timestamp}`;
|
2023-05-03 20:53:28 +00:00
|
|
|
|
|
|
|
log.info(`onInvalidPlaintextMessage/${logId}: Starting...`);
|
|
|
|
|
|
|
|
const conversation = window.ConversationController.getOrCreate(
|
2023-08-10 16:43:33 +00:00
|
|
|
senderAci,
|
2023-05-03 20:53:28 +00:00
|
|
|
'private'
|
|
|
|
);
|
|
|
|
|
|
|
|
const name = conversation.getTitle();
|
|
|
|
maybeShowDecryptionToast(logId, name, senderDevice);
|
|
|
|
}
|
|
|
|
|
2024-09-30 22:23:32 +00:00
|
|
|
export async function handleDecryptionError(
|
2021-07-15 23:48:09 +00:00
|
|
|
event: DecryptionErrorEvent
|
|
|
|
): Promise<void> {
|
2021-10-20 21:50:00 +00:00
|
|
|
const { confirm, decryptionError } = event;
|
2023-08-10 16:43:33 +00:00
|
|
|
const { senderAci, senderDevice, timestamp } = decryptionError;
|
|
|
|
const logId = `${senderAci}.${senderDevice} ${timestamp}`;
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2024-09-30 22:23:32 +00:00
|
|
|
log.info(`handleDecryptionError/${logId}: Starting...`);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2024-09-30 22:23:32 +00:00
|
|
|
const retryKey = getRetryKey(decryptionError);
|
2023-10-11 18:38:03 +00:00
|
|
|
const retryCount = (retryRecord.get(retryKey) || 0) + 1;
|
|
|
|
retryRecord.set(retryKey, retryCount);
|
2021-11-18 22:22:24 +00:00
|
|
|
if (retryCount > RETRY_LIMIT) {
|
|
|
|
log.warn(
|
2024-09-30 22:23:32 +00:00
|
|
|
`handleDecryptionError/${logId}: retryCount is ${retryCount}; returning early.`
|
2021-11-18 22:22:24 +00:00
|
|
|
);
|
|
|
|
confirm();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
const conversation = window.ConversationController.getOrCreate(
|
2023-08-10 16:43:33 +00:00
|
|
|
senderAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
'private'
|
|
|
|
);
|
2021-12-07 00:21:30 +00:00
|
|
|
const name = conversation.getTitle();
|
|
|
|
maybeShowDecryptionToast(logId, name, senderDevice);
|
2021-08-05 17:25:59 +00:00
|
|
|
|
2023-08-07 23:12:57 +00:00
|
|
|
if (RemoteConfig.isEnabled('desktop.senderKey.retry')) {
|
2021-07-15 23:48:09 +00:00
|
|
|
await requestResend(decryptionError);
|
|
|
|
} else {
|
|
|
|
await startAutomaticSessionReset(decryptionError);
|
|
|
|
}
|
|
|
|
|
2021-10-20 21:50:00 +00:00
|
|
|
confirm();
|
2024-09-30 22:23:32 +00:00
|
|
|
log.info(`handleDecryptionError/${logId}: ...complete`);
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Helpers
|
|
|
|
|
|
|
|
async function archiveSessionOnMatch({
|
2021-07-23 17:44:21 +00:00
|
|
|
ratchetKey,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
requesterDevice,
|
|
|
|
senderDevice,
|
2021-11-18 22:22:24 +00:00
|
|
|
}: RetryRequestEventData): Promise<boolean> {
|
2021-07-15 23:48:09 +00:00
|
|
|
const ourDeviceId = parseIntOrThrow(
|
|
|
|
window.textsecure.storage.user.getDeviceId(),
|
|
|
|
'archiveSessionOnMatch/getDeviceId'
|
|
|
|
);
|
2021-07-23 17:44:21 +00:00
|
|
|
if (ourDeviceId !== senderDevice || !ratchetKey) {
|
2021-11-18 22:22:24 +00:00
|
|
|
return false;
|
2021-07-23 17:44:21 +00:00
|
|
|
}
|
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
const ourAci = window.textsecure.storage.user.getCheckedAci();
|
2021-09-10 02:38:11 +00:00
|
|
|
const address = new QualifiedAddress(
|
2023-08-10 16:43:33 +00:00
|
|
|
ourAci,
|
|
|
|
Address.create(requesterAci, requesterDevice)
|
2021-09-10 02:38:11 +00:00
|
|
|
);
|
2021-07-23 17:44:21 +00:00
|
|
|
const session = await window.textsecure.storage.protocol.loadSession(address);
|
|
|
|
|
|
|
|
if (session && session.currentRatchetKeyMatches(ratchetKey)) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-07-23 17:44:21 +00:00
|
|
|
'archiveSessionOnMatch: Matching device and ratchetKey, archiving session'
|
|
|
|
);
|
2021-07-15 23:48:09 +00:00
|
|
|
await window.textsecure.storage.protocol.archiveSession(address);
|
2021-11-18 22:22:24 +00:00
|
|
|
return true;
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
2021-11-18 22:22:24 +00:00
|
|
|
|
|
|
|
return false;
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async function sendDistributionMessageOrNullMessage(
|
|
|
|
logId: string,
|
2021-11-18 22:22:24 +00:00
|
|
|
options: RetryRequestEventData,
|
|
|
|
didArchive: boolean
|
2021-07-15 23:48:09 +00:00
|
|
|
): Promise<void> {
|
2023-08-10 16:43:33 +00:00
|
|
|
const { groupId, requesterAci } = options;
|
2021-07-15 23:48:09 +00:00
|
|
|
let sentDistributionMessage = false;
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`sendDistributionMessageOrNullMessage/${logId}: Starting...`);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2022-06-13 21:39:35 +00:00
|
|
|
const { messaging } = window.textsecure;
|
|
|
|
if (!messaging) {
|
|
|
|
throw new Error(
|
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: messaging is not available!`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
const conversation = window.ConversationController.getOrCreate(
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
'private'
|
|
|
|
);
|
|
|
|
|
|
|
|
if (groupId) {
|
|
|
|
const group = window.ConversationController.get(groupId);
|
|
|
|
const distributionId = group?.get('senderKeyInfo')?.distributionId;
|
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
if (group && !group.hasMember(requesterAci)) {
|
2021-07-15 23:48:09 +00:00
|
|
|
throw new Error(
|
2023-08-10 16:43:33 +00:00
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: Requester ${requesterAci} is not a member of ${conversation.idForLogging()}`
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (group && distributionId) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-07-23 17:44:21 +00:00
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: Found matching group, sending sender key distribution message`
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
try {
|
2023-03-14 20:25:05 +00:00
|
|
|
await conversationJobQueue.add({
|
|
|
|
type: 'SenderKeyDistribution',
|
|
|
|
conversationId: conversation.id,
|
|
|
|
groupId,
|
|
|
|
});
|
2021-07-15 23:48:09 +00:00
|
|
|
sentDistributionMessage = true;
|
|
|
|
} catch (error) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2023-03-14 20:25:05 +00:00
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: Failed to queue sender key distribution message`,
|
2022-11-22 18:43:43 +00:00
|
|
|
Errors.toLogFormat(error)
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!sentDistributionMessage) {
|
2021-11-18 22:22:24 +00:00
|
|
|
if (!didArchive) {
|
|
|
|
log.info(
|
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: Did't send distribution message and didn't archive session. Returning early.`
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(
|
2021-07-15 23:48:09 +00:00
|
|
|
`sendDistributionMessageOrNullMessage/${logId}: Did not send distribution message, sending null message`
|
|
|
|
);
|
|
|
|
|
2022-01-14 21:34:52 +00:00
|
|
|
// Enqueue a null message using the newly-created session
|
2021-07-15 23:48:09 +00:00
|
|
|
try {
|
2023-03-14 20:25:05 +00:00
|
|
|
await conversationJobQueue.add({
|
|
|
|
type: 'NullMessage',
|
|
|
|
conversationId: conversation.id,
|
2022-02-08 17:30:42 +00:00
|
|
|
});
|
2021-07-15 23:48:09 +00:00
|
|
|
} catch (error) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2023-03-14 20:25:05 +00:00
|
|
|
'sendDistributionMessageOrNullMessage: Failed to queue null message',
|
2022-01-14 21:34:52 +00:00
|
|
|
Errors.toLogFormat(error)
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getRetryConversation({
|
|
|
|
logId,
|
|
|
|
messageIds,
|
|
|
|
requestGroupId,
|
|
|
|
}: {
|
|
|
|
logId: string;
|
|
|
|
messageIds: Array<string>;
|
|
|
|
requestGroupId?: string;
|
|
|
|
}): Promise<ConversationModel | undefined> {
|
|
|
|
if (messageIds.length !== 1) {
|
|
|
|
// Fail over to requested groupId
|
|
|
|
return window.ConversationController.get(requestGroupId);
|
|
|
|
}
|
|
|
|
|
|
|
|
const [messageId] = messageIds;
|
2024-07-22 18:16:33 +00:00
|
|
|
const message = await DataReader.getMessageById(messageId);
|
2021-07-15 23:48:09 +00:00
|
|
|
if (!message) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-11-18 22:22:24 +00:00
|
|
|
`getRetryConversation/${logId}: Unable to find message ${messageId}`
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
// Fail over to requested groupId
|
|
|
|
return window.ConversationController.get(requestGroupId);
|
|
|
|
}
|
|
|
|
|
2021-12-10 22:51:54 +00:00
|
|
|
const { conversationId } = message;
|
2021-07-15 23:48:09 +00:00
|
|
|
return window.ConversationController.get(conversationId);
|
|
|
|
}
|
|
|
|
|
2022-10-07 17:02:08 +00:00
|
|
|
async function checkDistributionListAndAddSKDM({
|
|
|
|
contentProto,
|
|
|
|
timestamp,
|
|
|
|
confirm,
|
|
|
|
logId,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2022-10-07 17:02:08 +00:00
|
|
|
messaging,
|
|
|
|
}: {
|
|
|
|
contentProto: Proto.IContent;
|
|
|
|
timestamp: number;
|
|
|
|
confirm: () => void;
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci: AciString;
|
2022-10-07 17:02:08 +00:00
|
|
|
logId: string;
|
|
|
|
messaging: MessageSender;
|
|
|
|
}): Promise<Proto.IContent | undefined> {
|
|
|
|
let distributionList: StoryDistributionListDataType | undefined;
|
|
|
|
const { storyDistributionLists } = window.reduxStore.getState();
|
2023-08-10 16:43:33 +00:00
|
|
|
const membersByListId = new Map<string, Set<ServiceIdString>>();
|
2022-10-07 17:02:08 +00:00
|
|
|
const listsById = new Map<string, StoryDistributionListDataType>();
|
|
|
|
storyDistributionLists.distributionLists.forEach(list => {
|
2023-08-10 16:43:33 +00:00
|
|
|
membersByListId.set(list.id, new Set(list.memberServiceIds));
|
2022-10-07 17:02:08 +00:00
|
|
|
listsById.set(list.id, list);
|
|
|
|
});
|
|
|
|
|
2024-07-22 18:16:33 +00:00
|
|
|
const messages = await DataReader.getMessagesBySentAt(timestamp);
|
2022-10-07 17:02:08 +00:00
|
|
|
const isInAnyDistributionList = messages.some(message => {
|
|
|
|
const listId = message.storyDistributionListId;
|
|
|
|
if (!listId) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const members = membersByListId.get(listId);
|
|
|
|
if (!members) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
const isInList = members.has(requesterAci);
|
2022-10-07 17:02:08 +00:00
|
|
|
|
|
|
|
if (isInList) {
|
|
|
|
distributionList = listsById.get(listId);
|
|
|
|
}
|
|
|
|
|
|
|
|
return isInList;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!isInAnyDistributionList) {
|
|
|
|
log.warn(
|
2023-08-10 16:43:33 +00:00
|
|
|
`checkDistributionListAndAddSKDM/${logId}: requesterAci is not in distribution list. Dropping.`
|
2022-10-07 17:02:08 +00:00
|
|
|
);
|
|
|
|
confirm();
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
strictAssert(
|
|
|
|
distributionList,
|
|
|
|
`checkDistributionListAndAddSKDM/${logId}: Should have a distribution list by this point`
|
|
|
|
);
|
2024-07-22 18:16:33 +00:00
|
|
|
const distributionDetails = await DataReader.getStoryDistributionWithMembers(
|
|
|
|
distributionList.id
|
|
|
|
);
|
2022-10-07 17:02:08 +00:00
|
|
|
const distributionId = distributionDetails?.senderKeyInfo?.distributionId;
|
|
|
|
if (!distributionId) {
|
|
|
|
log.warn(
|
|
|
|
`onRetryRequest/${logId}: No sender key info for distribution list ${distributionList.id}`
|
|
|
|
);
|
|
|
|
return contentProto;
|
|
|
|
}
|
|
|
|
|
|
|
|
const protoWithDistributionMessage =
|
|
|
|
await messaging.getSenderKeyDistributionMessage(distributionId, {
|
|
|
|
throwIfNotInDatabase: true,
|
|
|
|
timestamp,
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
...contentProto,
|
|
|
|
senderKeyDistributionMessage:
|
|
|
|
protoWithDistributionMessage.senderKeyDistributionMessage,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
async function maybeAddSenderKeyDistributionMessage({
|
|
|
|
contentProto,
|
|
|
|
logId,
|
|
|
|
messageIds,
|
|
|
|
requestGroupId,
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci,
|
2022-02-08 17:30:42 +00:00
|
|
|
timestamp,
|
2021-07-15 23:48:09 +00:00
|
|
|
}: {
|
|
|
|
contentProto: Proto.IContent;
|
|
|
|
logId: string;
|
|
|
|
messageIds: Array<string>;
|
|
|
|
requestGroupId?: string;
|
2023-08-10 16:43:33 +00:00
|
|
|
requesterAci: AciString;
|
2022-02-08 17:30:42 +00:00
|
|
|
timestamp: number;
|
2021-11-18 22:22:24 +00:00
|
|
|
}): Promise<{
|
|
|
|
contentProto: Proto.IContent;
|
|
|
|
groupId?: string;
|
|
|
|
}> {
|
2021-07-15 23:48:09 +00:00
|
|
|
const conversation = await getRetryConversation({
|
|
|
|
logId,
|
|
|
|
messageIds,
|
|
|
|
requestGroupId,
|
|
|
|
});
|
|
|
|
|
2022-06-13 21:39:35 +00:00
|
|
|
const { messaging } = window.textsecure;
|
|
|
|
if (!messaging) {
|
|
|
|
throw new Error(
|
|
|
|
`maybeAddSenderKeyDistributionMessage/${logId}: messaging is not available!`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
if (!conversation) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-07-15 23:48:09 +00:00
|
|
|
`maybeAddSenderKeyDistributionMessage/${logId}: Unable to find conversation`
|
|
|
|
);
|
|
|
|
return {
|
|
|
|
contentProto,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
if (!conversation.hasMember(requesterAci)) {
|
2021-07-15 23:48:09 +00:00
|
|
|
throw new Error(
|
2023-08-10 16:43:33 +00:00
|
|
|
`maybeAddSenderKeyDistributionMessage/${logId}: Recipient ${requesterAci} is not a member of ${conversation.idForLogging()}`
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isGroupV2(conversation.attributes)) {
|
|
|
|
return {
|
|
|
|
contentProto,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const senderKeyInfo = conversation.get('senderKeyInfo');
|
|
|
|
if (senderKeyInfo && senderKeyInfo.distributionId) {
|
2022-02-08 17:30:42 +00:00
|
|
|
const protoWithDistributionMessage =
|
2022-06-13 21:39:35 +00:00
|
|
|
await messaging.getSenderKeyDistributionMessage(
|
2022-02-08 17:30:42 +00:00
|
|
|
senderKeyInfo.distributionId,
|
|
|
|
{ throwIfNotInDatabase: true, timestamp }
|
2021-11-11 22:43:05 +00:00
|
|
|
);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
contentProto: {
|
|
|
|
...contentProto,
|
2022-02-08 17:30:42 +00:00
|
|
|
senderKeyDistributionMessage:
|
|
|
|
protoWithDistributionMessage.senderKeyDistributionMessage,
|
2021-07-15 23:48:09 +00:00
|
|
|
},
|
|
|
|
groupId: conversation.get('groupId'),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
contentProto,
|
|
|
|
groupId: conversation.get('groupId'),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
async function requestResend(decryptionError: DecryptionErrorEventData) {
|
|
|
|
const {
|
|
|
|
cipherTextBytes,
|
|
|
|
cipherTextType,
|
|
|
|
contentHint,
|
|
|
|
groupId,
|
|
|
|
receivedAtCounter,
|
|
|
|
receivedAtDate,
|
|
|
|
senderDevice,
|
2023-08-10 16:43:33 +00:00
|
|
|
senderAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
timestamp,
|
|
|
|
} = decryptionError;
|
2023-08-10 16:43:33 +00:00
|
|
|
const logId = `${senderAci}.${senderDevice} ${timestamp}`;
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`requestResend/${logId}: Starting...`, {
|
2021-07-15 23:48:09 +00:00
|
|
|
cipherTextBytesLength: cipherTextBytes?.byteLength,
|
|
|
|
cipherTextType,
|
|
|
|
contentHint,
|
|
|
|
groupId: groupId ? `groupv2(${groupId})` : undefined,
|
|
|
|
});
|
|
|
|
|
2022-06-13 21:39:35 +00:00
|
|
|
const { messaging } = window.textsecure;
|
|
|
|
if (!messaging) {
|
|
|
|
throw new Error(`requestResend/${logId}: messaging is not available!`);
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
// 1. Find the target conversation
|
|
|
|
|
|
|
|
const sender = window.ConversationController.getOrCreate(
|
2023-08-10 16:43:33 +00:00
|
|
|
senderAci,
|
2021-07-15 23:48:09 +00:00
|
|
|
'private'
|
|
|
|
);
|
|
|
|
|
2023-03-14 20:25:05 +00:00
|
|
|
// 2. Prepare resend request
|
2021-07-15 23:48:09 +00:00
|
|
|
|
|
|
|
if (!cipherTextBytes || !isNumber(cipherTextType)) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2021-07-15 23:48:09 +00:00
|
|
|
`requestResend/${logId}: Missing cipherText information, failing over to automatic reset`
|
|
|
|
);
|
|
|
|
startAutomaticSessionReset(decryptionError);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-03-14 20:25:05 +00:00
|
|
|
const message = DecryptionErrorMessage.forOriginal(
|
|
|
|
Buffer.from(cipherTextBytes),
|
|
|
|
cipherTextType,
|
|
|
|
timestamp,
|
|
|
|
senderDevice
|
|
|
|
);
|
|
|
|
|
|
|
|
const plaintext = PlaintextContent.from(message);
|
|
|
|
|
|
|
|
// 3. Queue resend request
|
|
|
|
|
2021-07-15 23:48:09 +00:00
|
|
|
try {
|
2023-03-14 20:25:05 +00:00
|
|
|
await conversationJobQueue.add({
|
|
|
|
type: 'ResendRequest',
|
|
|
|
contentHint,
|
|
|
|
conversationId: sender.id,
|
|
|
|
groupId,
|
|
|
|
plaintext: Bytes.toBase64(plaintext.serialize()),
|
|
|
|
receivedAtCounter,
|
|
|
|
receivedAtDate,
|
2023-08-16 20:54:39 +00:00
|
|
|
senderAci,
|
2023-03-14 20:25:05 +00:00
|
|
|
senderDevice,
|
2021-07-15 23:48:09 +00:00
|
|
|
timestamp,
|
2023-03-14 20:25:05 +00:00
|
|
|
});
|
2021-07-15 23:48:09 +00:00
|
|
|
} catch (error) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.error(
|
2023-03-14 20:25:05 +00:00
|
|
|
`requestResend/${logId}: Failed to queue resend request, failing over to automatic reset`,
|
2022-11-22 18:43:43 +00:00
|
|
|
Errors.toLogFormat(error)
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
startAutomaticSessionReset(decryptionError);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
function scheduleSessionReset(senderAci: AciString, senderDevice: number) {
|
2021-07-15 23:48:09 +00:00
|
|
|
// Postpone sending light session resets until the queue is empty
|
|
|
|
const { lightSessionResetQueue } = window.Signal.Services;
|
|
|
|
|
|
|
|
if (!lightSessionResetQueue) {
|
|
|
|
throw new Error(
|
|
|
|
'scheduleSessionReset: lightSessionResetQueue is not available!'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-12-21 18:41:48 +00:00
|
|
|
drop(
|
|
|
|
lightSessionResetQueue.add(async () => {
|
2023-08-10 16:43:33 +00:00
|
|
|
const ourAci = window.textsecure.storage.user.getCheckedAci();
|
2021-09-10 02:38:11 +00:00
|
|
|
|
2022-12-21 18:41:48 +00:00
|
|
|
await window.textsecure.storage.protocol.lightSessionReset(
|
2023-08-10 16:43:33 +00:00
|
|
|
new QualifiedAddress(ourAci, Address.create(senderAci, senderDevice))
|
2022-12-21 18:41:48 +00:00
|
|
|
);
|
|
|
|
})
|
|
|
|
);
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|
|
|
|
|
2023-03-14 20:25:05 +00:00
|
|
|
export function startAutomaticSessionReset(
|
|
|
|
decryptionError: Pick<
|
|
|
|
DecryptionErrorEventData,
|
2023-08-10 16:43:33 +00:00
|
|
|
'senderAci' | 'senderDevice' | 'timestamp'
|
2023-03-14 20:25:05 +00:00
|
|
|
>
|
|
|
|
): void {
|
2023-08-10 16:43:33 +00:00
|
|
|
const { senderAci, senderDevice, timestamp } = decryptionError;
|
|
|
|
const logId = `${senderAci}.${senderDevice} ${timestamp}`;
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2021-09-17 18:27:53 +00:00
|
|
|
log.info(`startAutomaticSessionReset/${logId}: Starting...`);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2023-08-10 16:43:33 +00:00
|
|
|
scheduleSessionReset(senderAci, senderDevice);
|
2021-07-15 23:48:09 +00:00
|
|
|
|
2022-08-09 21:39:00 +00:00
|
|
|
const conversation = window.ConversationController.lookupOrCreate({
|
2023-08-16 20:54:39 +00:00
|
|
|
serviceId: senderAci,
|
2022-12-03 01:05:27 +00:00
|
|
|
reason: 'startAutomaticSessionReset',
|
2021-07-15 23:48:09 +00:00
|
|
|
});
|
|
|
|
if (!conversation) {
|
2021-09-17 18:27:53 +00:00
|
|
|
log.warn(
|
2023-03-14 20:25:05 +00:00
|
|
|
'startAutomaticSessionReset: No conversation, cannot add message to timeline'
|
2021-07-15 23:48:09 +00:00
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const receivedAt = Date.now();
|
2023-04-11 03:54:43 +00:00
|
|
|
const receivedAtCounter = incrementMessageCounter();
|
2022-12-21 18:41:48 +00:00
|
|
|
drop(
|
|
|
|
conversation.queueJob('addChatSessionRefreshed', async () => {
|
2023-03-14 20:25:05 +00:00
|
|
|
await conversation.addChatSessionRefreshed({
|
|
|
|
receivedAt,
|
|
|
|
receivedAtCounter,
|
|
|
|
});
|
2022-12-21 18:41:48 +00:00
|
|
|
})
|
|
|
|
);
|
2021-07-15 23:48:09 +00:00
|
|
|
}
|