Bugfixes und Bezahlen

This commit is contained in:
Moritz Utcke
2025-03-11 09:15:22 -03:00
parent 9b50053007
commit 8b489a94cc
13 changed files with 326 additions and 47 deletions

View File

@@ -0,0 +1,150 @@
import { UUidWithPrefix, getAusweisartFromUUID } from "#components/Ausweis/types.js"
import { Enums } from "#lib/client/prisma.js"
import { authorizationHeaders, authorizationMiddleware } from "#lib/middleware/authorization.js"
import { mollieClient } from "#lib/mollie.js"
import { prisma } from "#lib/server/prisma.js"
import { PaymentMethod } from "@mollie/api-client"
import { defineApiRoute, APIError } from "astro-typesafe-api/server"
import { RechnungSchema } from "src/generated/zod/rechnung.js"
import { z } from "zod"
export const PATCH = defineApiRoute({
input: RechnungSchema.pick({
bezahlmethode: true,
ort: true,
plz: true,
strasse: true,
telefon: true,
empfaenger: true,
versand_empfaenger: true,
versand_ort: true,
versand_plz: true,
versand_strasse: true,
versand_zusatzzeile: true,
abweichende_versand_adresse: true
}),
output: z.object({
checkout_url: z.string().optional(),
uid: UUidWithPrefix,
}),
middleware: authorizationMiddleware,
headers: authorizationHeaders,
async fetch(input, context, user) {
// Wir holen uns die Rechnung
const rechnung = await prisma.rechnung.findUnique({
where: {
uid: context.params.uid,
benutzer: {
id: user.id
}
},
include: {
bedarfsausweis_gewerbe: true,
bedarfsausweis_wohnen: true,
geg_nachweis_gewerbe: true,
geg_nachweis_wohnen: true,
verbrauchsausweis_gewerbe: true,
verbrauchsausweis_wohnen: true
}
})
if (!rechnung) {
throw new APIError({
code: "NOT_FOUND",
message: "Rechnung konnte nicht gefunden werden oder gehört einem anderen Benutzer."
})
}
await prisma.rechnung.update({
where: {
id: rechnung.id
},
data: {
bezahlmethode: input.bezahlmethode,
ort: input.ort,
plz: input.plz,
strasse: input.strasse,
telefon: input.telefon,
empfaenger: input.empfaenger,
versand_empfaenger: input.versand_empfaenger,
versand_ort: input.versand_ort,
versand_plz: input.versand_plz,
versand_strasse: input.versand_strasse,
versand_zusatzzeile: input.versand_zusatzzeile,
abweichende_versand_adresse: input.abweichende_versand_adresse,
}
})
if (input.bezahlmethode === Enums.Bezahlmethoden.rechnung) {
return { uid: rechnung.uid }
}
const ausweis = rechnung.verbrauchsausweis_gewerbe || rechnung.verbrauchsausweis_wohnen || rechnung.bedarfsausweis_gewerbe || rechnung.bedarfsausweis_wohnen || rechnung.geg_nachweis_gewerbe || rechnung.geg_nachweis_wohnen;
if (!ausweis) {
throw new APIError({
code: "NOT_FOUND",
message: "Rechnung hat keinen verknüpften Ausweis"
})
}
const ausweisart = getAusweisartFromUUID(ausweis.uid)
if (!ausweisart) {
throw new APIError({
code: "INTERNAL_SERVER_ERROR",
message: "Ausweis ist ungültig."
})
}
const description = getPaymentDescription(ausweisart);
// Wir erstellen eine Mollie Payment Referenz.
const payment = await mollieClient.payments.create({
amount: {
value: rechnung.betrag.toFixed(2),
currency: "EUR",
},
metadata: {
rechnung_uid: rechnung.uid,
},
method: input.bezahlmethode as PaymentMethod,
description,
redirectUrl: `https://ibcornelsen.de/payment/success?a=${ausweis.uid}&r=${rechnung.uid}`,
webhookUrl: `http://ibcornelsen.de/api/webhooks/mollie?uid=${rechnung.uid}`,
});
const checkoutUrl = payment.getCheckoutUrl();
if (!checkoutUrl) {
throw new APIError({
code: "INTERNAL_SERVER_ERROR",
message: "Checkout URL konnte nicht erstellt werden.",
});
}
return {
uid: rechnung.uid,
checkout_url: checkoutUrl,
};
},
})
function getPaymentDescription(ausweisart: Enums.Ausweisart) {
switch(ausweisart) {
case "BedarfsausweisWohnen":
return "Bedarfsausweis Wohngebäude"
case "BedarfsausweisGewerbe":
return "Bedarfsausweis Gewerbegebäude"
case "VerbrauchsausweisGewerbe":
return "Verbrauchsausweis Gewerbegebäude"
case "VerbrauchsausweisWohnen":
return "Verbrauchsausweis Wohngebäude"
case "GEGNachweisBedarfsausweis":
case "GEGNachweisGewerbe":
case "GEGNachweisWohnen":
return "GEG Nachweis"
default:
return ""
}
}

