// Copyright 2016 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { isEqual } from 'lodash'; import type { MessageModel } from '../models/messages'; import type { MessageAttributesType } from '../model-types.d'; import type { SendStateByConversationId } from '../messages/MessageSendState'; import { isOutgoing, isStory } from '../state/selectors/message'; import { getOwn } from '../util/getOwn'; import { missingCaseError } from '../util/missingCaseError'; import { createWaitBatcher } from '../util/waitBatcher'; import type { ServiceIdString } from '../types/ServiceId'; import * as Errors from '../types/errors'; import { SendActionType, SendStatus, UNDELIVERED_SEND_STATUSES, sendStateReducer, } from '../messages/MessageSendState'; import type { DeleteSentProtoRecipientOptionsType } from '../sql/Interface'; import dataInterface from '../sql/Client'; import * as log from '../logging/log'; import { getSourceServiceId } from '../messages/helpers'; import { queueUpdateMessage } from '../util/messageBatcher'; import { getMessageSentTimestamp } from '../util/getMessageSentTimestamp'; import { getMessageIdForLogging } from '../util/idForLogging'; import { generateCacheKey } from './generateCacheKey'; import { getPropForTimestamp } from '../util/editHelpers'; const { deleteSentProtoRecipient } = dataInterface; export enum MessageReceiptType { Delivery = 'Delivery', Read = 'Read', View = 'View', } export type MessageReceiptAttributesType = { envelopeId: string; messageSentAt: number; receiptTimestamp: number; removeFromMessageReceiverCache: () => void; sourceConversationId: string; sourceDevice: number; sourceServiceId: ServiceIdString; type: MessageReceiptType; wasSentEncrypted: boolean; }; const receipts = new Map(); const deleteSentProtoBatcher = createWaitBatcher({ name: 'deleteSentProtoBatcher', wait: 250, maxSize: 30, async processBatch(items: Array) { log.info( `MessageReceipts: Batching ${items.length} sent proto recipients deletes` ); const { successfulPhoneNumberShares } = await deleteSentProtoRecipient( items ); for (const serviceId of successfulPhoneNumberShares) { const convo = window.ConversationController.get(serviceId); if (!convo) { continue; } log.info( 'MessageReceipts: unsetting shareMyPhoneNumber ' + `for ${convo.idForLogging()}` ); // `deleteSentProtoRecipient` has already updated the database so there // is no need in calling `updateConversation` convo.unset('shareMyPhoneNumber'); } }, }); function remove(receipt: MessageReceiptAttributesType): void { receipts.delete( generateCacheKey({ sender: receipt.sourceServiceId, timestamp: receipt.messageSentAt, type: receipt.type, }) ); receipt.removeFromMessageReceiverCache(); } function getTargetMessage({ sourceConversationId, messages, targetTimestamp, }: { sourceConversationId: string; messages: ReadonlyArray; targetTimestamp: number; }): MessageModel | null { if (messages.length === 0) { return null; } const matchingMessages = messages .filter(msg => isOutgoing(msg) || isStory(msg)) .filter(msg => { const sendStateByConversationId = getPropForTimestamp({ message: msg, prop: 'sendStateByConversationId', targetTimestamp, log, }); const isRecipient = Object.hasOwn( sendStateByConversationId ?? {}, sourceConversationId ); if (!isRecipient) { return false; } const sendStatus = sendStateByConversationId?.[sourceConversationId]?.status; if ( sendStatus === undefined || UNDELIVERED_SEND_STATUSES.includes(sendStatus) ) { log.warn(` MessageReceipts.getTargetMessage: received receipt for undelivered message, status: ${sendStatus}, sourceConversationId: ${sourceConversationId}, message: ${getMessageIdForLogging(message)}. `); return false; } return true; }); if (matchingMessages.length === 0) { return null; } if (matchingMessages.length > 1) { log.warn(` MessageReceipts.getTargetMessage: multiple (${matchingMessages.length}) matching messages for receipt, sentAt=${targetTimestamp}, sourceConversationId=${sourceConversationId} `); } const message = matchingMessages[0]; return window.MessageCache.__DEPRECATED$register( message.id, message, 'MessageReceipts.getTargetMessage' ); } const wasDeliveredWithSealedSender = ( conversationId: string, message: MessageModel ): boolean => (message.get('unidentifiedDeliveries') || []).some( identifier => window.ConversationController.getConversationId(identifier) === conversationId ); const shouldDropReceipt = ( receipt: MessageReceiptAttributesType, message: MessageModel ): boolean => { const { type } = receipt; switch (type) { case MessageReceiptType.Delivery: return false; case MessageReceiptType.Read: return !window.storage.get('read-receipt-setting'); case MessageReceiptType.View: if (isStory(message.attributes)) { return !window.Events.getStoryViewReceiptsEnabled(); } return !window.storage.get('read-receipt-setting'); default: throw missingCaseError(type); } }; export function forMessage( message: MessageModel ): Array { if (!isOutgoing(message.attributes) && !isStory(message.attributes)) { return []; } const logId = `MessageReceipts.forMessage(${getMessageIdForLogging( message.attributes )})`; const ourAci = window.textsecure.storage.user.getCheckedAci(); const sourceServiceId = getSourceServiceId(message.attributes); if (ourAci !== sourceServiceId) { return []; } const receiptValues = Array.from(receipts.values()); const sentAt = getMessageSentTimestamp(message.attributes, { log }); const result = receiptValues.filter(item => item.messageSentAt === sentAt); if (result.length > 0) { log.info(`${logId}: found early receipts for message ${sentAt}`); result.forEach(receipt => { remove(receipt); }); } return result.filter(receipt => { if (shouldDropReceipt(receipt, message)) { log.info( `${logId}: Dropping an early receipt ${receipt.type} for message ${sentAt}` ); return false; } return true; }); } function getNewSendStateByConversationId( oldSendStateByConversationId: SendStateByConversationId, receipt: MessageReceiptAttributesType ): SendStateByConversationId { const { receiptTimestamp, sourceConversationId, type } = receipt; const oldSendState = getOwn( oldSendStateByConversationId, sourceConversationId ) ?? { status: SendStatus.Sent, updatedAt: undefined }; let sendActionType: SendActionType; switch (type) { case MessageReceiptType.Delivery: sendActionType = SendActionType.GotDeliveryReceipt; break; case MessageReceiptType.Read: sendActionType = SendActionType.GotReadReceipt; break; case MessageReceiptType.View: sendActionType = SendActionType.GotViewedReceipt; break; default: throw missingCaseError(type); } const newSendState = sendStateReducer(oldSendState, { type: sendActionType, updatedAt: receiptTimestamp, }); return { ...oldSendStateByConversationId, [sourceConversationId]: newSendState, }; } async function updateMessageSendState( receipt: MessageReceiptAttributesType, message: MessageModel ): Promise { const { messageSentAt } = receipt; const logId = `MessageReceipts.updateMessageSendState(sentAt=${receipt.messageSentAt})`; if (shouldDropReceipt(receipt, message)) { log.info( `${logId}: Dropping a receipt ${receipt.type} for message ${messageSentAt}` ); return; } let hasChanges = false; const editHistory = message.get('editHistory') ?? []; const newEditHistory = editHistory?.map(edit => { if (messageSentAt !== edit.timestamp) { return edit; } const oldSendStateByConversationId = edit.sendStateByConversationId ?? {}; const newSendStateByConversationId = getNewSendStateByConversationId( oldSendStateByConversationId, receipt ); return { ...edit, sendStateByConversationId: newSendStateByConversationId, }; }); if (!isEqual(newEditHistory, editHistory)) { message.set('editHistory', newEditHistory); hasChanges = true; } const editMessageTimestamp = message.get('editMessageTimestamp'); if ( (!editMessageTimestamp && messageSentAt === message.get('timestamp')) || messageSentAt === editMessageTimestamp ) { const oldSendStateByConversationId = message.get('sendStateByConversationId') ?? {}; const newSendStateByConversationId = getNewSendStateByConversationId( oldSendStateByConversationId, receipt ); // The send state may not change. For example, this can happen if we get a read // receipt before a delivery receipt. if (!isEqual(oldSendStateByConversationId, newSendStateByConversationId)) { message.set('sendStateByConversationId', newSendStateByConversationId); hasChanges = true; } } if (hasChanges) { queueUpdateMessage(message.attributes); // notify frontend listeners const conversation = window.ConversationController.get( message.get('conversationId') ); const updateLeftPane = conversation ? conversation.debouncedUpdateLastMessage : undefined; if (updateLeftPane) { updateLeftPane(); } } const { sourceConversationId, type } = receipt; if ( (type === MessageReceiptType.Delivery && wasDeliveredWithSealedSender(sourceConversationId, message) && receipt.wasSentEncrypted) || type === MessageReceiptType.Read ) { const recipient = window.ConversationController.get(sourceConversationId); const recipientServiceId = recipient?.getServiceId(); const deviceId = receipt.sourceDevice; if (recipientServiceId && deviceId) { await Promise.all([ deleteSentProtoBatcher.add({ timestamp: messageSentAt, recipientServiceId, deviceId, }), // We want the above call to not be delayed when testing with // CI. window.SignalCI ? deleteSentProtoBatcher.flushAndWait() : Promise.resolve(), ]); } else { log.warn( `${logId}: Missing serviceId or deviceId for deliveredTo ${sourceConversationId}` ); } } } export async function onReceipt( receipt: MessageReceiptAttributesType ): Promise { receipts.set( generateCacheKey({ sender: receipt.sourceServiceId, timestamp: receipt.messageSentAt, type: receipt.type, }), receipt ); const { messageSentAt, sourceConversationId, sourceServiceId, type } = receipt; const logId = `MessageReceipts.onReceipt(sentAt=${receipt.messageSentAt})`; try { const messages = await window.Signal.Data.getMessagesBySentAt( messageSentAt ); const message = getTargetMessage({ sourceConversationId, messages, targetTimestamp: receipt.messageSentAt, }); if (message) { await updateMessageSendState(receipt, message); } else { // We didn't find any messages but maybe it's a story sent message const targetMessages = messages.filter( item => item.storyDistributionListId && item.sendStateByConversationId && !item.deletedForEveryone && Boolean(item.sendStateByConversationId[sourceConversationId]) ); // Nope, no target message was found if (!targetMessages.length) { log.info( `${logId}: No message for receipt`, type, sourceConversationId, sourceServiceId ); return; } await Promise.all( targetMessages.map(msg => { const model = window.MessageCache.__DEPRECATED$register( msg.id, msg, 'MessageReceipts.onReceipt' ); return updateMessageSendState(receipt, model); }) ); } remove(receipt); } catch (error) { remove(receipt); log.error(`${logId} error:`, Errors.toLogFormat(error)); } }