Files
kochwas/src/routes/api/domains/+server.ts

46 lines
1.6 KiB
TypeScript
Raw Normal View History

import type { RequestHandler } from './$types';
import { json, error, isHttpError } from '@sveltejs/kit';
import { z } from 'zod';
import { getDb } from '$lib/server/db';
import { validateBody } from '$lib/server/api-helpers';
import { addDomain, listDomains, setDomainFavicon } from '$lib/server/domains/repository';
import { ensureFavicons, fetchAndStoreFavicon } from '$lib/server/domains/favicons';
import { IMAGE_DIR } from '$lib/server/paths';
const CreateSchema = z.object({
domain: z.string().min(3).max(253),
display_name: z.string().max(100).nullable().optional(),
added_by_profile_id: z.number().int().positive().nullable().optional()
});
export const GET: RequestHandler = async () => {
const db = getDb();
// Favicons lazy nachziehen — beim zweiten Aufruf gibt es nichts mehr zu tun.
await ensureFavicons(db, IMAGE_DIR);
return json(listDomains(db));
};
export const POST: RequestHandler = async ({ request }) => {
const data = validateBody(await request.json().catch(() => null), CreateSchema);
try {
const db = getDb();
const d = addDomain(
db,
data.domain,
data.display_name ?? null,
data.added_by_profile_id ?? null
);
// Favicon direkt nach dem Insert mitziehen, damit die Antwort schon das
// Icon enthält — der POST ist eh ein interaktiver Admin-Vorgang.
const favicon = await fetchAndStoreFavicon(d.domain, IMAGE_DIR);
if (favicon) {
setDomainFavicon(db, d.id, favicon);
d.favicon_path = favicon;
}
return json(d, { status: 201 });
} catch (e) {
if (isHttpError(e)) throw e;
error(409, { message: (e as Error).message });
}
};