// Copyright 2017 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { z } from 'zod'; import type { MessageModel } from '../models/messages'; import * as Errors from '../types/errors'; import * as log from '../logging/log'; import { StartupQueue } from '../util/StartupQueue'; import { drop } from '../util/drop'; import { getMessageIdForLogging } from '../util/idForLogging'; import { getMessageSentTimestamp } from '../util/getMessageSentTimestamp'; import { isIncoming } from '../state/selectors/message'; import { isMessageUnread } from '../util/isMessageUnread'; import { notificationService } from '../services/notifications'; import { queueUpdateMessage } from '../util/messageBatcher'; import { strictAssert } from '../util/assert'; import { isAciString } from '../util/isAciString'; import dataInterface from '../sql/Client'; const { removeSyncTaskById } = dataInterface; export const readSyncTaskSchema = z.object({ type: z.literal('ReadSync').readonly(), readAt: z.number(), sender: z.string().optional(), senderAci: z.string().refine(isAciString), senderId: z.string(), timestamp: z.number(), }); export type ReadSyncTaskType = z.infer; export type ReadSyncAttributesType = { envelopeId: string; syncTaskId: string; readSync: ReadSyncTaskType; }; const readSyncs = new Map(); async function remove(sync: ReadSyncAttributesType): Promise { const { syncTaskId } = sync; readSyncs.delete(syncTaskId); await removeSyncTaskById(syncTaskId); } async function maybeItIsAReactionReadSync( sync: ReadSyncAttributesType ): Promise { const { readSync } = sync; const logId = `ReadSyncs.onSync(timestamp=${readSync.timestamp})`; const readReaction = await window.Signal.Data.markReactionAsRead( readSync.senderAci, Number(readSync.timestamp) ); if ( !readReaction || readReaction?.targetAuthorAci !== window.storage.user.getCheckedAci() ) { log.info( `${logId} not found:`, readSync.senderId, readSync.sender, readSync.senderAci ); return; } log.info( `${logId} read reaction sync found:`, readReaction.conversationId, readSync.senderId, readSync.sender, readSync.senderAci ); await remove(sync); notificationService.removeBy({ conversationId: readReaction.conversationId, emoji: readReaction.emoji, targetAuthorAci: readReaction.targetAuthorAci, targetTimestamp: readReaction.targetTimestamp, }); } export async function forMessage( message: MessageModel ): Promise { const logId = `ReadSyncs.forMessage(${getMessageIdForLogging( message.attributes )})`; const sender = window.ConversationController.lookupOrCreate({ e164: message.get('source'), serviceId: message.get('sourceServiceId'), reason: logId, }); const messageTimestamp = getMessageSentTimestamp(message.attributes, { log, }); const readSyncValues = Array.from(readSyncs.values()); const foundSync = readSyncValues.find(item => { const { readSync } = item; return ( readSync.senderId === sender?.id && readSync.timestamp === messageTimestamp ); }); if (foundSync) { log.info( `${logId}: Found early read sync for message ${foundSync.readSync.timestamp}` ); await remove(foundSync); return foundSync; } return null; } export async function onSync(sync: ReadSyncAttributesType): Promise { const { readSync, syncTaskId } = sync; readSyncs.set(syncTaskId, sync); const logId = `ReadSyncs.onSync(timestamp=${readSync.timestamp})`; try { const messages = await window.Signal.Data.getMessagesBySentAt( readSync.timestamp ); const found = messages.find(item => { const sender = window.ConversationController.lookupOrCreate({ e164: item.source, serviceId: item.sourceServiceId, reason: logId, }); return isIncoming(item) && sender?.id === readSync.senderId; }); if (!found) { await maybeItIsAReactionReadSync(sync); return; } notificationService.removeBy({ messageId: found.id }); const message = window.MessageCache.__DEPRECATED$register( found.id, found, 'ReadSyncs.onSync' ); const readAt = Math.min(readSync.readAt, Date.now()); const newestSentAt = readSync.timestamp; // If message is unread, we mark it read. Otherwise, we update the expiration // timer to the time specified by the read sync if it's earlier than // the previous read time. if (isMessageUnread(message.attributes)) { // TODO DESKTOP-1509: use MessageUpdater.markRead once this is TS message.markRead(readAt, { skipSave: true }); const updateConversation = async () => { const conversation = message.getConversation(); strictAssert(conversation, `${logId}: conversation not found`); // onReadMessage may result in messages older than this one being // marked read. We want those messages to have the same expire timer // start time as this one, so we pass the readAt value through. drop(conversation.onReadMessage(message, readAt, newestSentAt)); }; // only available during initialization if (StartupQueue.isAvailable()) { const conversation = message.getConversation(); strictAssert( conversation, `${logId}: conversation not found (StartupQueue)` ); StartupQueue.add( conversation.get('id'), message.get('sent_at'), updateConversation ); } else { // not awaiting since we don't want to block work happening in the // eventHandlerQueue drop(updateConversation()); } } else { log.info(`${logId}: updating expiration`); const now = Date.now(); const existingTimestamp = message.get('expirationStartTimestamp'); const expirationStartTimestamp = Math.min( now, Math.min(existingTimestamp || now, readAt || now) ); message.set({ expirationStartTimestamp }); } queueUpdateMessage(message.attributes); await remove(sync); } catch (error) { log.error(`${logId} error:`, Errors.toLogFormat(error)); await remove(sync); } }