wildebeest/functions/ap/users/[id]/outbox.ts

55 wiersze
1.9 KiB
TypeScript

import { parseHandle } from 'wildebeest/backend/src/utils/parse'
import { type Database, getDatabase } from 'wildebeest/backend/src/database'
import { getActorById } from 'wildebeest/backend/src/activitypub/actors'
import { actorURL } from 'wildebeest/backend/src/activitypub/actors'
import type { ContextData } from 'wildebeest/backend/src/types/context'
import type { Env } from 'wildebeest/backend/src/types/env'
export const onRequest: PagesFunction<Env, any, ContextData> = async ({ request, env, params }) => {
const domain = new URL(request.url).hostname
return handleRequest(domain, await getDatabase(env), params.id as string, env.userKEK)
}
const headers = {
'content-type': 'application/json; charset=utf-8',
}
// eslint-disable-next-line @typescript-eslint/no-unused-vars -- TODO: use userKEK
export async function handleRequest(domain: string, db: Database, id: string, userKEK: string): Promise<Response> {
const handle = parseHandle(id)
if (handle.domain !== null) {
return new Response('', { status: 403 })
}
const actorId = actorURL(domain, handle.localPart)
const actor = await getActorById(db, actorId)
if (actor === null) {
return new Response('', { status: 404 })
}
// TODO: eventually move to a shared file
const QUERY = `
SELECT count(*) as count
FROM outbox_objects
INNER JOIN objects ON objects.id = outbox_objects.object_id
WHERE outbox_objects.actor_id = ? AND objects.type = 'Note'
`
const { success, error, results } = await db.prepare(QUERY).bind(actorId.toString()).all()
if (!success) {
throw new Error('SQL error: ' + error)
}
const result: any = (results as any)[0]
const out = {
'@context': 'https://www.w3.org/ns/activitystreams',
id: actor.outbox,
type: 'OrderedCollection',
totalItems: result.count,
first: new URL(actor.outbox + '/page'),
last: new URL(actor.outbox + '/page?min_id=0'),
}
return new Response(JSON.stringify(out), { headers })
}