fiddy/apps/web/app/api/buckets/[id]/route.ts

79 lines
4.1 KiB
TypeScript

import { NextResponse } from "next/server";
import { requireSessionUser } from "@/lib/server/session";
import { deleteBucket, requireActiveGroup, updateBucket } from "@/lib/server/buckets";
import { toErrorResponse } from "@/lib/server/errors";
import { getRequestMeta } from "@/lib/server/request";
function parseTags(value: unknown) {
if (Array.isArray(value)) return value.map(tag => String(tag));
if (typeof value === "string") return value.split(",");
return [] as string[];
}
export async function PATCH(req: Request, { params }: { params: Promise<{ id: string }> }) {
const { requestId } = await getRequestMeta();
try {
const user = await requireSessionUser();
const groupId = await requireActiveGroup(user.id);
const { id: idParam } = await params;
const id = Number(idParam || 0);
if (!id) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_ID", message: "Invalid id" } }, { status: 400 });
const body = await req.json().catch(() => null);
const name = String(body?.name || "").trim();
const description = String(body?.description || "").trim();
const iconKey = body?.iconKey ? String(body.iconKey) : null;
const budgetLimitDollars = body?.budgetLimitDollars != null ? Number(body.budgetLimitDollars) : null;
const position = body?.position != null ? Number(body.position) : 0;
const tags = parseTags(body?.tags);
const necessity = String(body?.necessity || "BOTH").toUpperCase();
const windowDays = body?.windowDays != null ? Number(body.windowDays) : 30;
if (!name)
return NextResponse.json({ requestId, request_id: requestId, error: { code: "MISSING_NAME", message: "name is required" } }, { status: 400 });
if (budgetLimitDollars != null && (!Number.isFinite(budgetLimitDollars) || budgetLimitDollars < 0))
return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_BUDGET", message: "Invalid budgetLimitDollars" } }, { status: 400 });
if (!['NECESSARY', 'BOTH', 'UNNECESSARY'].includes(necessity))
return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_NECESSITY", message: "Invalid necessity" } }, { status: 400 });
if (!Number.isFinite(windowDays) || windowDays < 1 || windowDays > 365)
return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_WINDOW_DAYS", message: "Invalid windowDays" } }, { status: 400 });
const bucket = await updateBucket({
id,
groupId,
userId: user.id,
name,
description: description || undefined,
iconKey,
budgetLimitDollars,
position,
tags,
necessity: necessity as "NECESSARY" | "BOTH" | "UNNECESSARY",
windowDays
});
if (!bucket) return NextResponse.json({ requestId, request_id: requestId, error: { code: "NOT_FOUND", message: "Not found" } }, { status: 404 });
return NextResponse.json({ requestId, request_id: requestId, bucket });
} catch (e) {
const { status, body } = toErrorResponse(e, "PATCH /api/buckets/[id]", requestId);
return NextResponse.json(body, { status });
}
}
export async function DELETE(_: Request, { params }: { params: Promise<{ id: string }> }) {
const { requestId } = await getRequestMeta();
try {
const user = await requireSessionUser();
const groupId = await requireActiveGroup(user.id);
const { id: idParam } = await params;
const id = Number(idParam || 0);
if (!id) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_ID", message: "Invalid id" } }, { status: 400 });
await deleteBucket({ id, groupId, userId: user.id });
return NextResponse.json({ requestId, request_id: requestId, ok: true });
} catch (e) {
const { status, body } = toErrorResponse(e, "DELETE /api/buckets/[id]", requestId);
return NextResponse.json(body, { status });
}
}