From 825b2a003eff369d5eee9b5f5a965e831e2d73cf Mon Sep 17 00:00:00 2001 From: devin ivy Date: Tue, 24 Oct 2023 13:07:44 -0400 Subject: [PATCH] Miscellaneous query optimizations on appview (#1766) * split-up record fetching for notifs * simplify query for author posts with media * avoid hitting actor_handle_tgrm_idx when fetching actor by handle --- .../src/api/app/bsky/feed/getAuthorFeed.ts | 4 +- .../bsky/notification/listNotifications.ts | 55 +++++++++++-------- packages/bsky/src/services/actor/index.ts | 8 ++- 3 files changed, 42 insertions(+), 25 deletions(-) diff --git a/packages/bsky/src/api/app/bsky/feed/getAuthorFeed.ts b/packages/bsky/src/api/app/bsky/feed/getAuthorFeed.ts index 9f25eb131e1..c71ddd45791 100644 --- a/packages/bsky/src/api/app/bsky/feed/getAuthorFeed.ts +++ b/packages/bsky/src/api/app/bsky/feed/getAuthorFeed.ts @@ -89,8 +89,8 @@ export const skeleton = async ( if (filter === 'posts_with_media') { feedItemsQb = feedItemsQb - // and only your own posts/reposts - .where('post.creator', '=', actorDid) + // only your own posts + .where('type', '=', 'post') // only posts with media .whereExists((qb) => qb diff --git a/packages/bsky/src/api/app/bsky/notification/listNotifications.ts b/packages/bsky/src/api/app/bsky/notification/listNotifications.ts index 7bcc88f12d3..0013d13a7b0 100644 --- a/packages/bsky/src/api/app/bsky/notification/listNotifications.ts +++ b/packages/bsky/src/api/app/bsky/notification/listNotifications.ts @@ -53,10 +53,6 @@ const skeleton = async ( } let notifBuilder = db.db .selectFrom('notification as notif') - .innerJoin('record', 'record.uri', 'notif.recordUri') - .innerJoin('actor as author', 'author.did', 'notif.author') - .where(notSoftDeletedClause(ref('record'))) - .where(notSoftDeletedClause(ref('author'))) .where('notif.did', '=', viewer) .where((clause) => clause @@ -69,16 +65,12 @@ const skeleton = async ( ), ) .select([ + 'notif.author as authorDid', 'notif.recordUri as uri', 'notif.recordCid as cid', - 'author.did as authorDid', - 'author.handle as authorHandle', - 'author.indexedAt as authorIndexedAt', - 'author.takedownId as authorTakedownId', 'notif.reason as reason', 'notif.reasonSubject as reasonSubject', 'notif.sortAt as indexedAt', - 'record.json as recordJson', ]) const keyset = new NotifsKeyset(ref('notif.sortAt'), ref('notif.recordCid')) @@ -86,6 +78,7 @@ const skeleton = async ( cursor, limit, keyset, + tryIndex: true, }) const actorStateQuery = db.db @@ -107,17 +100,18 @@ const skeleton = async ( } const hydration = async (state: SkeletonState, ctx: Context) => { - const { graphService, actorService, labelService } = ctx + const { graphService, actorService, labelService, db } = ctx const { params, notifs } = state const { viewer } = params const dids = notifs.map((notif) => notif.authorDid) const uris = notifs.map((notif) => notif.uri) - const [actors, labels, bam] = await Promise.all([ + const [actors, records, labels, bam] = await Promise.all([ actorService.views.profiles(dids, viewer), + getRecordMap(db, uris), labelService.getLabelsForUris(uris), graphService.getBlockAndMuteState(dids.map((did) => [viewer, did])), ]) - return { ...state, actors, labels, bam } + return { ...state, actors, records, labels, bam } } const noBlockOrMutes = (state: HydrationState) => { @@ -131,11 +125,11 @@ const noBlockOrMutes = (state: HydrationState) => { } const presentation = (state: HydrationState) => { - const { notifs, cursor, actors, labels, lastSeenNotifs } = state + const { notifs, cursor, actors, records, labels, lastSeenNotifs } = state const notifications = mapDefined(notifs, (notif) => { const author = actors[notif.authorDid] - if (!author) return undefined - const record = jsonStringToLex(notif.recordJson) as Record + const record = records[notif.uri] + if (!author || !record) return undefined const recordLabels = labels[notif.uri] ?? [] const recordSelfLabels = getSelfLabels({ uri: notif.uri, @@ -157,6 +151,24 @@ const presentation = (state: HydrationState) => { return { notifications, cursor } } +const getRecordMap = async ( + db: Database, + uris: string[], +): Promise => { + if (!uris.length) return {} + const { ref } = db.db.dynamic + const recordRows = await db.db + .selectFrom('record') + .select(['uri', 'json']) + .where('uri', 'in', uris) + .where(notSoftDeletedClause(ref('record'))) + .execute() + return recordRows.reduce((acc, { uri, json }) => { + acc[uri] = jsonStringToLex(json) as Record + return acc + }, {} as RecordMap) +} + type Context = { db: Database actorService: ActorService @@ -178,20 +190,19 @@ type SkeletonState = { type HydrationState = SkeletonState & { bam: BlockAndMuteState actors: ActorInfoMap + records: RecordMap labels: Labels } +type RecordMap = { [uri: string]: Record } + type NotifRow = { - indexedAt: string - cid: string - uri: string authorDid: string - authorHandle: string | null - authorIndexedAt: string - authorTakedownId: number | null + uri: string + cid: string reason: string reasonSubject: string | null - recordJson: string + indexedAt: string } class NotifsKeyset extends TimeCidKeyset { diff --git a/packages/bsky/src/services/actor/index.ts b/packages/bsky/src/services/actor/index.ts index 0f90e550f3c..a2f980ce71d 100644 --- a/packages/bsky/src/services/actor/index.ts +++ b/packages/bsky/src/services/actor/index.ts @@ -66,7 +66,13 @@ export class ActorService { qb = qb.orWhere('actor.did', 'in', dids) } if (handles.length) { - qb = qb.orWhere('actor.handle', 'in', handles) + qb = qb.orWhere( + 'actor.handle', + 'in', + handles.length === 1 + ? [handles[0], handles[0]] // a silly (but worthwhile) optimization to avoid usage of actor_handle_tgrm_idx + : handles, + ) } return qb })