src/lib/server/paths.ts: zentrale Auflösung der env-vars; vorher 6× IMAGE_DIR und 2× DATABASE_PATH dupliziert mit identischen Defaults. Migrierte Sites: - src/lib/server/db/index.ts (DATABASE_PATH + IMAGE_DIR) - src/routes/api/admin/backup/+server.ts - src/routes/api/domains/+server.ts - src/routes/api/domains/[id]/+server.ts - src/routes/api/recipes/import/+server.ts - src/routes/api/recipes/[id]/image/+server.ts - src/routes/images/[filename]/+server.ts ARCHITECTURE.md: - 49 Flachwitze -> 150 (waren tatsaechlich 150) - 'search/' Route entfernt — wurde nie als eigene Route gebaut, Suche laeuft direkt auf der Homepage via API-Calls Findings aus zweiter Review-Runde (siehe OPEN-ISSUES-NEXT.md)
46 lines
1.6 KiB
TypeScript
46 lines
1.6 KiB
TypeScript
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 });
|
|
}
|
|
};
|