Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix edge cases around non-thread relations to thread roots and read receipts #3607

Merged
merged 8 commits into from
Jul 19, 2023
133 changes: 130 additions & 3 deletions spec/unit/read-receipt.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ describe("Read receipt", () => {
beforeEach(() => {
httpBackend = new MockHttpBackend();
client = new MatrixClient({
userId: "@user:server",
baseUrl: "https://my.home.server",
accessToken: "my.access.token",
fetchFn: httpBackend.fetchFn as typeof global.fetch,
Expand All @@ -81,7 +82,7 @@ describe("Read receipt", () => {
});

describe("sendReceipt", () => {
it("sends a thread read receipt", async () => {
it("sends a thread read receipt (deprecated)", async () => {
httpBackend
.when(
"POST",
Expand All @@ -102,7 +103,28 @@ describe("Read receipt", () => {
await flushPromises();
});

it("sends an unthreaded receipt", async () => {
it("sends a thread read receipt", async () => {
httpBackend
.when(
"POST",
encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: ROOM_ID,
$receiptType: ReceiptType.Read,
$eventId: threadEvent.getId()!,
}),
)
.check((request) => {
expect(request.data.thread_id).toEqual(THREAD_ID);
})
.respond(200, {});

client.sendReceipt(threadEvent, ReceiptType.Read, THREAD_ID);

await httpBackend.flushAllExpected();
await flushPromises();
});

it("sends an unthreaded receipt (deprecated)", async () => {
httpBackend
.when(
"POST",
Expand All @@ -123,7 +145,91 @@ describe("Read receipt", () => {
await flushPromises();
});

it("sends a room read receipt", async () => {
it("sends an unthreaded receipt", async () => {
httpBackend
.when(
"POST",
encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: ROOM_ID,
$receiptType: ReceiptType.Read,
$eventId: threadEvent.getId()!,
}),
)
.check((request) => {
expect(request.data.thread_id).toBeUndefined();
})
.respond(200, {});

client.sendReadReceipt(threadEvent, ReceiptType.Read, null);

await httpBackend.flushAllExpected();
await flushPromises();
});

it("sends an main timeline receipt for a thread root (deprecated)", async () => {
t3chguy marked this conversation as resolved.
Show resolved Hide resolved
const threadRootEvent = utils.mkEvent({
event: true,
type: EventType.RoomMessage,
user: "@bob:matrix.org",
room: ROOM_ID,
content: {
body: "Hello, this is a thread root",
},
});
threadRootEvent.setThreadId(threadRootEvent.getId());

httpBackend
.when(
"POST",
encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: ROOM_ID,
$receiptType: ReceiptType.Read,
$eventId: threadRootEvent.getId()!,
}),
)
.check((request) => {
expect(request.data.thread_id).toBe("main");
})
.respond(200, {});
client.sendReadReceipt(threadRootEvent, ReceiptType.Read, false);

await httpBackend.flushAllExpected();
await flushPromises();
});

it("sends an main timeline receipt for a reaction to a thread root (deprecated)", async () => {
t3chguy marked this conversation as resolved.
Show resolved Hide resolved
t3chguy marked this conversation as resolved.
Show resolved Hide resolved
const threadRootEvent = utils.mkEvent({
event: true,
type: EventType.RoomMessage,
user: "@bob:matrix.org",
room: ROOM_ID,
content: {
body: "Hello, this is a thread root",
},
});
const reactionToThreadRoot = utils.mkReaction(threadRootEvent, client, client.getSafeUserId(), ROOM_ID);
reactionToThreadRoot.setThreadId(threadRootEvent.getId());

httpBackend
.when(
"POST",
encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: ROOM_ID,
$receiptType: ReceiptType.Read,
$eventId: reactionToThreadRoot.getId()!,
}),
)
.check((request) => {
expect(request.data.thread_id).toBe("main");
})
.respond(200, {});
client.sendReadReceipt(reactionToThreadRoot, ReceiptType.Read, false);

await httpBackend.flushAllExpected();
await flushPromises();
});

it("sends a room read receipt (deprecated)", async () => {
httpBackend
.when(
"POST",
Expand All @@ -143,6 +249,27 @@ describe("Read receipt", () => {
await httpBackend.flushAllExpected();
await flushPromises();
});

it("sends a main timeline room read receipt", async () => {
httpBackend
.when(
"POST",
encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: ROOM_ID,
$receiptType: ReceiptType.Read,
$eventId: roomEvent.getId()!,
}),
)
.check((request) => {
expect(request.data.thread_id).toEqual(MAIN_ROOM_TIMELINE);
})
.respond(200, {});

client.sendReceipt(roomEvent, ReceiptType.Read, "main");

await httpBackend.flushAllExpected();
await flushPromises();
});
});

describe("synthesizeReceipt", () => {
Expand Down
86 changes: 73 additions & 13 deletions src/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4973,43 +4973,84 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
return this.sendMessage(roomId, threadId, content);
}

// Deprecated signature of MatrixClient::sendReceipt allows only specifying unthreaded (boolean) which means it is
// up to the SDK to guess which thread to send the receipt to. This isn't always possible, as for example a relation
// to the thread root shows up in both timelines and can be read separately in each.
// This method guesses which thread to send to, with known issues around thread root relations only sending to
// main timeline.
private guessThreadForReceipt(event: MatrixEvent): string {
// XXX: the spec currently says a threaded read receipt can be sent for the root of a thread,
// but in practice this isn't possible and the spec needs updating.
const isThread =
!!event.threadRootId &&
// A thread cannot be just a thread root and a thread root can only be read in the main timeline
!event.isThreadRoot &&
// Similarly non-thread relations upon the thread root (reactions, edits) should also be for the main timeline.
event.isRelation() &&
(event.isRelation(RelationType.Thread) || event.relationEventId !== event.threadRootId);
return isThread ? event.threadRootId : MAIN_ROOM_TIMELINE;
}

/**
* Send a receipt.
* @param event - The event being acknowledged
* @param receiptType - The kind of receipt e.g. "m.read". Other than
* ReceiptType.Read are experimental!
* @param body - Additional content to send alongside the receipt.
* @param unthreaded - An unthreaded receipt will clear room+thread notifications
* @param thread - the ID of the thread to which the read receipt corresponds,
* "main" for main timeline, `null` for unthreaded (all timelines).
* @returns Promise which resolves: to an empty object `{}`
* @returns Rejects: with an error response.
*/
public async sendReceipt(
event: MatrixEvent,
receiptType: ReceiptType,
thread: "main" | string | null,
body?: Record<string, any>,
unthreaded = false,
): Promise<{}>;
/**
* @deprecated backwards compatibility signature
*/
public async sendReceipt(
event: MatrixEvent,
receiptType: ReceiptType,
body?: Record<string, any>,
unthreaded?: boolean,
): Promise<{}>;
public async sendReceipt(
event: MatrixEvent,
receiptType: ReceiptType,
bodyOrThread?: Record<string, any> | string | null,
unthreadedOrBody?: boolean | Record<string, any>,
): Promise<{}> {
if (this.isGuest()) {
return Promise.resolve({}); // guests cannot send receipts so don't bother.
}

let thread: string | null;
let body: Record<string, any>;

if ((bodyOrThread !== null && typeof bodyOrThread === "object") || typeof unthreadedOrBody === "boolean") {
// Backwards compatible signature case
thread = !unthreadedOrBody ? this.guessThreadForReceipt(event) : null;
body = (bodyOrThread as Record<string, any>) ?? {};
} else {
thread = bodyOrThread as string | null;
andybalaam marked this conversation as resolved.
Show resolved Hide resolved
body = unthreadedOrBody ?? {};
}

const path = utils.encodeUri("/rooms/$roomId/receipt/$receiptType/$eventId", {
$roomId: event.getRoomId()!,
$receiptType: receiptType,
$eventId: event.getId()!,
});

if (!unthreaded) {
// A thread cannot be just a thread root and a thread root can only be read in the main timeline
const isThread = !!event.threadRootId && !event.isThreadRoot;
body = {
...body,
// Only thread replies should define a specific thread. Thread roots can only be read in the main timeline.
thread_id: isThread ? event.threadRootId : MAIN_ROOM_TIMELINE,
};
if (thread !== null) {
// Only thread replies should define a specific thread. Thread roots can only be read in the main timeline.
body.thread_id = thread;
}

const promise = this.http.authedRequest<{}>(Method.Post, path, undefined, body || {});
const promise = this.http.authedRequest<{}>(Method.Post, path, undefined, body);

const room = this.getRoom(event.getRoomId());
if (room && this.credentials.userId) {
Expand All @@ -5022,13 +5063,28 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
* Send a read receipt.
* @param event - The event that has been read.
* @param receiptType - other than ReceiptType.Read are experimental! Optional.
* @param thread - the ID of the thread to which the read receipt corresponds,
* "main" for main timeline, `null` for unthreaded (all timelines).
* @returns Promise which resolves: to an empty object `{}`
* @returns Rejects: with an error response.
*/
public async sendReadReceipt(
event: MatrixEvent,
receiptType: ReceiptType,
thread: "main" | string | null,
): Promise<{}>;
/**
* @deprecated backwards compatibility signature
*/
public async sendReadReceipt(
event: MatrixEvent | null,
receiptType?: ReceiptType,
unthreaded?: boolean,
): Promise<{} | undefined>;
public async sendReadReceipt(
event: MatrixEvent | null,
receiptType = ReceiptType.Read,
unthreaded = false,
threadOrUnthreaded: boolean | string | null = false,
): Promise<{} | undefined> {
if (!event) return;
const eventId = event.getId()!;
Expand All @@ -5037,7 +5093,11 @@ export class MatrixClient extends TypedEventEmitter<EmittedEvents, ClientEventHa
throw new Error(`Cannot set read receipt to a pending event (${eventId})`);
}

return this.sendReceipt(event, receiptType, {}, unthreaded);
if (typeof threadOrUnthreaded === "boolean") {
// Backwards compatible case
return this.sendReceipt(event, receiptType, {}, threadOrUnthreaded);
}
return this.sendReceipt(event, receiptType, threadOrUnthreaded);
}

/**
Expand Down