diff --git a/app/home/(user)/(dashboard)/cart/montonio-callback/actions.ts b/app/home/(user)/(dashboard)/cart/montonio-callback/actions.ts new file mode 100644 index 0000000..e48ac85 --- /dev/null +++ b/app/home/(user)/(dashboard)/cart/montonio-callback/actions.ts @@ -0,0 +1,130 @@ +'use server'; + +import jwt from 'jsonwebtoken'; +import { z } from "zod"; +import { MontonioOrderToken } from "@/app/home/(user)/_components/cart/types"; +import { loadCurrentUserAccount } from "@/app/home/(user)/_lib/server/load-user-account"; +import { listProductTypes } from "@lib/data/products"; +import { placeOrder, retrieveCart } from "@lib/data/cart"; +import { createI18nServerInstance } from "~/lib/i18n/i18n.server"; +import { createOrder } from '~/lib/services/order.service'; +import { getOrderedAnalysisElementsIds, sendOrderToMedipost } from '~/lib/services/medipost.service'; + +const ANALYSIS_PACKAGES_TYPE_HANDLE = 'analysis-packages'; +const MONTONIO_PAID_STATUS = 'PAID'; + +const env = () => z + .object({ + emailSender: z + .string({ + required_error: 'EMAIL_SENDER is required', + }) + .min(1), + siteUrl: z + .string({ + required_error: 'NEXT_PUBLIC_SITE_URL is required', + }) + .min(1), + }) + .parse({ + emailSender: process.env.EMAIL_SENDER, + siteUrl: process.env.NEXT_PUBLIC_SITE_URL!, + }); + +const sendEmail = async ({ email, analysisPackageName, personName, partnerLocationName, language }: { email: string, analysisPackageName: string, personName: string, partnerLocationName: string, language: string }) => { + try { + const { renderSynlabAnalysisPackageEmail } = await import('@kit/email-templates'); + const { getMailer } = await import('@kit/mailers'); + + const mailer = await getMailer(); + + const { html, subject } = await renderSynlabAnalysisPackageEmail({ + analysisPackageName, + personName, + partnerLocationName, + language, + }); + + await mailer + .sendEmail({ + from: env().emailSender, + to: email, + subject, + html, + }) + .catch((error) => { + throw new Error(`Failed to send email, message=${error}`); + }); + } catch (error) { + throw new Error(`Failed to send email, message=${error}`); + } +} + +export async function processMontonioCallback(orderToken: string) { + const { language } = await createI18nServerInstance(); + + const secretKey = process.env.MONTONIO_SECRET_KEY as string; + + const decoded = jwt.verify(orderToken, secretKey, { + algorithms: ['HS256'], + }) as MontonioOrderToken; + + if (decoded.paymentStatus !== MONTONIO_PAID_STATUS) { + throw new Error("Payment not successful"); + } + + const account = await loadCurrentUserAccount(); + if (!account) { + throw new Error("Account not found in context"); + } + + try { + const [,, cartId] = decoded.merchantReferenceDisplay.split(':'); + if (!cartId) { + throw new Error("Cart ID not found"); + } + + const cart = await retrieveCart(cartId); + if (!cart) { + throw new Error("Cart not found"); + } + + const medusaOrder = await placeOrder(cartId, { revalidateCacheTags: false }); + const orderedAnalysisElements = await getOrderedAnalysisElementsIds({ medusaOrder }); + await createOrder({ medusaOrder, orderedAnalysisElements }); + + const { productTypes } = await listProductTypes(); + const analysisPackagesType = productTypes.find(({ metadata }) => metadata?.handle === ANALYSIS_PACKAGES_TYPE_HANDLE); + const analysisPackageOrderItem = medusaOrder.items?.find(({ product_type_id }) => product_type_id === analysisPackagesType?.id); + + const orderResult = { + medusaOrderId: medusaOrder.id, + email: medusaOrder.email, + partnerLocationName: analysisPackageOrderItem?.metadata?.partner_location_name as string ?? '', + analysisPackageName: analysisPackageOrderItem?.title ?? '', + orderedAnalysisElements, + }; + + const { medusaOrderId, email, partnerLocationName, analysisPackageName } = orderResult; + const personName = account.name; + + if (email && analysisPackageName) { + try { + await sendEmail({ email, analysisPackageName, personName, partnerLocationName, language }); + } catch (error) { + console.error("Failed to send email", error); + } + } else { + // @TODO send email for separate analyses + console.error("Missing email or analysisPackageName", orderResult); + } + + // Send order to Medipost (no await to avoid blocking) + sendOrderToMedipost({ medusaOrderId, orderedAnalysisElements }); + + return { success: true }; + } catch (error) { + console.error("Failed to place order", error); + throw new Error(`Failed to place order, message=${error}`); + } +} diff --git a/app/home/(user)/(dashboard)/cart/montonio-callback/client-component.tsx b/app/home/(user)/(dashboard)/cart/montonio-callback/client-component.tsx new file mode 100644 index 0000000..f90efa4 --- /dev/null +++ b/app/home/(user)/(dashboard)/cart/montonio-callback/client-component.tsx @@ -0,0 +1,52 @@ +'use client'; + +import { useEffect, useState } from 'react'; +import { useRouter } from 'next/navigation'; +import { processMontonioCallback } from './actions'; + +export default function MontonioCallbackClient({ orderToken, error }: { + orderToken?: string; + error?: string; +}) { + const router = useRouter(); + + const [isProcessing, setIsProcessing] = useState(false); + const [hasProcessed, setHasProcessed] = useState(false); + + useEffect(() => { + if (error) { + console.error(error); + router.push('/home/cart/montonio-callback/error'); + return; + } + + if (!orderToken || hasProcessed || isProcessing) { + return; + } + + const processOrder = async () => { + setIsProcessing(true); + setHasProcessed(true); + + try { + await processMontonioCallback(orderToken); + router.push('/home/order'); + } catch (error) { + console.error("Failed to place order", error); + router.push('/home/cart/montonio-callback/error'); + } finally { + setIsProcessing(false); + } + }; + + processOrder(); + }, [orderToken, error, router, hasProcessed, isProcessing]); + + return ( +