From 3ca77f2bc054092ffe48f6eb25ae6bbbe63176e7 Mon Sep 17 00:00:00 2001 From: Till Faelligen <2353100+S7evinK@users.noreply.github.com> Date: Fri, 17 May 2024 10:26:09 +0200 Subject: [PATCH 1/3] Optimize getting the latest events in each room Signed-off-by: Till Faelligen <2353100+S7evinK@users.noreply.github.com> --- state/event_table.go | 31 ++++++++++++++++++++++++++----- 1 file changed, 26 insertions(+), 5 deletions(-) diff --git a/state/event_table.go b/state/event_table.go index 985185ba..f8cd5a90 100644 --- a/state/event_table.go +++ b/state/event_table.go @@ -312,14 +312,35 @@ func (t *EventTable) UpdateBeforeSnapshotID(txn *sqlx.Tx, eventNID, snapID, repl return err } -// query the latest events in each of the room IDs given, using highestNID as the highest event. +// LatestEventInRooms queries the latest events in each of the room IDs given, using highestNID as the highest event. +// +// The following query does: +// +// 1. Create a list of the passed in roomIDs (`room_ids` CTE) +// 2. Fetches the highest event_nid before or equal to $2 for each room in room_ids (`max_ev_nid` CTE) +// 3. Fetches the latest events for each room using the data provided from room_ids and max_ev_nid (the `evs` LATERAL) func (t *EventTable) LatestEventInRooms(txn *sqlx.Tx, roomIDs []string, highestNID int64) (events []Event, err error) { - // the position (event nid) may be for a random different room, so we need to find the highest nid <= this position for this room err = txn.Select( &events, - `SELECT event_nid, room_id, event_replaces_nid, before_state_snapshot_id, event_type, state_key, event FROM syncv3_events - WHERE event_nid IN (SELECT max(event_nid) FROM syncv3_events WHERE event_nid <= $1 AND room_id = ANY($2) GROUP BY room_id)`, - highestNID, pq.StringArray(roomIDs), + ` +WITH room_ids AS ( + select unnest($1::text[]) AS room_id +), +max_ev_nid AS ( + SELECT * + FROM room_ids, + LATERAL ( + SELECT max(event_nid) FROM syncv3_events e WHERE e.room_id = room_ids.room_id AND event_nid <= $2 + ) AS x) +SELECT evs.* +FROM room_ids, + max_ev_nid, + LATERAL ( + SELECT event_nid, room_id, event_replaces_nid, before_state_snapshot_id, event_type, state_key, event + FROM syncv3_events e + WHERE e.event_nid = max_ev_nid.max AND room_ids.room_id = e.room_id + ) AS evs`, + pq.StringArray(roomIDs), highestNID, ) if err == sql.ErrNoRows { err = nil From 1ae5b5cb3d6d8628d6b85415fbce81ff57ce3c1a Mon Sep 17 00:00:00 2001 From: Till Faelligen <2353100+S7evinK@users.noreply.github.com> Date: Fri, 17 May 2024 10:41:28 +0200 Subject: [PATCH 2/3] Also optimize LatestEventNIDInRooms --- state/event_table.go | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/state/event_table.go b/state/event_table.go index f8cd5a90..de4af2cc 100644 --- a/state/event_table.go +++ b/state/event_table.go @@ -348,13 +348,25 @@ FROM room_ids, return } +// LatestEventNIDInRooms queries the latest events in each of the room IDs given, using highestNID as the highest event. +// +// The following query does: +// +// 1. Create a list of the passed in roomIDs (`room_ids` CTE) +// 2. Fetches the latest eventNIDs for each room using the data provided from room_ids (the `evs` LATERAL) func (t *EventTable) LatestEventNIDInRooms(txn *sqlx.Tx, roomIDs []string, highestNID int64) (roomToNID map[string]int64, err error) { - // the position (event nid) may be for a random different room, so we need to find the highest nid <= this position for this room var events []Event err = txn.Select( &events, - `SELECT event_nid, room_id FROM syncv3_events - WHERE event_nid IN (SELECT max(event_nid) FROM syncv3_events WHERE event_nid <= $1 AND room_id = ANY($2) GROUP BY room_id)`, + ` +WITH room_ids AS ( + select unnest($1) AS room_id +) +SELECT evs.event_nid, room_id +FROM room_ids, + LATERAL ( + SELECT max(event_nid) event_nid FROM syncv3_events e WHERE e.room_id = room_ids.room_id AND event_nid <= $2 + ) AS evs;`, highestNID, pq.StringArray(roomIDs), ) if err == sql.ErrNoRows { From 425ed4efbaea56d4917e8cd8628caf6257da9c04 Mon Sep 17 00:00:00 2001 From: Till Faelligen <2353100+S7evinK@users.noreply.github.com> Date: Fri, 17 May 2024 11:04:59 +0200 Subject: [PATCH 3/3] Fix order of params and tell pq what $1 is --- state/event_table.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/state/event_table.go b/state/event_table.go index de4af2cc..48d81c58 100644 --- a/state/event_table.go +++ b/state/event_table.go @@ -360,14 +360,14 @@ func (t *EventTable) LatestEventNIDInRooms(txn *sqlx.Tx, roomIDs []string, highe &events, ` WITH room_ids AS ( - select unnest($1) AS room_id + select unnest($1::text[]) AS room_id ) SELECT evs.event_nid, room_id FROM room_ids, LATERAL ( SELECT max(event_nid) event_nid FROM syncv3_events e WHERE e.room_id = room_ids.room_id AND event_nid <= $2 - ) AS evs;`, - highestNID, pq.StringArray(roomIDs), + ) AS evs WHERE event_nid IS NOT NULL;`, + pq.StringArray(roomIDs), highestNID, ) if err == sql.ErrNoRows { err = nil