perf: hydrate repost events in reblog endpoint & streaming
This commit is contained in:
parent
24efca5ea0
commit
37bee709cd
@ -12,6 +12,7 @@ import { getLnurl } from '@/utils/lnurl.ts';
|
|||||||
import { nip05Cache } from '@/utils/nip05.ts';
|
import { nip05Cache } from '@/utils/nip05.ts';
|
||||||
import { asyncReplaceAll } from '@/utils/text.ts';
|
import { asyncReplaceAll } from '@/utils/text.ts';
|
||||||
import { eventsDB } from '@/storages.ts';
|
import { eventsDB } from '@/storages.ts';
|
||||||
|
import { hydrateEvents } from '@/storages/hydrate.ts';
|
||||||
|
|
||||||
const createStatusSchema = z.object({
|
const createStatusSchema = z.object({
|
||||||
in_reply_to_id: z.string().regex(/[0-9a-f]{64}/).nullish(),
|
in_reply_to_id: z.string().regex(/[0-9a-f]{64}/).nullish(),
|
||||||
@ -212,6 +213,7 @@ const favouritedByController: AppController = (c) => {
|
|||||||
/** https://docs.joinmastodon.org/methods/statuses/#boost */
|
/** https://docs.joinmastodon.org/methods/statuses/#boost */
|
||||||
const reblogStatusController: AppController = async (c) => {
|
const reblogStatusController: AppController = async (c) => {
|
||||||
const eventId = c.req.param('id');
|
const eventId = c.req.param('id');
|
||||||
|
const { signal } = c.req.raw;
|
||||||
|
|
||||||
const event = await getEvent(eventId, {
|
const event = await getEvent(eventId, {
|
||||||
kind: 1,
|
kind: 1,
|
||||||
@ -226,7 +228,14 @@ const reblogStatusController: AppController = async (c) => {
|
|||||||
tags: [['e', event.id], ['p', event.pubkey]],
|
tags: [['e', event.id], ['p', event.pubkey]],
|
||||||
}, c);
|
}, c);
|
||||||
|
|
||||||
const status = await renderReblog(reblogEvent, { loadOriginalPostEvent: true });
|
await hydrateEvents({
|
||||||
|
events: [reblogEvent, event],
|
||||||
|
relations: ['repost', 'author'],
|
||||||
|
storage: eventsDB,
|
||||||
|
signal: signal,
|
||||||
|
});
|
||||||
|
|
||||||
|
const status = await renderReblog(reblogEvent);
|
||||||
|
|
||||||
return c.json(status);
|
return c.json(status);
|
||||||
};
|
};
|
||||||
|
@ -6,6 +6,8 @@ import { getFeedPubkeys } from '@/queries.ts';
|
|||||||
import { Sub } from '@/subs.ts';
|
import { Sub } from '@/subs.ts';
|
||||||
import { bech32ToPubkey } from '@/utils.ts';
|
import { bech32ToPubkey } from '@/utils.ts';
|
||||||
import { renderReblog, renderStatus } from '@/views/mastodon/statuses.ts';
|
import { renderReblog, renderStatus } from '@/views/mastodon/statuses.ts';
|
||||||
|
import { hydrateEvents } from '@/storages/hydrate.ts';
|
||||||
|
import { eventsDB } from '@/storages.ts';
|
||||||
|
|
||||||
const debug = Debug('ditto:streaming');
|
const debug = Debug('ditto:streaming');
|
||||||
|
|
||||||
@ -64,7 +66,14 @@ const streamingController: AppController = (c) => {
|
|||||||
if (filter) {
|
if (filter) {
|
||||||
for await (const event of Sub.sub(socket, '1', [filter])) {
|
for await (const event of Sub.sub(socket, '1', [filter])) {
|
||||||
if (event.kind === 6) {
|
if (event.kind === 6) {
|
||||||
const status = await renderReblog(event, { loadOriginalPostEvent: true });
|
await hydrateEvents({
|
||||||
|
events: [event],
|
||||||
|
relations: ['repost', 'author'],
|
||||||
|
storage: eventsDB,
|
||||||
|
signal: AbortSignal.timeout(1000),
|
||||||
|
});
|
||||||
|
|
||||||
|
const status = await renderReblog(event);
|
||||||
if (status) {
|
if (status) {
|
||||||
send('update', status);
|
send('update', status);
|
||||||
}
|
}
|
||||||
|
@ -63,7 +63,7 @@ async function renderStatuses(c: AppContext, filters: NostrFilter[]) {
|
|||||||
|
|
||||||
const statuses = (await Promise.all(events.map((event) => {
|
const statuses = (await Promise.all(events.map((event) => {
|
||||||
if (event.kind === 6) {
|
if (event.kind === 6) {
|
||||||
return renderReblog(event, {});
|
return renderReblog(event);
|
||||||
}
|
}
|
||||||
return renderStatus(event, c.get('pubkey'));
|
return renderStatus(event, c.get('pubkey'));
|
||||||
}))).filter((boolean) => boolean);
|
}))).filter((boolean) => boolean);
|
||||||
|
@ -3,7 +3,6 @@ import { db } from '@/db.ts';
|
|||||||
import { type NostrEvent, type NStore } from '@/deps.ts';
|
import { type NostrEvent, type NStore } from '@/deps.ts';
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
import { type DittoRelation } from '@/interfaces/DittoFilter.ts';
|
import { type DittoRelation } from '@/interfaces/DittoFilter.ts';
|
||||||
import { eventsDB } from '@/storages.ts';
|
|
||||||
|
|
||||||
interface HydrateEventOpts {
|
interface HydrateEventOpts {
|
||||||
events: DittoEvent[];
|
events: DittoEvent[];
|
||||||
@ -35,7 +34,7 @@ async function hydrateEvents(opts: HydrateEventOpts): Promise<DittoEvent[]> {
|
|||||||
await hydrateUsers({ events, storage, signal });
|
await hydrateUsers({ events, storage, signal });
|
||||||
break;
|
break;
|
||||||
case 'repost':
|
case 'repost':
|
||||||
await hydrateRepostEvents(events);
|
await hydrateRepostEvents({ events, storage, signal });
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -115,8 +114,19 @@ async function hydrateEventStats(events: DittoEvent[]): Promise<DittoEvent[]> {
|
|||||||
return events;
|
return events;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function hydrateRepostEvents(events: DittoEvent[]): Promise<DittoEvent[]> {
|
async function hydrateRepostEvents(opts: Omit<HydrateEventOpts, 'relations'>): Promise<DittoEvent[]> {
|
||||||
const results = await eventsDB.query([{ kinds: [1], ids: events.map((event) => event.id) }]);
|
const { events, storage, signal } = opts;
|
||||||
|
const results = await storage.query([{
|
||||||
|
kinds: [1],
|
||||||
|
ids: events.map((event) => {
|
||||||
|
if (event.kind === 6) {
|
||||||
|
const originalPostId = event.tags.find(([name]) => name === 'e')?.[1];
|
||||||
|
if (!originalPostId) return event.id;
|
||||||
|
else return originalPostId;
|
||||||
|
}
|
||||||
|
return event.id;
|
||||||
|
}),
|
||||||
|
}]);
|
||||||
|
|
||||||
for (const event of events) {
|
for (const event of events) {
|
||||||
if (event.kind === 6) {
|
if (event.kind === 6) {
|
||||||
@ -126,6 +136,7 @@ async function hydrateRepostEvents(events: DittoEvent[]): Promise<DittoEvent[]>
|
|||||||
const originalPostEvent = results.find((event) => event.id === originalPostId);
|
const originalPostEvent = results.find((event) => event.id === originalPostId);
|
||||||
if (!originalPostEvent) continue;
|
if (!originalPostEvent) continue;
|
||||||
|
|
||||||
|
await hydrateAuthors({ events: [originalPostEvent], storage: storage, signal: signal });
|
||||||
event.repost = originalPostEvent;
|
event.repost = originalPostEvent;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,7 @@ import { Conf } from '@/config.ts';
|
|||||||
import { nip19 } from '@/deps.ts';
|
import { nip19 } from '@/deps.ts';
|
||||||
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
|
||||||
import { getMediaLinks, parseNoteContent } from '@/note.ts';
|
import { getMediaLinks, parseNoteContent } from '@/note.ts';
|
||||||
import { getAuthor, getEvent } from '@/queries.ts';
|
import { getAuthor } from '@/queries.ts';
|
||||||
import { jsonMediaDataSchema } from '@/schemas/nostr.ts';
|
import { jsonMediaDataSchema } from '@/schemas/nostr.ts';
|
||||||
import { eventsDB } from '@/storages.ts';
|
import { eventsDB } from '@/storages.ts';
|
||||||
import { findReplyTag } from '@/tags.ts';
|
import { findReplyTag } from '@/tags.ts';
|
||||||
@ -100,21 +100,12 @@ async function renderStatus(event: DittoEvent, viewerPubkey?: string) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
type reblogOpts = {
|
async function renderReblog(event: DittoEvent) {
|
||||||
loadOriginalPostEvent?: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
async function renderReblog(event: DittoEvent, opts: reblogOpts) {
|
|
||||||
const { loadOriginalPostEvent } = opts;
|
|
||||||
|
|
||||||
if (!event.author) return;
|
if (!event.author) return;
|
||||||
|
|
||||||
const repostId = event.tags.find(([name]) => name === 'e')?.[1];
|
const repostId = event.tags.find(([name]) => name === 'e')?.[1];
|
||||||
if (!repostId) return;
|
if (!repostId) return;
|
||||||
|
|
||||||
if (loadOriginalPostEvent) {
|
|
||||||
event.repost = await getEvent(repostId, { kind: 1 });
|
|
||||||
}
|
|
||||||
if (!event.repost) return;
|
if (!event.repost) return;
|
||||||
|
|
||||||
const reblog = await renderStatus(event.repost);
|
const reblog = await renderStatus(event.repost);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user