import { NextResponse } from "next/server"; import { requireSessionUser } from "@/lib/server/session"; import { createEntry, listEntries, requireActiveGroup } from "@/lib/server/entries"; 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 GET() { const { requestId } = await getRequestMeta(); try { const user = await requireSessionUser(); const groupId = await requireActiveGroup(user.id); const entries = await listEntries(groupId); return NextResponse.json({ requestId, request_id: requestId, entries }); } catch (e) { const { status, body } = toErrorResponse(e, "GET /api/entries", requestId); return NextResponse.json(body, { status }); } } export async function POST(req: Request) { const { requestId } = await getRequestMeta(); try { const user = await requireSessionUser(); const groupId = await requireActiveGroup(user.id); const body = await req.json().catch(() => null); const amountDollars = Number(body?.amountDollars || 0); const occurredAt = String(body?.occurredAt || ""); const necessity = String(body?.necessity || ""); const tags = parseTags(body?.tags); const purchaseType = String(body?.purchaseType || tags[0] || "General").trim(); const notes = String(body?.notes || "").trim(); const entryType = String(body?.entryType || "SPENDING").toUpperCase(); const isRecurring = Boolean(body?.isRecurring); const frequency = body?.frequency ? String(body.frequency).toUpperCase() : null; const intervalCount = Number(body?.intervalCount || 1); const endCondition = body?.endCondition ? String(body.endCondition).toUpperCase() : null; const endCount = body?.endCount != null ? Number(body.endCount) : null; const endDate = body?.endDate ? String(body.endDate) : null; const nextRunAt = body?.nextRunAt ? String(body.nextRunAt) : (isRecurring ? occurredAt : null); const bucketId = body?.bucketId != null ? Number(body.bucketId) : null; if (!Number.isFinite(amountDollars) || amountDollars <= 0) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_AMOUNT", message: "Invalid amount" } }, { status: 400 }); if (!occurredAt) return NextResponse.json({ requestId, request_id: requestId, error: { code: "MISSING_OCCURRED_AT", message: "occurredAt is required" } }, { status: 400 }); if (!purchaseType) return NextResponse.json({ requestId, request_id: requestId, error: { code: "MISSING_PURCHASE_TYPE", message: "purchaseType is required" } }, { 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 (!['SPENDING', 'INCOME'].includes(entryType)) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_ENTRY_TYPE", message: "Invalid entryType" } }, { status: 400 }); if (!Number.isFinite(intervalCount) || intervalCount <= 0) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_INTERVAL", message: "Invalid intervalCount" } }, { status: 400 }); if (frequency && !['DAILY', 'WEEKLY', 'BIWEEKLY', 'MONTHLY', 'QUARTERLY', 'YEARLY'].includes(frequency)) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_FREQUENCY", message: "Invalid frequency" } }, { status: 400 }); if (endCondition && !['NEVER', 'AFTER_COUNT', 'BY_DATE'].includes(endCondition)) return NextResponse.json({ requestId, request_id: requestId, error: { code: "INVALID_END_CONDITION", message: "Invalid endCondition" } }, { status: 400 }); const entry = await createEntry({ groupId, userId: user.id, entryType: entryType as "SPENDING" | "INCOME", amountDollars, occurredAt, necessity: necessity as "NECESSARY" | "BOTH" | "UNNECESSARY", purchaseType, notes: notes || undefined, tags, isRecurring, frequency: frequency as "DAILY" | "WEEKLY" | "BIWEEKLY" | "MONTHLY" | "QUARTERLY" | "YEARLY" | null, intervalCount, endCondition: endCondition as "NEVER" | "AFTER_COUNT" | "BY_DATE" | null, endCount, endDate, nextRunAt, bucketId }); return NextResponse.json({ requestId, request_id: requestId, entry }); } catch (e) { const { status, body } = toErrorResponse(e, "POST /api/entries", requestId); return NextResponse.json(body, { status }); } }