mirror of
https://gitlab.com/soapbox-pub/ditto.git
synced 2025-12-06 11:29:46 +00:00
212 lines
6.4 KiB
TypeScript
212 lines
6.4 KiB
TypeScript
import { NostrEvent, NostrFilter } from '@nostrify/nostrify';
|
|
import Debug from '@soapbox/stickynotes/debug';
|
|
import { type Context, HTTPException } from 'hono';
|
|
import { parseFormData } from 'formdata-helper';
|
|
import { EventTemplate } from 'nostr-tools';
|
|
import * as TypeFest from 'type-fest';
|
|
import { z } from 'zod';
|
|
|
|
import { type AppContext } from '@/app.ts';
|
|
import { Conf } from '@/config.ts';
|
|
import * as pipeline from '@/pipeline.ts';
|
|
import { RelayError } from '@/RelayError.ts';
|
|
import { AdminSigner } from '@/signers/AdminSigner.ts';
|
|
import { APISigner } from '@/signers/APISigner.ts';
|
|
import { Storages } from '@/storages.ts';
|
|
import { nostrNow } from '@/utils.ts';
|
|
|
|
const debug = Debug('ditto:api');
|
|
|
|
/** EventTemplate with defaults. */
|
|
type EventStub = TypeFest.SetOptional<EventTemplate, 'content' | 'created_at' | 'tags'>;
|
|
|
|
/** Publish an event through the pipeline. */
|
|
async function createEvent(t: EventStub, c: AppContext): Promise<NostrEvent> {
|
|
const signer = new APISigner(c);
|
|
|
|
const event = await signer.signEvent({
|
|
content: '',
|
|
created_at: nostrNow(),
|
|
tags: [],
|
|
...t,
|
|
});
|
|
|
|
return publishEvent(event, c);
|
|
}
|
|
|
|
/** Filter for fetching an existing event to update. */
|
|
interface UpdateEventFilter extends NostrFilter {
|
|
kinds: [number];
|
|
limit?: 1;
|
|
}
|
|
|
|
/** Fetch existing event, update it, then publish the new event. */
|
|
async function updateEvent<E extends EventStub>(
|
|
filter: UpdateEventFilter,
|
|
fn: (prev: NostrEvent | undefined) => E,
|
|
c: AppContext,
|
|
): Promise<NostrEvent> {
|
|
const [prev] = await Storages.db.query([filter], { limit: 1, signal: c.req.raw.signal });
|
|
return createEvent(fn(prev), c);
|
|
}
|
|
|
|
/** Fetch existing event, update its tags, then publish the new event. */
|
|
function updateListEvent(
|
|
filter: UpdateEventFilter,
|
|
fn: (tags: string[][]) => string[][],
|
|
c: AppContext,
|
|
): Promise<NostrEvent> {
|
|
return updateEvent(filter, (prev) => ({
|
|
kind: filter.kinds[0],
|
|
content: prev?.content ?? '',
|
|
tags: fn(prev?.tags ?? []),
|
|
}), c);
|
|
}
|
|
|
|
/** Publish an admin event through the pipeline. */
|
|
async function createAdminEvent(t: EventStub, c: AppContext): Promise<NostrEvent> {
|
|
const signer = new AdminSigner();
|
|
|
|
const event = await signer.signEvent({
|
|
content: '',
|
|
created_at: nostrNow(),
|
|
tags: [],
|
|
...t,
|
|
});
|
|
|
|
return publishEvent(event, c);
|
|
}
|
|
|
|
/** Fetch existing event, update its tags, then publish the new admin event. */
|
|
function updateListAdminEvent(
|
|
filter: UpdateEventFilter,
|
|
fn: (tags: string[][]) => string[][],
|
|
c: AppContext,
|
|
): Promise<NostrEvent> {
|
|
return updateAdminEvent(filter, (prev) => ({
|
|
kind: filter.kinds[0],
|
|
content: prev?.content ?? '',
|
|
tags: fn(prev?.tags ?? []),
|
|
}), c);
|
|
}
|
|
|
|
/** Fetch existing event, update it, then publish the new admin event. */
|
|
async function updateAdminEvent<E extends EventStub>(
|
|
filter: UpdateEventFilter,
|
|
fn: (prev: NostrEvent | undefined) => E,
|
|
c: AppContext,
|
|
): Promise<NostrEvent> {
|
|
const [prev] = await Storages.db.query([filter], { limit: 1, signal: c.req.raw.signal });
|
|
return createAdminEvent(fn(prev), c);
|
|
}
|
|
|
|
/** Push the event through the pipeline, rethrowing any RelayError. */
|
|
async function publishEvent(event: NostrEvent, c: AppContext): Promise<NostrEvent> {
|
|
debug('EVENT', event);
|
|
try {
|
|
await pipeline.handleEvent(event, c.req.raw.signal);
|
|
await Storages.client.event(event);
|
|
} catch (e) {
|
|
if (e instanceof RelayError) {
|
|
throw new HTTPException(422, {
|
|
res: c.json({ error: e.message }, 422),
|
|
});
|
|
}
|
|
}
|
|
|
|
return event;
|
|
}
|
|
|
|
/** Parse request body to JSON, depending on the content-type of the request. */
|
|
async function parseBody(req: Request): Promise<unknown> {
|
|
switch (req.headers.get('content-type')?.split(';')[0]) {
|
|
case 'multipart/form-data':
|
|
case 'application/x-www-form-urlencoded':
|
|
return parseFormData(await req.formData());
|
|
case 'application/json':
|
|
return req.json();
|
|
}
|
|
}
|
|
|
|
/** Schema to parse pagination query params. */
|
|
const paginationSchema = z.object({
|
|
since: z.coerce.number().optional().catch(undefined),
|
|
until: z.coerce.number().optional().catch(undefined),
|
|
limit: z.coerce.number().catch(20).transform((value) => Math.min(Math.max(value, 0), 40)),
|
|
});
|
|
|
|
/** Mastodon API pagination query params. */
|
|
type PaginationParams = z.infer<typeof paginationSchema>;
|
|
|
|
/** Build HTTP Link header for Mastodon API pagination. */
|
|
function buildLinkHeader(url: string, events: NostrEvent[]): string | undefined {
|
|
if (events.length <= 1) return;
|
|
const firstEvent = events[0];
|
|
const lastEvent = events[events.length - 1];
|
|
|
|
const { origin } = Conf.url;
|
|
const { pathname, search } = new URL(url);
|
|
const next = new URL(pathname + search, origin);
|
|
const prev = new URL(pathname + search, origin);
|
|
|
|
next.searchParams.set('until', String(lastEvent.created_at));
|
|
prev.searchParams.set('since', String(firstEvent.created_at));
|
|
|
|
return `<${next}>; rel="next", <${prev}>; rel="prev"`;
|
|
}
|
|
|
|
type Entity = { id: string };
|
|
type HeaderRecord = Record<string, string | string[]>;
|
|
|
|
/** Return results with pagination headers. Assumes chronological sorting of events. */
|
|
function paginated(c: AppContext, events: NostrEvent[], entities: (Entity | undefined)[], headers: HeaderRecord = {}) {
|
|
const link = buildLinkHeader(c.req.url, events);
|
|
|
|
if (link) {
|
|
headers.link = link;
|
|
}
|
|
|
|
// Filter out undefined entities.
|
|
const results = entities.filter((entity): entity is Entity => Boolean(entity));
|
|
return c.json(results, 200, headers);
|
|
}
|
|
|
|
/** JSON-LD context. */
|
|
type LDContext = (string | Record<string, string | Record<string, string>>)[];
|
|
|
|
/** Add a basic JSON-LD context to ActivityStreams object, if it doesn't already exist. */
|
|
function maybeAddContext<T>(object: T): T & { '@context': LDContext } {
|
|
return {
|
|
'@context': ['https://www.w3.org/ns/activitystreams'],
|
|
...object,
|
|
};
|
|
}
|
|
|
|
/** Like hono's `c.json()` except returns JSON-LD. */
|
|
function activityJson<T, P extends string>(c: Context<any, P>, object: T) {
|
|
const response = c.json(maybeAddContext(object));
|
|
response.headers.set('content-type', 'application/activity+json; charset=UTF-8');
|
|
return response;
|
|
}
|
|
|
|
/** Rewrite the URL of the request object to use the local domain. */
|
|
function localRequest(c: Context): Request {
|
|
return Object.create(c.req.raw, {
|
|
url: { value: Conf.local(c.req.url) },
|
|
});
|
|
}
|
|
|
|
export {
|
|
activityJson,
|
|
createAdminEvent,
|
|
createEvent,
|
|
type EventStub,
|
|
localRequest,
|
|
paginated,
|
|
type PaginationParams,
|
|
paginationSchema,
|
|
parseBody,
|
|
updateEvent,
|
|
updateListAdminEvent,
|
|
updateListEvent,
|
|
};
|