Files
AnnasRechnungsManager/app/routes/api.ausgaben.ts
T
hwinkel d582c748a2 feat: add financial transactions management for companies
- Implemented a new route for managing financial transactions (money) for companies, including creating, editing, and deleting transactions.
- Added a new model `Buchung` to represent transactions with fields for date, account type, transaction type, amount, and description.
- Updated the `companies` model to include a relation to the new `Buchung` model.
- Enhanced the company overview page to link to the new financial transactions page.
- Added migration scripts to create the necessary database tables and fields for the new functionality.
- Created utility scripts for resetting the admin password and setting up the initial admin user.
2026-03-24 19:25:48 +01:00

84 lines
2.7 KiB
TypeScript

import { getApiUser } from "@/session.server";
import prisma from "@/lib/prisma.server";
import { z } from "zod";
import { AusgabeKategorie } from "@prisma/client";
const createSchema = z.object({
companyId: z.string().min(1),
kategorie: z.nativeEnum(AusgabeKategorie),
betrag: z.number().positive(),
steuersatz: z.number().min(0).default(0),
zahlungsart: z.enum(["KASSE", "BANK"]).default("BANK"),
datum: z.string().min(1),
beschreibung: z.string().optional(),
});
export async function loader({ request }: { request: Request }) {
const user = await getApiUser(request);
if (!user) return Response.json({ error: "Unauthorized" }, { status: 401 });
const { searchParams } = new URL(request.url);
const companyId = searchParams.get("companyId");
const year = searchParams.get("year") ? parseInt(searchParams.get("year")!) : null;
if (!companyId) return Response.json({ error: "companyId required" }, { status: 400 });
const company = await prisma.company.findFirst({ where: { id: companyId, userId: user.id } });
if (!company) return Response.json({ error: "Not found" }, { status: 404 });
const ausgaben = await prisma.betriebsausgabe.findMany({
where: {
companyId,
...(year ? {
datum: {
gte: new Date(`${year}-01-01`),
lt: new Date(`${year + 1}-01-01`),
},
} : {}),
},
orderBy: { datum: "desc" },
});
return Response.json(
ausgaben.map((a) => ({
...a,
betrag: Number(a.betrag),
steuersatz: Number(a.steuersatz),
datum: a.datum.toISOString(),
}))
);
}
export async function action({ request }: { request: Request }) {
const user = await getApiUser(request);
if (!user) return Response.json({ error: "Unauthorized" }, { status: 401 });
const body = await request.json();
const parsed = createSchema.safeParse(body);
if (!parsed.success) return Response.json({ error: parsed.error.issues }, { status: 400 });
const company = await prisma.company.findFirst({
where: { id: parsed.data.companyId, userId: user.id },
});
if (!company) return Response.json({ error: "Company not found" }, { status: 404 });
const ausgabe = await prisma.betriebsausgabe.create({
data: {
companyId: parsed.data.companyId,
kategorie: parsed.data.kategorie,
betrag: parsed.data.betrag,
steuersatz: parsed.data.steuersatz,
zahlungsart: parsed.data.zahlungsart,
datum: new Date(parsed.data.datum),
beschreibung: parsed.data.beschreibung,
},
});
return Response.json({
...ausgabe,
betrag: Number(ausgabe.betrag),
steuersatz: Number(ausgabe.steuersatz),
datum: ausgabe.datum.toISOString(),
}, { status: 201 });
}