View File

@@ -7,7 +7,7 @@ import {
authorizationHeaders,
authorizationMiddleware,
} from "#lib/middleware/authorization.js";
import { UUidWithPrefix } from "#components/Ausweis/types.js";
import { getAusweisartFromUUID, UUidWithPrefix } from "#components/Ausweis/types.js";
import { getPrismaAusweisAdapter } from "#lib/server/ausweis.js";
import { PRICES, SERVICES } from "#lib/constants.js";
import { Rechnung } from "#lib/client/prisma.js";
@@ -162,7 +162,7 @@ export const PUT = defineApiRoute({
// Wir erstellen eine Mollie Payment Referenz.
const payment = await mollieClient.payments.create({
amount: {
value: "45.00",
value: betrag.toFixed(2),
currency: "EUR",
},
metadata: {
@@ -189,3 +189,5 @@ export const PUT = defineApiRoute({
};
},
});

View File

@@ -12,6 +12,8 @@ export const PATCH = defineApiRoute({
id: true,
benutzer_id: true,
aufnahme_id: true,
updated_at: true,
created_at: true
}),
output: z.void(),
headers: {

View File

@@ -18,7 +18,9 @@ export const PUT = defineApiRoute({
benutzer_id: true,
uid: true,
aufnahme_id: true,
rechnung_id: true
rechnung_id: true,
updated_at: true,
created_at: true
}).merge(z.object({
startdatum: z.coerce.date()
})),

View File

@@ -37,12 +37,36 @@ const aufnahme = await prisma.aufnahme.findUnique({
objekt: true,
bilder: true,
unterlagen: true,
bedarfsausweise_wohnen: true,
verbrauchsausweise_gewerbe: true,
verbrauchsausweise_wohnen: true,
bedarfsausweise_gewerbe: true,
geg_nachweise_gewerbe: true,
geg_nachweise_wohnen: true,
bedarfsausweise_wohnen: {
include: {
rechnung: true
}
},
verbrauchsausweise_gewerbe: {
include: {
rechnung: true
}
},
verbrauchsausweise_wohnen: {
include: {
rechnung: true
}
},
bedarfsausweise_gewerbe: {
include: {
rechnung: true
}
},
geg_nachweise_gewerbe: {
include: {
rechnung: true
}
},
geg_nachweise_wohnen: {
include: {
rechnung: true
}
},
events: true
}
})

View File

@@ -0,0 +1,52 @@
---
import { getAusweisartFromUUID } from "#components/Ausweis/types";
import AusweisLayoutPruefung from "#layouts/AusweisLayoutPruefung.astro";
import { getPrismaAusweisAdapter } from "#lib/server/ausweis";
import { Enums } from "#lib/server/prisma";
import { getCurrentUser } from "#lib/server/user";
import KundendatenModule from "#modules/KundendatenModule.svelte";
import { PaymentStatus } from "@mollie/api-client";
import { AusweisTyp } from "src/generated/enums";
const uid = Astro.url.searchParams.get("uid")
if (!uid) {
return Astro.redirect("/404")
}
const user = await getCurrentUser(Astro)
const adapter = getPrismaAusweisAdapter(uid)
const ausweisart = getAusweisartFromUUID(uid)
if (!user || !adapter || !ausweisart) {
return Astro.redirect("/404")
}
const ausweis = await adapter.findUnique({
where: {
uid,
benutzer_id: user.id
},
include: {
aufnahme: {
include: {
objekt: true,
bilder: true
}
},
rechnung: true
}
})
if (!ausweis) {
return Astro.redirect("/404")
}
if (ausweis.rechnung.status === PaymentStatus.paid) {
return Astro.redirect("/dashboard")
}
---
<AusweisLayoutPruefung title="Energieausweis Bezahlung">
<KundendatenModule {user} {ausweis} objekt={ausweis.aufnahme.objekt} rechnung={ausweis.rechnung} aufnahme={ausweis.aufnahme} bilder={ausweis.aufnahme.bilder} {ausweisart} ausweistyp={AusweisTyp.Standard} aktiveBezahlmethode={Enums.Bezahlmethoden.paypal} client:only></KundendatenModule>
</AusweisLayoutPruefung>

View File

@@ -9,6 +9,7 @@ import { getCurrentUser } from "#lib/server/user";
const user = await getCurrentUser(Astro) || {}
// POST Body
const params = new URLSearchParams(await Astro.request.text());
if (!params.has("ausweis") || !params.has("aufnahme") || !params.has("objekt") || !params.has("bilder") || !params.has("ausweisart")) {