mirror of
https://github.com/vector-im/element-web.git
synced 2024-12-01 15:11:11 +08:00
491f0cd08a
* Copyright headers 1 * Licence headers 2 * Copyright Headers 3 * Copyright Headers 4 * Copyright Headers 5 * Copyright Headers 6 * Copyright headers 7 * Add copyright headers for html and config file * Replace license files and update package.json * Update with CLA * lint
555 lines
22 KiB
TypeScript
555 lines
22 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2023 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
/* See readme.md for tips on writing these tests. */
|
|
|
|
import { test } from ".";
|
|
|
|
test.describe("Read receipts", () => {
|
|
test.describe("redactions", () => {
|
|
test.describe("in threads", () => {
|
|
test("Redacting the threaded message pointed to by my receipt leaves the room read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given I have some threads
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root1",
|
|
msg.threadedOff("Root1", "ThreadMsg1"),
|
|
msg.threadedOff("Root1", "ThreadMsg2"),
|
|
"Root2",
|
|
msg.threadedOff("Root2", "Root2->A"),
|
|
]);
|
|
await util.assertUnread(room2, 2);
|
|
|
|
await util.goTo(room2);
|
|
await util.assertUnreadThread("Root1");
|
|
await util.assertUnreadThread("Root2");
|
|
|
|
// And I have read them
|
|
await util.assertUnreadThread("Root1");
|
|
await util.openThread("Root1");
|
|
await util.assertRead(room2);
|
|
await util.backToThreadsList();
|
|
await util.assertReadThread("Root1");
|
|
|
|
await util.openThread("Root2");
|
|
await util.assertReadThread("Root2");
|
|
await util.closeThreadsPanel();
|
|
await util.goTo(room1);
|
|
|
|
// When the latest message in a thread is redacted
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg2")]);
|
|
|
|
// Then the room and thread are still read
|
|
await util.assertStillRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root1");
|
|
});
|
|
|
|
test("Reading an unread thread after a redaction of the latest message makes it read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given an unread thread where the latest message was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.assertUnreadThread("Root");
|
|
|
|
// When I read the thread
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.closeThreadsPanel();
|
|
await util.goTo(room1);
|
|
|
|
// Then the thread is read
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading an unread thread after a redaction of the latest message makes it read after restart", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message is not counted in the unread count
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.assertUnreadThread("Root");
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.closeThreadsPanel();
|
|
await util.goTo(room1);
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
|
|
// When I restart
|
|
await util.saveAndReload();
|
|
|
|
// Then the room and thread are still read
|
|
await util.assertRead(room2);
|
|
await util.openThreadList();
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading an unread thread after a redaction of an older message makes it read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given an unread thread where an older message was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg1")]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.assertUnreadThread("Root");
|
|
|
|
// When I read the thread
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.closeThreadsPanel();
|
|
await util.goTo(room1);
|
|
|
|
// Then the thread is read
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Marking an unread thread as read after a redaction makes it read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given an unread thread where an older message was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg1")]);
|
|
await util.assertUnread(room2, 1);
|
|
|
|
// When I mark the room as read
|
|
await util.markAsRead(room2);
|
|
await util.assertRead(room2);
|
|
|
|
// Then the thread is read
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Sending and redacting a message after marking the thread as read leaves it read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a thread exists and is marked as read
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.markAsRead(room2);
|
|
await util.assertRead(room2);
|
|
|
|
// When I send and redact a message
|
|
await util.receiveMessages(room2, [msg.threadedOff("Root", "Msg3")]);
|
|
await util.goTo(room2);
|
|
await util.openThreadList();
|
|
await util.assertUnreadThread("Root");
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
|
|
// Then the room and thread are read
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Redacting a message after marking the thread as read leaves it read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a thread exists and is marked as read
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.markAsRead(room2);
|
|
await util.assertRead(room2);
|
|
|
|
// When I redact a message
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg1")]);
|
|
|
|
// Then the room and thread are read
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reacting to a redacted message leaves the thread read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a message in a thread was redacted and everything is read
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.assertRead(room2);
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.backToThreadsList();
|
|
await util.assertReadThread("Root");
|
|
await util.goTo(room1);
|
|
|
|
// When we receive a reaction to the redacted event
|
|
await util.receiveMessages(room2, [msg.reactionTo("Msg2", "z")]);
|
|
|
|
// Then the room is read
|
|
await util.assertStillRead(room2);
|
|
await util.goTo(room2);
|
|
await util.openThreadList();
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Editing a redacted message leaves the thread read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a message in a thread was redacted and everything is read
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.assertRead(room2);
|
|
await util.openThreadList();
|
|
await util.assertUnreadThread("Root");
|
|
await util.openThread("Root");
|
|
await util.assertReadThread("Root");
|
|
await util.goTo(room1);
|
|
|
|
// When we receive an edit of the redacted message
|
|
await util.receiveMessages(room2, [msg.editOf("Msg2", "New Msg2")]);
|
|
|
|
// Then the room is unread
|
|
await util.assertStillRead(room2);
|
|
// and so is the thread
|
|
await util.goTo(room2);
|
|
await util.openThreadList();
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading a thread after a reaction to a redacted message marks the thread as read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message in a thread exists, but someone reacted to it before it was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
msg.reactionTo("Msg3", "x"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
await util.assertUnread(room2, 1);
|
|
|
|
// When we read the thread
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
|
|
// Then the thread (and room) are read
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading a thread containing a redacted, edited message marks the thread as read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message in a thread exists, but someone edited it before it was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
msg.editOf("Msg3", "Msg3 Edited"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
|
|
// When we read the thread
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
|
|
// Then the thread (and room) are read
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading a reply to a redacted message marks the thread as read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message in a thread exists, but someone replied before it was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
msg.replyTo("Msg3", "Msg3Reply"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
|
|
// When we read the thread, creating a receipt that points at the edit
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
|
|
// Then the thread (and room) are read
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("Reading a thread root when its only message has been redacted leaves the room read", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given we had a thread
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, ["Root", msg.threadedOff("Root", "Msg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
|
|
// And then redacted the message that makes it a thread
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg2")]);
|
|
await util.assertUnread(room2, 1);
|
|
|
|
// When we read the main timeline
|
|
await util.goTo(room2);
|
|
|
|
// Then the room is read
|
|
await util.assertRead(room2);
|
|
// and that thread is read
|
|
await util.openThreadList();
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("A thread with a redacted unread is still read after restart", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given I sent and redacted a message in an otherwise-read thread
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "ThreadMsg1"),
|
|
msg.threadedOff("Root", "ThreadMsg2"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
await util.receiveMessages(room2, [msg.threadedOff("Root", "Msg3")]);
|
|
await util.assertRead(room2);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
await util.goTo(room1);
|
|
|
|
// When I restart
|
|
await util.saveAndReload();
|
|
|
|
// Then the room and thread are still read
|
|
await util.assertRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
/*
|
|
* Disabled: this doesn't seem to work as, at some point after syncing from cache, the redaction and redacted
|
|
* event get removed from the thread timeline such that we have no record of the events that the read receipt
|
|
* points to. I suspect this may have been passing by fluke before.
|
|
*/
|
|
test.skip("A thread with a read redaction is still read after restart", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given my receipt points at a redacted thread message
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root1",
|
|
msg.threadedOff("Root1", "ThreadMsg1"),
|
|
msg.threadedOff("Root1", "ThreadMsg2"),
|
|
"Root2",
|
|
msg.threadedOff("Root2", "Root2->A"),
|
|
]);
|
|
await util.assertUnread(room2, 2);
|
|
await util.goTo(room2);
|
|
await util.assertUnreadThread("Root1");
|
|
await util.openThread("Root1");
|
|
await util.assertRead(room2);
|
|
await util.openThread("Root2");
|
|
await util.assertRead(room2);
|
|
await util.closeThreadsPanel();
|
|
await util.goTo(room1);
|
|
await util.assertRead(room2);
|
|
await util.receiveMessages(room2, [msg.redactionOf("ThreadMsg2")]);
|
|
await util.assertStillRead(room2);
|
|
await util.goTo(room2);
|
|
await util.assertReadThread("Root1");
|
|
|
|
// When I restart
|
|
await util.saveAndReload();
|
|
|
|
// Then the room is still read
|
|
await util.assertRead(room2);
|
|
// and so is the thread
|
|
await util.openThreadList();
|
|
await util.assertReadThread("Root1");
|
|
await util.assertReadThread("Root2");
|
|
});
|
|
|
|
test("A thread with an unread reply to a redacted message is still unread after restart", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message in a thread exists, but someone replied before it was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
msg.replyTo("Msg3", "Msg3Reply"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
|
|
// And we have read all this
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
|
|
// When I restart
|
|
await util.saveAndReload();
|
|
|
|
// Then the room is still read
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
|
|
test("A thread with a read reply to a redacted message is still read after restart", async ({
|
|
roomAlpha: room1,
|
|
roomBeta: room2,
|
|
util,
|
|
msg,
|
|
}) => {
|
|
// Given a redacted message in a thread exists, but someone replied before it was redacted
|
|
await util.goTo(room1);
|
|
await util.receiveMessages(room2, [
|
|
"Root",
|
|
msg.threadedOff("Root", "Msg2"),
|
|
msg.threadedOff("Root", "Msg3"),
|
|
msg.replyTo("Msg3", "Msg3Reply"),
|
|
]);
|
|
await util.assertUnread(room2, 1);
|
|
await util.receiveMessages(room2, [msg.redactionOf("Msg3")]);
|
|
|
|
// And I read it, so the room is read
|
|
await util.goTo(room2);
|
|
await util.openThread("Root");
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
|
|
// When I restart
|
|
await util.saveAndReload();
|
|
|
|
// Then the room is still read
|
|
await util.assertRead(room2);
|
|
await util.assertReadThread("Root");
|
|
});
|
|
});
|
|
});
|
|
});
|