RocketChat/Rocket.Chat

View on GitHub
apps/meteor/ee/server/local-services/message-reads/service.ts

Summary

Maintainability
A
2 hrs
Test Coverage
import { ServiceClassInternal, api } from '@rocket.chat/core-services';
import { Messages, MessageReads, Subscriptions } from '@rocket.chat/models';

import { MAX_ROOM_SIZE_CHECK_INDIVIDUAL_READ_RECEIPTS } from '../../lib/constants';
import { ReadReceipt } from '../../lib/message-read-receipt/ReadReceipt';
import type { IMessageReadsService } from '../../sdk/types/IMessageReadsService';

export class MessageReadsService extends ServiceClassInternal implements IMessageReadsService {
    protected name = 'message-reads';

    async readThread(userId: string, tmid: string): Promise<void> {
        const read = await MessageReads.findOneByUserIdAndThreadId(userId, tmid);

        const threadMessage = await Messages.findOneById(tmid, { projection: { ts: 1, tlm: 1, rid: 1 } });
        if (!threadMessage?.tlm) {
            return;
        }

        await MessageReads.updateReadTimestampByUserIdAndThreadId(userId, tmid);
        await ReadReceipt.storeThreadMessagesReadReceipts(tmid, userId, read?.ls || threadMessage.ts);

        // doesn't mark as read if not all room members have read the thread
        const membersCount = await Subscriptions.countUnarchivedByRoomId(threadMessage.rid);

        if (membersCount <= MAX_ROOM_SIZE_CHECK_INDIVIDUAL_READ_RECEIPTS) {
            const subscriptions = await Subscriptions.findUnarchivedByRoomId(threadMessage.rid, {
                projection: { 'u._id': 1 },
            }).toArray();
            const members = subscriptions.map((s) => s.u._id);

            const totalMessageReads = await MessageReads.countByThreadAndUserIds(tmid, members);
            if (totalMessageReads < membersCount) {
                return;
            }
        } else {
            // for large rooms, mark as read if there are as many reads as room members to improve performance (instead of checking each read)
            const totalMessageReads = await MessageReads.countByThreadId(tmid);
            if (totalMessageReads < membersCount) {
                return;
            }
        }

        const firstRead = await MessageReads.getMinimumLastSeenByThreadId(tmid);
        if (firstRead?.ls) {
            const result = await Messages.setThreadMessagesAsRead(tmid, firstRead.ls);
            if (result.modifiedCount > 0) {
                void api.broadcast('notify.messagesRead', { rid: threadMessage.rid, tmid, until: firstRead.ls });
            }
        }
    }
}