develop -> main

develop -> main
This commit is contained in:
danelkungla
2025-09-29 11:11:03 +03:00
committed by GitHub
139 changed files with 13090 additions and 3924 deletions

View File

@@ -43,6 +43,7 @@ MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK=true
#MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK=false #MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK=false
# MEDUSA # MEDUSA
COMPANY_BENEFITS_PAYMENT_SECRET_KEY=NzcwMzE2NmEtOThiMS0xMWYwLWI4NjYtMDMwZDQzMjFhMjExCg==
MEDUSA_BACKEND_URL=http://localhost:9000 MEDUSA_BACKEND_URL=http://localhost:9000
MEDUSA_BACKEND_PUBLIC_URL=http://localhost:9000 MEDUSA_BACKEND_PUBLIC_URL=http://localhost:9000

View File

@@ -1,12 +1,11 @@
import { SIDEBAR_WIDTH_PROPERTY } from '@/packages/ui/src/shadcn/constants';
import { AppLogo } from '@kit/shared/components/app-logo'; import { AppLogo } from '@kit/shared/components/app-logo';
import { ProfileAccountDropdownContainer } from '@kit/shared/components/personal-account-dropdown-container'; import { ProfileAccountDropdownContainer } from '@kit/shared/components/personal-account-dropdown-container';
import { SIDEBAR_WIDTH_PROPERTY } from '@/packages/ui/src/shadcn/constants';
import type { UserWorkspace } from '../../home/(user)/_lib/server/load-user-workspace'; import type { UserWorkspace } from '../../home/(user)/_lib/server/load-user-workspace';
export function AdminMenuNavigation(props: { export function AdminMenuNavigation(props: { workspace: UserWorkspace }) {
workspace: UserWorkspace;
}) {
const { accounts } = props.workspace; const { accounts } = props.workspace;
return ( return (
@@ -17,9 +16,7 @@ export function AdminMenuNavigation(props: {
<div className="flex items-center justify-end gap-3"> <div className="flex items-center justify-end gap-3">
<div> <div>
<ProfileAccountDropdownContainer <ProfileAccountDropdownContainer accounts={accounts} />
accounts={accounts}
/>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -3,8 +3,8 @@ import { cache } from 'react';
import { AdminAccountPage } from '@kit/admin/components/admin-account-page'; import { AdminAccountPage } from '@kit/admin/components/admin-account-page';
import { AdminGuard } from '@kit/admin/components/admin-guard'; import { AdminGuard } from '@kit/admin/components/admin-guard';
import { getAccount } from '~/lib/services/account.service';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import { getAccount } from '~/lib/services/account.service';
interface Params { interface Params {
params: Promise<{ params: Promise<{

View File

@@ -8,6 +8,7 @@ import { getSupabaseServerClient } from '@kit/supabase/server-client';
import { AppBreadcrumbs } from '@kit/ui/app-breadcrumbs'; import { AppBreadcrumbs } from '@kit/ui/app-breadcrumbs';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
import { PageBody, PageHeader } from '@kit/ui/page'; import { PageBody, PageHeader } from '@kit/ui/page';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
interface SearchParams { interface SearchParams {

View File

@@ -5,8 +5,8 @@ import { cookies } from 'next/headers';
import { Page, PageMobileNavigation, PageNavigation } from '@kit/ui/page'; import { Page, PageMobileNavigation, PageNavigation } from '@kit/ui/page';
import { SidebarProvider } from '@kit/ui/shadcn-sidebar'; import { SidebarProvider } from '@kit/ui/shadcn-sidebar';
import { AdminSidebar } from '~/admin/_components/admin-sidebar';
import { AdminMenuNavigation } from '~/admin/_components/admin-menu-navigation'; import { AdminMenuNavigation } from '~/admin/_components/admin-menu-navigation';
import { AdminSidebar } from '~/admin/_components/admin-sidebar';
import { AdminMobileNavigation } from '~/admin/_components/mobile-navigation'; import { AdminMobileNavigation } from '~/admin/_components/mobile-navigation';
import { loadUserWorkspace } from '../home/(user)/_lib/server/load-user-workspace'; import { loadUserWorkspace } from '../home/(user)/_lib/server/load-user-workspace';

View File

@@ -1,6 +1,7 @@
import { AdminDashboard } from '@kit/admin/components/admin-dashboard'; import { AdminDashboard } from '@kit/admin/components/admin-dashboard';
import { AdminGuard } from '@kit/admin/components/admin-guard'; import { AdminGuard } from '@kit/admin/components/admin-guard';
import { PageBody, PageHeader } from '@kit/ui/page'; import { PageBody, PageHeader } from '@kit/ui/page';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
function AdminPage() { function AdminPage() {

View File

@@ -2,10 +2,45 @@ import axios from 'axios';
import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client'; import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client';
import type { ISearchLoadResponse } from '~/lib/types/connected-online'; import { logSyncResult } from '~/lib/services/audit.service';
import { SyncStatus } from '~/lib/types/audit';
import type {
ISearchLoadResponse,
P_JobTitleTranslation,
} from '~/lib/types/connected-online';
function createTranslationMap(translations: P_JobTitleTranslation[]) {
const result: Map<
number,
Map<number, { textEN: string; textRU: string; textET: string }>
> = new Map();
for (const translation of translations) {
const { ClinicID, TextET, TextEN, TextRU, SyncID } = translation;
if (!result.has(ClinicID)) {
result.set(ClinicID, new Map());
}
result.get(ClinicID)!.set(SyncID, {
textET: TextET,
textEN: TextEN,
textRU: TextRU,
});
}
return result;
}
function getSpokenLanguages(spokenLanguages?: string) {
if (!spokenLanguages || !spokenLanguages.length) return [];
return spokenLanguages.split(',');
}
export default async function syncConnectedOnline() { export default async function syncConnectedOnline() {
const isProd = process.env.NODE_ENV === 'production'; const isProd = !['test', 'localhost'].some((pathString) =>
process.env.NEXT_PUBLIC_SITE_URL?.includes(pathString),
);
const baseUrl = process.env.CONNECTED_ONLINE_URL; const baseUrl = process.env.CONNECTED_ONLINE_URL;
@@ -16,14 +51,19 @@ export default async function syncConnectedOnline() {
const supabase = getSupabaseServerAdminClient(); const supabase = getSupabaseServerAdminClient();
try { try {
const response = await axios.post<{ d: string }>(`${baseUrl}/Search_Load`, { const searchLoadResponse = await axios.post<{ d: string }>(
headers: { `${baseUrl}/Search_Load`,
'Content-Type': 'application/json; charset=utf-8', {
headers: {
'Content-Type': 'application/json; charset=utf-8',
},
param: "{'Value':'|et|-1'}", // get all available services in Estonian
}, },
param: "{'Value':'|et|-1'}", // get all available services in Estonian );
});
const responseData: ISearchLoadResponse = JSON.parse(response.data.d); const responseData: ISearchLoadResponse = JSON.parse(
searchLoadResponse.data.d,
);
if (responseData?.ErrorCode !== 0) { if (responseData?.ErrorCode !== 0) {
throw new Error('Failed to get Connected Online data'); throw new Error('Failed to get Connected Online data');
@@ -43,18 +83,23 @@ export default async function syncConnectedOnline() {
let clinics; let clinics;
let services; let services;
let serviceProviders;
let jobTitleTranslations;
// Filter out "Dentas Demo OÜ" in prod or only sync "Dentas Demo OÜ" in any other environment // Filter out "Dentas Demo OÜ" in prod or only sync "Dentas Demo OÜ" in any other environment
if (isProd) { const isDemoClinic = (clinicId: number) =>
clinics = responseData.Data.T_Lic.filter((clinic) => clinic.ID !== 2); isProd ? clinicId !== 2 : clinicId === 2;
services = responseData.Data.T_Service.filter( clinics = responseData.Data.T_Lic.filter(({ ID }) => isDemoClinic(ID));
(service) => service.ClinicID !== 2, services = responseData.Data.T_Service.filter(({ ClinicID }) =>
); isDemoClinic(ClinicID),
} else { );
clinics = responseData.Data.T_Lic.filter((clinic) => clinic.ID === 2); serviceProviders = responseData.Data.T_Doctor.filter(({ ClinicID }) =>
services = responseData.Data.T_Service.filter( isDemoClinic(ClinicID),
(service) => service.ClinicID === 2, );
); jobTitleTranslations = createTranslationMap(
} responseData.Data.P_JobTitleTranslations.filter(({ ClinicID }) =>
isDemoClinic(ClinicID),
),
);
const mappedClinics = clinics.map((clinic) => { const mappedClinics = clinics.map((clinic) => {
return { return {
@@ -64,6 +109,8 @@ export default async function syncConnectedOnline() {
name: clinic.Name, name: clinic.Name,
personal_code_required: !!clinic.PersonalCodeRequired, personal_code_required: !!clinic.PersonalCodeRequired,
phone_number: clinic.Phone || null, phone_number: clinic.Phone || null,
key: clinic.Key,
address: clinic.Address,
}; };
}); });
@@ -71,7 +118,7 @@ export default async function syncConnectedOnline() {
return { return {
id: service.ID, id: service.ID,
clinic_id: service.ClinicID, clinic_id: service.ClinicID,
sync_id: service.SyncID, sync_id: Number(service.SyncID),
name: service.Name, name: service.Name,
description: service.Description || null, description: service.Description || null,
price: service.Price, price: service.Price,
@@ -87,45 +134,133 @@ export default async function syncConnectedOnline() {
}; };
}); });
const mappedServiceProviders = serviceProviders.map((serviceProvider) => {
const jobTitleTranslation = serviceProvider.JobTitleID
? jobTitleTranslations
.get(serviceProvider.ClinicID)
?.get(serviceProvider.JobTitleID)
: null;
return {
id: serviceProvider.ID,
prefix: serviceProvider.Prefix,
name: serviceProvider.Name,
spoken_languages: getSpokenLanguages(serviceProvider.SpokenLanguages),
job_title_et: jobTitleTranslation?.textET,
job_title_en: jobTitleTranslation?.textEN,
job_title_ru: jobTitleTranslation?.textRU,
job_title_id: serviceProvider.JobTitleID,
is_deleted: !!serviceProvider.Deleted,
clinic_id: serviceProvider.ClinicID,
};
});
const { error: providersError } = await supabase const { error: providersError } = await supabase
.schema('medreport') .schema('medreport')
.from('connected_online_providers') .from('connected_online_providers')
.upsert(mappedClinics); .upsert(mappedClinics);
if (providersError) {
return logSyncResult({
operation: 'CONNECTED_ONLINE_SYNC',
comment:
'Error saving connected online providers: ' +
JSON.stringify(providersError),
status: SyncStatus.Fail,
changed_by_role: 'service_role',
});
}
const { error: servicesError } = await supabase const { error: servicesError } = await supabase
.schema('medreport') .schema('medreport')
.from('connected_online_services') .from('connected_online_services')
.upsert(mappedServices, { onConflict: 'id', ignoreDuplicates: false }); .upsert(mappedServices, {
onConflict: 'id',
ignoreDuplicates: false,
});
if (providersError || servicesError) { if (servicesError) {
return supabase return logSyncResult({
.schema('audit') operation: 'CONNECTED_ONLINE_SYNC',
.from('sync_entries') comment:
.insert({ 'Error saving connected online services: ' +
operation: 'CONNECTED_ONLINE_SYNC', JSON.stringify(servicesError),
comment: providersError status: SyncStatus.Fail,
? 'Error saving providers: ' + JSON.stringify(providersError) changed_by_role: 'service_role',
: 'Error saving services: ' + JSON.stringify(servicesError), });
status: 'FAIL',
changed_by_role: 'service_role',
});
} }
await supabase.schema('audit').from('sync_entries').insert({ const { error: serviceProvidersError } = await supabase
.schema('medreport')
.from('connected_online_service_providers')
.upsert(mappedServiceProviders, {
onConflict: 'id',
ignoreDuplicates: false,
});
if (serviceProvidersError) {
return logSyncResult({
operation: 'CONNECTED_ONLINE_SYNC',
comment:
'Error saving service providers: ' +
JSON.stringify(serviceProvidersError),
status: SyncStatus.Fail,
changed_by_role: 'service_role',
});
}
for (const mappedClinic of mappedClinics) {
const defaultLoadResponse = await axios.post<{ d: string }>(
`${baseUrl}/Default_Load`,
{
headers: {
'Content-Type': 'application/json; charset=utf-8',
},
param: `{'Value':'${mappedClinic.key}|et'}`,
},
);
const defaultLoadResponseData = JSON.parse(defaultLoadResponse.data.d);
if (defaultLoadResponseData?.ErrorCode !== 0) {
throw new Error('Failed to get Connected Online location data');
}
const clinicLocations: {
SyncID: number;
Address: string;
Name: string;
}[] = defaultLoadResponseData.Data.T_SelectableLocation;
if (clinicLocations?.length) {
const mappedLocations = clinicLocations.map(
({ SyncID, Address, Name }) => ({
address: Address,
clinic_id: mappedClinic.id,
sync_id: SyncID,
name: Name,
}),
);
await supabase
.schema('medreport')
.from('connected_online_locations')
.insert(mappedLocations)
.throwOnError();
}
}
await logSyncResult({
operation: 'CONNECTED_ONLINE_SYNC', operation: 'CONNECTED_ONLINE_SYNC',
status: 'SUCCESS', status: SyncStatus.Success,
changed_by_role: 'service_role', changed_by_role: 'service_role',
}); });
} catch (e) { } catch (e) {
await supabase await logSyncResult({
.schema('audit') operation: 'CONNECTED_ONLINE_SYNC',
.from('sync_entries') status: SyncStatus.Fail,
.insert({ comment: JSON.stringify(e),
operation: 'CONNECTED_ONLINE_SYNC', changed_by_role: 'service_role',
status: 'FAIL', });
comment: JSON.stringify(e),
changed_by_role: 'service_role',
});
throw new Error( throw new Error(
`Failed to sync Connected Online data, error: ${JSON.stringify(e)}`, `Failed to sync Connected Online data, error: ${JSON.stringify(e)}`,
); );

View File

@@ -1,12 +1,20 @@
import { redirect } from 'next/navigation';
import { HomeLayoutPageHeader } from '@/app/home/(user)/_components/home-page-header'; import { HomeLayoutPageHeader } from '@/app/home/(user)/_components/home-page-header';
import { loadCategory } from '@/app/home/(user)/_lib/server/load-category'; import { loadCategory } from '@/app/home/(user)/_lib/server/load-category';
import { pathsConfig } from '@kit/shared/config';
import { AppBreadcrumbs } from '@kit/ui/makerkit/app-breadcrumbs'; import { AppBreadcrumbs } from '@kit/ui/makerkit/app-breadcrumbs';
import { PageBody } from '@kit/ui/page';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import BookingContainer from '~/home/(user)/_components/booking/booking-container';
import { loadCurrentUserAccount } from '~/home/(user)/_lib/server/load-user-account';
import { createI18nServerInstance } from '~/lib/i18n/i18n.server'; import { createI18nServerInstance } from '~/lib/i18n/i18n.server';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import {
PageViewAction,
createPageViewLog,
} from '~/lib/services/audit/pageView.service';
export const generateMetadata = async () => { export const generateMetadata = async () => {
const i18n = await createI18nServerInstance(); const i18n = await createI18nServerInstance();
@@ -17,9 +25,30 @@ export const generateMetadata = async () => {
}; };
}; };
async function BookingHandlePage({ params }: { params: { handle: string } }) { async function BookingHandlePage({
const handle = await params.handle; params,
}: {
params: Promise<{ handle: string }>;
}) {
const { handle } = await params;
const { category } = await loadCategory({ handle }); const { category } = await loadCategory({ handle });
const { account } = await loadCurrentUserAccount();
if (!category) {
return <div>Category not found</div>;
}
if (!account) {
return redirect(pathsConfig.auth.signIn);
}
await createPageViewLog({
accountId: account.id,
action: PageViewAction.VIEW_TTO_SERVICE_BOOKING,
extraData: {
handle,
},
});
return ( return (
<> <>
@@ -30,10 +59,10 @@ async function BookingHandlePage({ params }: { params: { handle: string } }) {
/> />
<HomeLayoutPageHeader <HomeLayoutPageHeader
title={<Trans i18nKey={'booking:title'} />} title={<Trans i18nKey={'booking:title'} />}
description={<Trans i18nKey={'booking:description'} />} description=""
/> />
<PageBody></PageBody> <BookingContainer category={category} />
</> </>
); );
} }

View File

@@ -2,100 +2,13 @@
import { MontonioOrderToken } from '@/app/home/(user)/_components/cart/types'; import { MontonioOrderToken } from '@/app/home/(user)/_components/cart/types';
import { loadCurrentUserAccount } from '@/app/home/(user)/_lib/server/load-user-account'; import { loadCurrentUserAccount } from '@/app/home/(user)/_lib/server/load-user-account';
import { placeOrder, retrieveCart } from '@lib/data/cart'; import { retrieveCart } from '@lib/data/cart';
import { listProductTypes } from '@lib/data/products';
import type { StoreOrder } from '@medusajs/types';
import jwt from 'jsonwebtoken'; import jwt from 'jsonwebtoken';
import { z } from 'zod';
import type { AccountWithParams } from '@kit/accounts/types/accounts'; import { handlePlaceOrder } from '../../../_lib/server/cart-actions';
import { createNotificationsApi } from '@kit/notifications/api';
import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client';
import { createI18nServerInstance } from '~/lib/i18n/i18n.server';
import { sendOrderToMedipost } from '~/lib/services/medipost/medipostPrivateMessage.service';
import { getOrderedAnalysisIds } from '~/lib/services/medusaOrder.service';
import {
createAnalysisOrder,
getAnalysisOrder,
} from '~/lib/services/order.service';
const ANALYSIS_PACKAGES_TYPE_HANDLE = 'analysis-packages';
const ANALYSIS_TYPE_HANDLE = 'synlab-analysis';
const MONTONIO_PAID_STATUS = 'PAID'; const MONTONIO_PAID_STATUS = 'PAID';
const env = () =>
z
.object({
emailSender: z
.string({
error: 'EMAIL_SENDER is required',
})
.min(1),
siteUrl: z
.string({
error: 'NEXT_PUBLIC_SITE_URL is required',
})
.min(1),
isEnabledDispatchOnMontonioCallback: z.boolean({
error: 'MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK is required',
}),
})
.parse({
emailSender: process.env.EMAIL_SENDER,
siteUrl: process.env.NEXT_PUBLIC_SITE_URL!,
isEnabledDispatchOnMontonioCallback:
process.env.MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK === 'true',
});
const sendEmail = async ({
account,
email,
analysisPackageName,
partnerLocationName,
language,
}: {
account: Pick<AccountWithParams, 'name' | 'id'>;
email: string;
analysisPackageName: string;
partnerLocationName: string;
language: string;
}) => {
const client = getSupabaseServerAdminClient();
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: account.name,
partnerLocationName,
language,
});
await mailer
.sendEmail({
from: env().emailSender,
to: email,
subject,
html,
})
.catch((error) => {
throw new Error(`Failed to send email, message=${error}`);
});
await createNotificationsApi(client).createNotification({
account_id: account.id,
body: html,
});
} catch (error) {
throw new Error(`Failed to send email, message=${error}`);
}
};
async function decodeOrderToken(orderToken: string) { async function decodeOrderToken(orderToken: string) {
const secretKey = process.env.MONTONIO_SECRET_KEY as string; const secretKey = process.env.MONTONIO_SECRET_KEY as string;
@@ -122,74 +35,6 @@ async function getCartByOrderToken(decoded: MontonioOrderToken) {
return cart; return cart;
} }
async function getOrderResultParameters(medusaOrder: StoreOrder) {
const { productTypes } = await listProductTypes();
const analysisPackagesType = productTypes.find(
({ metadata }) => metadata?.handle === ANALYSIS_PACKAGES_TYPE_HANDLE,
);
const analysisType = productTypes.find(
({ metadata }) => metadata?.handle === ANALYSIS_TYPE_HANDLE,
);
const analysisPackageOrderItem = medusaOrder.items?.find(
({ product_type_id }) => product_type_id === analysisPackagesType?.id,
);
const analysisItems = medusaOrder.items?.filter(
({ product_type_id }) => product_type_id === analysisType?.id,
);
return {
medusaOrderId: medusaOrder.id,
email: medusaOrder.email,
analysisPackageOrder: analysisPackageOrderItem
? {
partnerLocationName:
(analysisPackageOrderItem?.metadata
?.partner_location_name as string) ?? '',
analysisPackageName: analysisPackageOrderItem?.title ?? '',
}
: null,
analysisItemsOrder:
Array.isArray(analysisItems) && analysisItems.length > 0
? analysisItems.map(({ product }) => ({
analysisName: product?.title ?? '',
analysisId: (product?.metadata?.analysisIdOriginal as string) ?? '',
}))
: null,
};
}
async function sendAnalysisPackageOrderEmail({
account,
email,
analysisPackageOrder,
}: {
account: AccountWithParams;
email: string;
analysisPackageOrder: {
partnerLocationName: string;
analysisPackageName: string;
};
}) {
const { language } = await createI18nServerInstance();
const { analysisPackageName, partnerLocationName } = analysisPackageOrder;
try {
await sendEmail({
account: { id: account.id, name: account.name },
email,
analysisPackageName,
partnerLocationName,
language,
});
console.info(`Successfully sent analysis package order email to ${email}`);
} catch (error) {
console.error(
`Failed to send analysis package order email to ${email}`,
error,
);
}
}
export async function processMontonioCallback(orderToken: string) { export async function processMontonioCallback(orderToken: string) {
const { account } = await loadCurrentUserAccount(); const { account } = await loadCurrentUserAccount();
if (!account) { if (!account) {
@@ -199,63 +44,8 @@ export async function processMontonioCallback(orderToken: string) {
try { try {
const decoded = await decodeOrderToken(orderToken); const decoded = await decodeOrderToken(orderToken);
const cart = await getCartByOrderToken(decoded); const cart = await getCartByOrderToken(decoded);
const result = await handlePlaceOrder({ cart });
const medusaOrder = await placeOrder(cart.id, { return result;
revalidateCacheTags: false,
});
const orderedAnalysisElements = await getOrderedAnalysisIds({
medusaOrder,
});
try {
const existingAnalysisOrder = await getAnalysisOrder({
medusaOrderId: medusaOrder.id,
});
console.info(
`Analysis order already exists for medusaOrderId=${medusaOrder.id}, orderId=${existingAnalysisOrder.id}`,
);
return { success: true, orderId: existingAnalysisOrder.id };
} catch {
// ignored
}
const orderId = await createAnalysisOrder({
medusaOrder,
orderedAnalysisElements,
});
const orderResult = await getOrderResultParameters(medusaOrder);
const { medusaOrderId, email, analysisPackageOrder, analysisItemsOrder } =
orderResult;
if (email) {
if (analysisPackageOrder) {
await sendAnalysisPackageOrderEmail({
account,
email,
analysisPackageOrder,
});
} else {
console.info(`Order has no analysis package, skipping email.`);
}
if (analysisItemsOrder) {
// @TODO send email for separate analyses
console.warn(
`Order has analysis items, but no email template exists yet`,
);
} else {
console.info(`Order has no analysis items, skipping email.`);
}
} else {
console.error('Missing email to send order result email', orderResult);
}
if (env().isEnabledDispatchOnMontonioCallback) {
await sendOrderToMedipost({ medusaOrderId, orderedAnalysisElements });
}
return { success: true, orderId };
} catch (error) { } catch (error) {
console.error('Failed to place order', error); console.error('Failed to place order', error);
throw new Error(`Failed to place order, message=${error}`); throw new Error(`Failed to place order, message=${error}`);

View File

@@ -34,8 +34,15 @@ export default function MontonioCallbackClient({
setHasProcessed(true); setHasProcessed(true);
try { try {
const { orderId } = await processMontonioCallback(orderToken); const result = await processMontonioCallback(orderToken);
router.push(`/home/order/${orderId}/confirmed`); if (result.success) {
return router.push(`/home/order/${result.orderId}/confirmed`);
}
if (result.failedServiceBookings?.length) {
router.push(
`/home/cart/montonio-callback/error?reasonFailed=${result.failedServiceBookings.map(({ reason }) => reason).join(',')}`,
);
}
} catch (error) { } catch (error) {
console.error('Failed to place order', error); console.error('Failed to place order', error);
router.push('/home/cart/montonio-callback/error'); router.push('/home/cart/montonio-callback/error');

View File

@@ -1,8 +1,11 @@
import { use } from 'react';
import Link from 'next/link'; import Link from 'next/link';
import { createI18nServerInstance } from '@/lib/i18n/i18n.server'; import { createI18nServerInstance } from '@/lib/i18n/i18n.server';
import { PageBody, PageHeader } from '@/packages/ui/src/makerkit/page'; import { PageBody, PageHeader } from '@/packages/ui/src/makerkit/page';
import { toArray } from '@kit/shared/utils';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
import { Alert, AlertDescription } from '@kit/ui/shadcn/alert'; import { Alert, AlertDescription } from '@kit/ui/shadcn/alert';
import { AlertTitle } from '@kit/ui/shadcn/alert'; import { AlertTitle } from '@kit/ui/shadcn/alert';
@@ -16,7 +19,15 @@ export async function generateMetadata() {
}; };
} }
export default async function MontonioCheckoutCallbackErrorPage() { export default async function MontonioCheckoutCallbackErrorPage({
searchParams,
}: {
searchParams: Promise<{ reasonFailed: string }>;
}) {
const params = await searchParams;
const failedBookingData: string[] = toArray(params.reasonFailed?.split(','));
return ( return (
<div className={'flex h-full flex-1 flex-col'}> <div className={'flex h-full flex-1 flex-col'}>
<PageHeader title={<Trans i18nKey="cart:montonioCallback.title" />} /> <PageHeader title={<Trans i18nKey="cart:montonioCallback.title" />} />
@@ -28,9 +39,15 @@ export default async function MontonioCheckoutCallbackErrorPage() {
</AlertTitle> </AlertTitle>
<AlertDescription> <AlertDescription>
<p> {failedBookingData.length ? (
failedBookingData.map((failureReason, index) => (
<p key={index}>
<Trans i18nKey={`checkout.error.${failureReason}`} />
</p>
))
) : (
<Trans i18nKey={'checkout.error.description'} /> <Trans i18nKey={'checkout.error.description'} />
</p> )}
</AlertDescription> </AlertDescription>
</Alert> </Alert>

View File

@@ -1,5 +1,3 @@
import { notFound } from 'next/navigation';
import { createI18nServerInstance } from '@/lib/i18n/i18n.server'; import { createI18nServerInstance } from '@/lib/i18n/i18n.server';
import { PageBody, PageHeader } from '@/packages/ui/src/makerkit/page'; import { PageBody, PageHeader } from '@/packages/ui/src/makerkit/page';
import { retrieveCart } from '@lib/data/cart'; import { retrieveCart } from '@lib/data/cart';
@@ -8,9 +6,14 @@ import { listProductTypes } from '@lib/data/products';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import { getCartReservations } from '~/lib/services/reservation.service';
import { findProductTypeIdByHandle } from '~/lib/utils';
import Cart from '../../_components/cart'; import Cart from '../../_components/cart';
import CartTimer from '../../_components/cart/cart-timer'; import CartTimer from '../../_components/cart/cart-timer';
import { loadCurrentUserAccount } from '../../_lib/server/load-user-account';
import { AccountBalanceService } from '@kit/accounts/services/account-balance.service';
import { EnrichedCartItem } from '../../_components/cart/types';
export async function generateMetadata() { export async function generateMetadata() {
const { t } = await createI18nServerInstance(); const { t } = await createI18nServerInstance();
@@ -21,35 +24,48 @@ export async function generateMetadata() {
} }
async function CartPage() { async function CartPage() {
const cart = await retrieveCart().catch((error) => { const [
console.error('Failed to retrieve cart', error); cart,
return notFound(); { productTypes },
}); { account },
] = await Promise.all([
retrieveCart(),
listProductTypes(),
loadCurrentUserAccount(),
]);
const { productTypes } = await listProductTypes(); if (!account) {
const analysisPackagesType = productTypes.find( return null;
({ metadata }) => metadata?.handle === 'analysis-packages', }
const balanceSummary = await new AccountBalanceService().getBalanceSummary(account.id);
const synlabAnalysisTypeId = findProductTypeIdByHandle(
productTypes,
'synlab-analysis',
); );
const synlabAnalysisType = productTypes.find( const analysisPackagesTypeId = findProductTypeIdByHandle(
({ metadata }) => metadata?.handle === 'synlab-analysis', productTypes,
'analysis-packages',
); );
const synlabAnalyses = const synlabAnalyses =
analysisPackagesType && synlabAnalysisType && cart?.items analysisPackagesTypeId && synlabAnalysisTypeId && cart?.items
? cart.items.filter((item) => { ? cart.items.filter((item) => {
const productTypeId = item.product?.type_id; const productTypeId = item.product?.type_id;
if (!productTypeId) { if (!productTypeId) {
return false; return false;
} }
return [analysisPackagesType.id, synlabAnalysisType.id].includes( return [analysisPackagesTypeId, synlabAnalysisTypeId].includes(
productTypeId, productTypeId,
); );
}) })
: []; : [];
const ttoServiceItems =
cart?.items?.filter(
(item) => !synlabAnalyses.some((analysis) => analysis.id === item.id),
) ?? [];
let ttoServiceItems: EnrichedCartItem[] = [];
if (cart?.items?.length) {
ttoServiceItems = await getCartReservations(cart);
}
const otherItemsSorted = ttoServiceItems.sort((a, b) => const otherItemsSorted = ttoServiceItems.sort((a, b) =>
(a.updated_at ?? '') > (b.updated_at ?? '') ? -1 : 1, (a.updated_at ?? '') > (b.updated_at ?? '') ? -1 : 1,
); );
@@ -63,9 +79,11 @@ async function CartPage() {
{isTimerShown && <CartTimer cartItem={item} />} {isTimerShown && <CartTimer cartItem={item} />}
</PageHeader> </PageHeader>
<Cart <Cart
accountId={account.id}
cart={cart} cart={cart}
synlabAnalyses={synlabAnalyses} synlabAnalyses={synlabAnalyses}
ttoServiceItems={ttoServiceItems} ttoServiceItems={ttoServiceItems}
balanceSummary={balanceSummary}
/> />
</PageBody> </PageBody>
); );

View File

@@ -0,0 +1,81 @@
'use client';
import CartTotals from '@/app/home/(user)/_components/order/cart-totals';
import OrderDetails from '@/app/home/(user)/_components/order/order-details';
import OrderItems from '@/app/home/(user)/_components/order/order-items';
import Divider from '@modules/common/components/divider';
import { PageBody, PageHeader } from '@kit/ui/page';
import { Trans } from '@kit/ui/trans';
import { StoreOrder } from '@medusajs/types';
import { AnalysisOrder } from '~/lib/types/analysis-order';
import { useEffect, useRef, useState } from 'react';
import { retrieveOrder } from '@lib/data/orders';
import { GlobalLoader } from '@kit/ui/makerkit/global-loader';
function OrderConfirmedLoadingWrapper({
medusaOrder: initialMedusaOrder,
order,
}: {
medusaOrder: StoreOrder;
order: AnalysisOrder;
}) {
const [medusaOrder, setMedusaOrder] = useState<StoreOrder>(initialMedusaOrder);
const fetchingRef = useRef(false);
const paymentStatus = medusaOrder.payment_status;
const medusaOrderId = order.medusa_order_id;
useEffect(() => {
if (paymentStatus === 'captured') {
return;
}
const interval = setInterval(async () => {
if (fetchingRef.current) {
return;
}
fetchingRef.current = true;
const medusaOrder = await retrieveOrder(medusaOrderId, false);
fetchingRef.current = false;
setMedusaOrder(medusaOrder);
}, 2_000);
return () => clearInterval(interval);
}, [paymentStatus, medusaOrderId]);
const isPaid = paymentStatus === 'captured';
if (!isPaid) {
return (
<PageBody>
<div className="flex flex-col justify-start items-center h-full pt-[10vh]">
<div>
<GlobalLoader />
</div>
<h4 className="text-center">
<Trans i18nKey="cart:orderConfirmed.paymentConfirmationLoading" />
</h4>
</div>
</PageBody>
);
}
return (
<PageBody>
<PageHeader title={<Trans i18nKey="cart:orderConfirmed.title" />} />
<Divider />
<div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 gap-y-6 lg:px-4">
<OrderDetails order={order} />
<Divider />
<OrderItems medusaOrder={medusaOrder} />
<CartTotals medusaOrder={medusaOrder} />
</div>
</PageBody>
);
}
export default OrderConfirmedLoadingWrapper;

View File

@@ -1,18 +1,13 @@
import { redirect } from 'next/navigation'; import { redirect } from 'next/navigation';
import CartTotals from '@/app/home/(user)/_components/order/cart-totals';
import OrderDetails from '@/app/home/(user)/_components/order/order-details';
import OrderItems from '@/app/home/(user)/_components/order/order-items';
import { createI18nServerInstance } from '@/lib/i18n/i18n.server'; import { createI18nServerInstance } from '@/lib/i18n/i18n.server';
import { retrieveOrder } from '@lib/data/orders'; import { retrieveOrder } from '@lib/data/orders';
import Divider from '@modules/common/components/divider';
import { pathsConfig } from '@kit/shared/config'; import { pathsConfig } from '@kit/shared/config';
import { PageBody, PageHeader } from '@kit/ui/page';
import { Trans } from '@kit/ui/trans';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import { getAnalysisOrder } from '~/lib/services/order.service'; import { getAnalysisOrder } from '~/lib/services/order.service';
import OrderConfirmedLoadingWrapper from './order-confirmed-loading-wrapper';
export async function generateMetadata() { export async function generateMetadata() {
const { t } = await createI18nServerInstance(); const { t } = await createI18nServerInstance();
@@ -41,18 +36,7 @@ async function OrderConfirmedPage(props: {
redirect(pathsConfig.app.myOrders); redirect(pathsConfig.app.myOrders);
} }
return ( return <OrderConfirmedLoadingWrapper medusaOrder={medusaOrder} order={order} />;
<PageBody>
<PageHeader title={<Trans i18nKey="cart:orderConfirmed.title" />} />
<Divider />
<div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 gap-y-6 lg:px-4">
<OrderDetails order={order} />
<Divider />
<OrderItems medusaOrder={medusaOrder} />
<CartTotals medusaOrder={medusaOrder} />
</div>
</PageBody>
);
} }
export default withI18n(OrderConfirmedPage); export default withI18n(OrderConfirmedPage);

View File

@@ -26,17 +26,7 @@ async function OrderConfirmedPage(props: {
params: Promise<{ orderId: string }>; params: Promise<{ orderId: string }>;
}) { }) {
const params = await props.params; const params = await props.params;
const medusaOrder = await retrieveOrder(params.orderId).catch(() => null);
const order = await getAnalysisOrder({
analysisOrderId: Number(params.orderId),
}).catch(() => null);
if (!order) {
redirect(pathsConfig.app.myOrders);
}
const medusaOrder = await retrieveOrder(order.medusa_order_id).catch(
() => null,
);
if (!medusaOrder) { if (!medusaOrder) {
redirect(pathsConfig.app.myOrders); redirect(pathsConfig.app.myOrders);
} }
@@ -46,7 +36,12 @@ async function OrderConfirmedPage(props: {
<PageHeader title={<Trans i18nKey="cart:order.title" />} /> <PageHeader title={<Trans i18nKey="cart:order.title" />} />
<Divider /> <Divider />
<div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 gap-y-6 lg:px-4"> <div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 gap-y-6 lg:px-4">
<OrderDetails order={order} /> <OrderDetails
order={{
id: medusaOrder.id,
created_at: medusaOrder.created_at,
}}
/>
<Divider /> <Divider />
<OrderItems medusaOrder={medusaOrder} /> <OrderItems medusaOrder={medusaOrder} />
<CartTotals medusaOrder={medusaOrder} /> <CartTotals medusaOrder={medusaOrder} />

View File

@@ -11,12 +11,15 @@ import { PageBody } from '@kit/ui/makerkit/page';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import { getAnalysisOrders } from '~/lib/services/order.service'; import { getAnalysisOrders, getTtoOrders } from '~/lib/services/order.service';
import { findProductTypeIdByHandle } from '~/lib/utils';
import { listOrders } from '~/medusa/lib/data/orders'; import { listOrders } from '~/medusa/lib/data/orders';
import { HomeLayoutPageHeader } from '../../_components/home-page-header'; import { HomeLayoutPageHeader } from '../../_components/home-page-header';
import OrderBlock from '../../_components/orders/order-block'; import OrderBlock from '../../_components/orders/order-block';
const ORDERS_LIMIT = 50;
export async function generateMetadata() { export async function generateMetadata() {
const { t } = await createI18nServerInstance(); const { t } = await createI18nServerInstance();
@@ -26,17 +29,25 @@ export async function generateMetadata() {
} }
async function OrdersPage() { async function OrdersPage() {
const medusaOrders = await listOrders(); const [medusaOrders, analysisOrders, ttoOrders, { productTypes }] = await Promise.all([
const analysisOrders = await getAnalysisOrders(); listOrders(ORDERS_LIMIT),
const { productTypes } = await listProductTypes(); getAnalysisOrders(),
getTtoOrders(),
listProductTypes(),
]);
if (!medusaOrders || !productTypes) { if (!medusaOrders || !productTypes || !ttoOrders) {
redirect(pathsConfig.auth.signIn); redirect(pathsConfig.auth.signIn);
} }
const analysisPackagesType = productTypes.find( const analysisPackagesTypeId = findProductTypeIdByHandle(
({ metadata }) => metadata?.handle === 'analysis-packages', productTypes,
)!; 'analysis-package',
);
const ttoServiceTypeId = findProductTypeIdByHandle(
productTypes,
'tto-service',
);
return ( return (
<> <>
@@ -45,34 +56,45 @@ async function OrdersPage() {
description={<Trans i18nKey={'orders:description'} />} description={<Trans i18nKey={'orders:description'} />}
/> />
<PageBody> <PageBody>
{analysisOrders.map((analysisOrder) => { {medusaOrders.map((medusaOrder) => {
const medusaOrder = medusaOrders.find( const analysisOrder = analysisOrders.find(
({ id }) => id === analysisOrder.medusa_order_id, ({ medusa_order_id }) => medusa_order_id === medusaOrder.id,
); );
if (!medusaOrder) { if (!medusaOrder) {
return null; return null;
} }
const medusaOrderItems = medusaOrder.items || []; const medusaOrderItems = medusaOrder.items || [];
const medusaOrderItemsAnalysisPackages = medusaOrderItems.filter( const medusaOrderItemsAnalysisPackages = medusaOrderItems.filter(
(item) => item.product_type_id === analysisPackagesType?.id, (item) => item.product_type_id === analysisPackagesTypeId,
);
const medusaOrderItemsTtoServices = medusaOrderItems.filter(
(item) => item.product_type_id === ttoServiceTypeId,
); );
const medusaOrderItemsOther = medusaOrderItems.filter( const medusaOrderItemsOther = medusaOrderItems.filter(
(item) => item.product_type_id !== analysisPackagesType?.id, (item) =>
!item.product_type_id ||
![analysisPackagesTypeId, ttoServiceTypeId].includes(
item.product_type_id,
),
); );
return ( return (
<React.Fragment key={analysisOrder.id}> <React.Fragment key={medusaOrder.id}>
<Divider className="my-6" /> <Divider className="my-6" />
<OrderBlock <OrderBlock
medusaOrderId={medusaOrder.id}
analysisOrder={analysisOrder} analysisOrder={analysisOrder}
medusaOrderStatus={medusaOrder.status}
itemsAnalysisPackage={medusaOrderItemsAnalysisPackages} itemsAnalysisPackage={medusaOrderItemsAnalysisPackages}
itemsTtoService={medusaOrderItemsTtoServices}
itemsOther={medusaOrderItemsOther} itemsOther={medusaOrderItemsOther}
/> />
</React.Fragment> </React.Fragment>
); );
})} })}
{analysisOrders.length === 0 && ( {analysisOrders.length === 0 && ttoOrders.length === 0 && (
<h5 className="mt-6"> <h5 className="mt-6">
<Trans i18nKey="orders:noOrders" /> <Trans i18nKey="orders:noOrders" />
</h5> </h5>

View File

@@ -16,6 +16,7 @@ import Dashboard from '../_components/dashboard';
import DashboardCards from '../_components/dashboard-cards'; import DashboardCards from '../_components/dashboard-cards';
import Recommendations from '../_components/recommendations'; import Recommendations from '../_components/recommendations';
import RecommendationsSkeleton from '../_components/recommendations-skeleton'; import RecommendationsSkeleton from '../_components/recommendations-skeleton';
import { isValidOpenAiEnv } from '../_lib/server/is-valid-open-ai-env';
import { loadCurrentUserAccount } from '../_lib/server/load-user-account'; import { loadCurrentUserAccount } from '../_lib/server/load-user-account';
export const generateMetadata = async () => { export const generateMetadata = async () => {
@@ -52,17 +53,16 @@ async function UserHomePage() {
/> />
<PageBody> <PageBody>
<Dashboard account={account} bmiThresholds={bmiThresholds} /> <Dashboard account={account} bmiThresholds={bmiThresholds} />
{process.env.OPENAI_API_KEY && {(await isValidOpenAiEnv()) && (
process.env.PROMPT_ID_ANALYSIS_RECOMMENDATIONS && ( <>
<> <h4>
<h4> <Trans i18nKey="dashboard:recommendations.title" />
<Trans i18nKey="dashboard:recommendations.title" /> </h4>
</h4> <Suspense fallback={<RecommendationsSkeleton />}>
<Suspense fallback={<RecommendationsSkeleton />}> <Recommendations account={account} />
<Recommendations account={account} /> </Suspense>
</Suspense> </>
</> )}
)}
</PageBody> </PageBody>
</> </>
); );

View File

@@ -0,0 +1,40 @@
'use client';
import { isBefore, isSameDay } from 'date-fns';
import { uniq } from 'lodash';
import { Calendar } from '@kit/ui/shadcn/calendar';
import { Card } from '@kit/ui/shadcn/card';
import { cn } from '@kit/ui/utils';
import { useBooking } from './booking.provider';
export default function BookingCalendar() {
const { selectedDate, setSelectedDate, isLoadingTimeSlots, timeSlots } =
useBooking();
const availableDates = uniq(timeSlots?.map((timeSlot) => timeSlot.StartTime));
return (
<Card className="mb-4">
<Calendar
mode="single"
selected={selectedDate}
onSelect={setSelectedDate}
disabled={(date) => {
const today = new Date();
today.setHours(0, 0, 0, 0);
return (
isBefore(date, today) ||
!availableDates.some((dateWithBooking) =>
isSameDay(date, dateWithBooking),
)
);
}}
className={cn('rounded-md border', {
'pointer-events-none rounded-md border opacity-50':
isLoadingTimeSlots,
})}
/>
</Card>
);
}

View File

@@ -0,0 +1,49 @@
'use client';
import { StoreProduct } from '@medusajs/types';
import { Trans } from '@kit/ui/trans';
import { EnrichedCartItem } from '../cart/types';
import BookingCalendar from './booking-calendar';
import { BookingProvider } from './booking.provider';
import LocationSelector from './location-selector';
import ServiceSelector from './service-selector';
import TimeSlots from './time-slots';
const BookingContainer = ({
category,
cartItem,
onComplete,
}: {
category: { products: StoreProduct[]; countryCode: string };
cartItem?: EnrichedCartItem;
onComplete?: () => void;
}) => {
const products = cartItem?.product ? [cartItem.product] : category.products;
if (!cartItem || !products?.length) {
<p>
<Trans i18nKey="booking:noProducts" />
</p>;
}
return (
<BookingProvider category={{ products }} service={cartItem?.product}>
<div className="xs:flex-row flex max-h-full flex-col gap-6">
<div className="flex flex-col">
<ServiceSelector products={products} />
<BookingCalendar />
<LocationSelector />
</div>
<TimeSlots
countryCode={category.countryCode}
cartItem={cartItem}
onComplete={onComplete}
/>
</div>
</BookingProvider>
);
};
export default BookingContainer;

View File

@@ -0,0 +1,77 @@
import { createContext } from 'react';
import { StoreProduct } from '@medusajs/types';
import { noop } from 'lodash';
import { Tables } from '@kit/supabase/database';
export type Location = Tables<
{ schema: 'medreport' },
'connected_online_locations'
>;
export type TimeSlotResponse = {
timeSlots: TimeSlot[];
locations: Location[];
};
export type TimeSlot = {
ClinicID: number;
LocationID: number;
UserID: number;
SyncUserID: number;
ServiceID: number;
HKServiceID: number;
StartTime: Date;
EndTime: Date;
PayorCode: string;
serviceProvider?: ServiceProvider;
syncedService?: SyncedService;
} & { location?: Location };
export type ServiceProvider = {
name: string;
id: number;
jobTitleEn: string | null;
jobTitleEt: string | null;
jobTitleRu: string | null;
clinicId: number;
};
export type SyncedService = Tables<
{ schema: 'medreport' },
'connected_online_services'
> & {
providerClinic: ProviderClinic;
};
export type ProviderClinic = Tables<
{ schema: 'medreport' },
'connected_online_providers'
> & { locations: Location[] };
const BookingContext = createContext<{
timeSlots: TimeSlot[] | null;
selectedService: StoreProduct | null;
locations: Location[] | null;
selectedLocationId: number | null;
selectedDate?: Date;
isLoadingTimeSlots?: boolean;
setSelectedService: (selectedService: StoreProduct | null) => void;
setSelectedLocationId: (selectedLocationId: number | null) => void;
updateTimeSlots: (serviceIds: number[]) => Promise<void>;
setSelectedDate: (selectedDate?: Date) => void;
}>({
timeSlots: null,
selectedService: null,
locations: null,
selectedLocationId: null,
selectedDate: new Date(),
isLoadingTimeSlots: false,
setSelectedService: (_) => _,
setSelectedLocationId: (_) => _,
updateTimeSlots: async (_) => noop(),
setSelectedDate: (_) => _,
});
export { BookingContext };

View File

@@ -0,0 +1,80 @@
import React, { useContext, useEffect, useState } from 'react';
import { StoreProduct } from '@medusajs/types';
import { getAvailableTimeSlotsForDisplay } from '~/lib/services/connected-online.service';
import { BookingContext, Location, TimeSlot } from './booking.context';
export function useBooking() {
const context = useContext(BookingContext);
if (!context) {
throw new Error('useBooking must be used within a BookingProvider.');
}
return context;
}
export const BookingProvider: React.FC<{
children: React.ReactElement;
category: { products: StoreProduct[] };
service?: StoreProduct;
}> = ({ children, category, service }) => {
const [selectedService, setSelectedService] = useState<StoreProduct | null>(
(service ?? category?.products?.[0]) || null,
);
const [selectedLocationId, setSelectedLocationId] = useState<number | null>(
null,
);
const [selectedDate, setSelectedDate] = useState<Date>();
const [timeSlots, setTimeSlots] = useState<TimeSlot[] | null>(null);
const [locations, setLocations] = useState<Location[] | null>(null);
const [isLoadingTimeSlots, setIsLoadingTimeSlots] = useState(false);
useEffect(() => {
const metadataServiceIds = selectedService?.metadata?.serviceIds as string;
if (metadataServiceIds) {
const json = JSON.parse(metadataServiceIds);
if (Array.isArray(json)) {
updateTimeSlots(json);
}
}
}, [selectedService, selectedLocationId]);
const updateTimeSlots = async (serviceIds: number[]) => {
setIsLoadingTimeSlots(true);
try {
console.log('serviceIds', serviceIds, selectedLocationId);
const response = await getAvailableTimeSlotsForDisplay(
serviceIds,
selectedLocationId,
);
setTimeSlots(response.timeSlots);
setLocations(response.locations);
} catch (error) {
setTimeSlots(null);
} finally {
setIsLoadingTimeSlots(false);
}
};
return (
<BookingContext.Provider
value={{
timeSlots,
locations,
selectedService,
selectedLocationId,
setSelectedLocationId,
selectedDate,
isLoadingTimeSlots,
setSelectedService,
updateTimeSlots,
setSelectedDate,
}}
>
{children}
</BookingContext.Provider>
);
};

View File

@@ -0,0 +1,55 @@
import { Label } from '@medusajs/ui';
import { useTranslation } from 'react-i18next';
import { RadioGroup, RadioGroupItem } from '@kit/ui/radio-group';
import { Card } from '@kit/ui/shadcn/card';
import { Trans } from '@kit/ui/trans';
import { useBooking } from './booking.provider';
const LocationSelector = () => {
const { t } = useTranslation();
const { selectedLocationId, setSelectedLocationId, locations } = useBooking();
const onLocationSelect = (locationId: number | string | null) => {
if (locationId === 'all') return setSelectedLocationId(null);
setSelectedLocationId(Number(locationId));
};
return (
<Card className="mb-4 p-4">
<h5 className="text-semibold mb-2">
<Trans i18nKey="booking:locations" />
</h5>
<div className="flex flex-col">
<RadioGroup
className="mb-2 flex flex-col"
onValueChange={(val) => onLocationSelect(val)}
>
<div className="flex items-center gap-2">
<RadioGroupItem
value={'all'}
id={'all'}
checked={selectedLocationId === null}
/>
<Label htmlFor={'all'}>{t('booking:showAllLocations')}</Label>
</div>
{locations?.map((location) => (
<div key={location.sync_id} className="flex items-center gap-2">
<RadioGroupItem
value={location.sync_id.toString()}
id={location.sync_id.toString()}
checked={selectedLocationId === location.sync_id}
/>
<Label htmlFor={location.sync_id.toString()}>
{location.name}
</Label>
</div>
))}
</RadioGroup>
</div>
</Card>
);
};
export default LocationSelector;

View File

@@ -0,0 +1,85 @@
import { useState } from 'react';
import { StoreProduct } from '@medusajs/types';
import { ChevronDown } from 'lucide-react';
import { Card } from '@kit/ui/shadcn/card';
import { Label } from '@kit/ui/shadcn/label';
import {
Popover,
PopoverContent,
PopoverTrigger,
} from '@kit/ui/shadcn/popover';
import { RadioGroup, RadioGroupItem } from '@kit/ui/shadcn/radio-group';
import { Trans } from '@kit/ui/trans';
import { useBooking } from './booking.provider';
const ServiceSelector = ({ products }: { products: StoreProduct[] }) => {
const { selectedService, setSelectedService } = useBooking();
const [collapsed, setCollapsed] = useState(false);
const [firstFourProducts] = useState<StoreProduct[]>(products?.slice(0, 4));
const onServiceSelect = async (productId: StoreProduct['id']) => {
const product = products.find((p) => p.id === productId);
setSelectedService(product ?? null);
setCollapsed(false);
};
return (
<Card className="mb-4 p-4">
<h5 className="text-semibold mb-2">
<Trans i18nKey="booking:services" />
</h5>
<Popover open={collapsed} onOpenChange={setCollapsed}>
<div className="flex flex-col">
<RadioGroup
defaultValue={selectedService?.id || ''}
className="mb-2 flex flex-col"
onValueChange={onServiceSelect}
>
{firstFourProducts?.map((product) => (
<div key={product.id} className="flex items-center gap-2">
<RadioGroupItem
value={product.id}
id={product.id}
checked={selectedService?.id === product.id}
/>
<Label htmlFor={product.id}>{product.title}</Label>
</div>
))}
</RadioGroup>
{products.length > 4 && (
<PopoverTrigger asChild>
<div
onClick={() => setCollapsed((_) => !_)}
className="flex cursor-pointer items-center justify-between border-t py-1"
>
<span>
<Trans i18nKey="booking:showAll" />
</span>
<ChevronDown />
</div>
</PopoverTrigger>
)}
</div>
<PopoverContent sideOffset={10}>
<RadioGroup onValueChange={onServiceSelect}>
{products?.map((product) => (
<div key={product.id + '-2'} className="flex items-center gap-2">
<RadioGroupItem
value={product.id}
id={product.id + '-2'}
checked={selectedService?.id === product.id}
/>
<Label htmlFor={product.id + '-2'}>{product.title}</Label>
</div>
))}
</RadioGroup>
</PopoverContent>
</Popover>
</Card>
);
};
export default ServiceSelector;

View File

@@ -0,0 +1,319 @@
import { useMemo, useState } from 'react';
import { useRouter } from 'next/navigation';
import { formatCurrency } from '@/packages/shared/src/utils';
import { addHours, isAfter, isSameDay } from 'date-fns';
import { orderBy } from 'lodash';
import { useTranslation } from 'react-i18next';
import { pathsConfig } from '@kit/shared/config';
import { formatDateAndTime } from '@kit/shared/utils';
import { Button } from '@kit/ui/shadcn/button';
import { Card } from '@kit/ui/shadcn/card';
import { toast } from '@kit/ui/sonner';
import { Trans } from '@kit/ui/trans';
import { cn } from '@kit/ui/utils';
import { updateReservationTime } from '~/lib/services/reservation.service';
import { createInitialReservationAction } from '../../_lib/server/actions';
import { EnrichedCartItem } from '../cart/types';
import { ServiceProvider, TimeSlot } from './booking.context';
import { useBooking } from './booking.provider';
const getServiceProviderTitle = (
currentLocale: string,
serviceProvider?: ServiceProvider,
) => {
if (!serviceProvider) return null;
if (currentLocale === 'en') return serviceProvider.jobTitleEn;
if (currentLocale === 'ru') return serviceProvider.jobTitleRu;
return serviceProvider.jobTitleEt;
};
const PAGE_SIZE = 7;
const TimeSlots = ({
countryCode,
cartItem,
onComplete,
}: {
countryCode: string;
cartItem?: EnrichedCartItem;
onComplete?: () => void;
}) => {
const [currentPage, setCurrentPage] = useState(1);
const {
t,
i18n: { language: currentLocale },
} = useTranslation();
const booking = useBooking();
const router = useRouter();
const selectedDate = booking.selectedDate ?? new Date();
const filteredBookings = useMemo(
() =>
orderBy(
booking?.timeSlots?.filter(({ StartTime }) => {
const firstAvailableTimeToSelect = isSameDay(selectedDate, new Date())
? addHours(new Date(), 0.5)
: selectedDate;
return isAfter(StartTime, firstAvailableTimeToSelect);
}) ?? [],
'StartTime',
'asc',
),
[booking.timeSlots, selectedDate],
);
const totalPages = Math.ceil(filteredBookings.length / PAGE_SIZE);
const paginatedBookings = useMemo(() => {
const startIndex = (currentPage - 1) * PAGE_SIZE;
const endIndex = startIndex + PAGE_SIZE;
return filteredBookings.slice(startIndex, endIndex);
}, [filteredBookings, currentPage, PAGE_SIZE]);
const handlePageChange = (page: number) => {
setCurrentPage(page);
};
const generatePageNumbers = () => {
const pages = [];
const maxVisiblePages = 5;
if (totalPages <= maxVisiblePages) {
for (let i = 1; i <= totalPages; i++) {
pages.push(i);
}
} else {
if (currentPage <= 3) {
for (let i = 1; i <= 4; i++) {
pages.push(i);
}
pages.push('...');
pages.push(totalPages);
} else if (currentPage >= totalPages - 2) {
pages.push(1);
pages.push('...');
for (let i = totalPages - 3; i <= totalPages; i++) {
pages.push(i);
}
} else {
pages.push(1);
pages.push('...');
for (let i = currentPage - 1; i <= currentPage + 1; i++) {
pages.push(i);
}
pages.push('...');
pages.push(totalPages);
}
}
return pages;
};
if (!booking?.timeSlots?.length) {
return null;
}
const handleBookTime = async (timeSlot: TimeSlot, comments?: string) => {
const selectedService = booking.selectedService;
const selectedVariant = selectedService?.variants?.[0];
const syncedService = timeSlot.syncedService;
if (!syncedService || !selectedVariant) {
return toast.error(t('booking:serviceNotFound'));
}
const bookTimePromise = createInitialReservationAction(
selectedVariant,
countryCode,
Number(syncedService.id),
syncedService?.clinic_id,
timeSlot.UserID,
timeSlot.SyncUserID,
timeSlot.StartTime,
booking.selectedLocationId ? booking.selectedLocationId : null,
comments,
).then(() => {
if (onComplete) {
onComplete();
}
router.push(pathsConfig.app.cart);
});
toast.promise(() => bookTimePromise, {
success: <Trans i18nKey={'booking:bookTimeSuccess'} />,
error: <Trans i18nKey={'booking:bookTimeError'} />,
loading: <Trans i18nKey={'booking:bookTimeLoading'} />,
});
};
const handleChangeTime = async (
timeSlot: TimeSlot,
reservationId: number,
cartId: string,
) => {
const syncedService = timeSlot.syncedService;
if (!syncedService) {
return toast.error(t('booking:serviceNotFound'));
}
const bookTimePromise = updateReservationTime({
reservationId,
newStartTime: timeSlot.StartTime,
newServiceId: Number(syncedService.id),
newAppointmentUserId: timeSlot.UserID,
newSyncUserId: timeSlot.SyncUserID,
newLocationId: booking.selectedLocationId
? booking.selectedLocationId
: null,
cartId,
});
toast.promise(() => bookTimePromise, {
success: <Trans i18nKey={'booking:bookTimeSuccess'} />,
error: <Trans i18nKey={'booking:bookTimeError'} />,
loading: <Trans i18nKey={'booking:bookTimeLoading'} />,
});
if (onComplete) {
onComplete();
}
};
const handleTimeSelect = async (timeSlot: TimeSlot) => {
if (cartItem?.reservation.id) {
return handleChangeTime(
timeSlot,
cartItem.reservation.id,
cartItem.cart_id,
);
}
return handleBookTime(timeSlot);
};
return (
<div className="flex w-full flex-col gap-4">
<div className="flex h-full w-full flex-col gap-2 overflow-auto">
{paginatedBookings.map((timeSlot, index) => {
const isEHIF = timeSlot.HKServiceID > 0;
const serviceProviderTitle = getServiceProviderTitle(
currentLocale,
timeSlot.serviceProvider,
);
const price =
booking.selectedService?.variants?.[0]?.calculated_price
?.calculated_amount ?? cartItem?.unit_price;
return (
<Card
className="xs:flex xs:justify-between grid w-full justify-center-safe gap-3 p-4"
key={index}
>
<div>
<span>{formatDateAndTime(timeSlot.StartTime.toString())}</span>
<div className="flex">
<h5
className={cn(
(serviceProviderTitle || isEHIF) &&
"after:mx-2 after:content-['·']",
)}
>
{timeSlot.serviceProvider?.name}
</h5>
{serviceProviderTitle && (
<span
className={cn(isEHIF && "after:mx-2 after:content-['·']")}
>
{serviceProviderTitle}
</span>
)}
{isEHIF && <span>{t('booking:ehifBooking')}</span>}
</div>
<div className="flex text-xs">{timeSlot.location?.address}</div>
</div>
<div className="flex-end not-last:xs:justify-center flex items-center justify-between gap-2">
<span className="text-sm font-semibold">
{formatCurrency({
currencyCode: 'EUR',
locale: 'et-EE',
value: price ?? '',
})}
</span>
<Button onClick={() => handleTimeSelect(timeSlot)} size="sm">
<Trans i18nKey="common:book" />
</Button>
</div>
</Card>
);
})}
{!paginatedBookings.length && (
<div className="wrap text-muted-foreground flex size-full content-center-safe justify-center-safe">
<p>{t('booking:noResults')}</p>
</div>
)}
</div>
{totalPages > 1 && (
<div className="flex items-center justify-between">
<div className="text-muted-foreground text-sm">
{t('common:pageOfPages', {
page: currentPage,
total: totalPages,
})}
</div>
<div className="flex items-center gap-1">
<Button
variant="outline"
size="sm"
onClick={() => handlePageChange(currentPage - 1)}
disabled={currentPage === 1}
>
<Trans i18nKey="common:previous" defaultValue="Previous" />
</Button>
{generatePageNumbers().map((page, index) => (
<Button
key={index}
variant={page === currentPage ? 'default' : 'outline'}
size="sm"
onClick={() =>
typeof page === 'number' && handlePageChange(page)
}
disabled={page === '...'}
className={cn(
'min-w-[2rem]',
page === '...' && 'cursor-default hover:bg-transparent',
)}
>
{page}
</Button>
))}
<Button
variant="outline"
size="sm"
onClick={() => handlePageChange(currentPage + 1)}
disabled={currentPage === totalPages}
>
<Trans i18nKey="common:next" defaultValue="Next" />
</Button>
</div>
</div>
)}
</div>
);
};
export default TimeSlots;

View File

@@ -0,0 +1,146 @@
'use client';
import { useState } from 'react';
import { formatCurrency, formatDateAndTime } from '@/packages/shared/src/utils';
import { useTranslation } from 'react-i18next';
import { Button } from '@kit/ui/button';
import {
Dialog,
DialogContent,
DialogDescription,
DialogHeader,
DialogTitle,
} from '@kit/ui/dialog';
import { TableCell, TableRow } from '@kit/ui/table';
import { Trans } from '@kit/ui/trans';
import BookingContainer from '../booking/booking-container';
import CartItemDelete from './cart-item-delete';
import { EnrichedCartItem } from './types';
const EditCartServiceItemModal = ({
item,
onComplete,
}: {
item: EnrichedCartItem | null;
onComplete: () => void;
}) => {
if (!item) return null;
return (
<Dialog defaultOpen>
<DialogContent className="xs:max-w-[90vw] flex max-h-screen max-w-full flex-col items-center gap-4 space-y-4 overflow-y-scroll">
<DialogHeader className="items-center text-center">
<DialogTitle>
<Trans i18nKey="cart:editServiceItem.title" />
</DialogTitle>
<DialogDescription>
<Trans i18nKey="cart:editServiceItem.description" />
</DialogDescription>
</DialogHeader>
<div>
{item.product && item.reservation.countryCode ? (
<BookingContainer
category={{
products: [item.product],
countryCode: item.reservation.countryCode,
}}
cartItem={item}
onComplete={onComplete}
/>
) : (
<p>
<Trans i18nKey="booking:noProducts" />
</p>
)}
</div>
</DialogContent>
</Dialog>
);
};
export default function CartServiceItem({
item,
currencyCode,
isUnavailable,
}: {
item: EnrichedCartItem;
currencyCode: string;
isUnavailable?: boolean;
}) {
const [editingItem, setEditingItem] = useState<EnrichedCartItem | null>(null);
const {
i18n: { language },
} = useTranslation();
return (
<>
<TableRow className="w-full" data-testid="product-row">
<TableCell className="w-[100%] px-4 text-left sm:px-6">
<p
className="txt-medium-plus text-ui-fg-base"
data-testid="product-title"
>
{item.product_title}
</p>
</TableCell>
<TableCell className="px-4 sm:px-6">
{formatDateAndTime(item.reservation.startTime.toString())}
</TableCell>
<TableCell className="px-4 sm:px-6">
{item.reservation.location?.address ?? '-'}
</TableCell>
<TableCell className="px-4 sm:px-6">{item.quantity}</TableCell>
<TableCell className="min-w-[80px] px-4 sm:px-6">
{formatCurrency({
value: item.unit_price,
currencyCode,
locale: language,
})}
</TableCell>
<TableCell className="min-w-[80px] px-4 text-right sm:px-6">
{formatCurrency({
value: item.total,
currencyCode,
locale: language,
})}
</TableCell>
<TableCell className="px-4 text-right sm:px-6">
<span className="flex justify-end gap-x-1">
<Button size="sm" onClick={() => setEditingItem(item)}>
<Trans i18nKey="common:change" />
</Button>
</span>
</TableCell>
<TableCell className="px-4 text-right sm:px-6">
<span className="flex w-[60px] justify-end gap-x-1">
<CartItemDelete id={item.id} />
</span>
</TableCell>
</TableRow>
{isUnavailable && (
<TableRow>
<TableCell
colSpan={8}
className="text-destructive px-4 text-left sm:px-6"
>
<Trans i18nKey="booking:timeSlotUnavailable" />
</TableCell>
</TableRow>
)}
<EditCartServiceItemModal
item={editingItem}
onComplete={() => setEditingItem(null)}
/>
</>
);
}

View File

@@ -0,0 +1,72 @@
import { StoreCart } from '@medusajs/types';
import {
Table,
TableBody,
TableHead,
TableHeader,
TableRow,
} from '@kit/ui/table';
import { Trans } from '@kit/ui/trans';
import CartServiceItem from './cart-service-item';
import { EnrichedCartItem } from './types';
export default function CartServiceItems({
cart,
items,
productColumnLabelKey,
unavailableLineItemIds,
}: {
cart: StoreCart;
items: EnrichedCartItem[];
productColumnLabelKey: string;
unavailableLineItemIds?: string[];
}) {
if (!items || items.length === 0) {
return null;
}
return (
<Table className="border-separate rounded-lg border">
<TableHeader className="text-ui-fg-subtle txt-medium-plus">
<TableRow>
<TableHead className="px-4 sm:px-6">
<Trans i18nKey={productColumnLabelKey} />
</TableHead>
<TableHead className="px-4 sm:px-6">
<Trans i18nKey="cart:table.time" />
</TableHead>
<TableHead className="px-4 sm:px-6">
<Trans i18nKey="cart:table.location" />
</TableHead>
<TableHead className="px-4 sm:px-6">
<Trans i18nKey="cart:table.quantity" />
</TableHead>
<TableHead className="min-w-[100px] px-4 sm:px-6">
<Trans i18nKey="cart:table.price" />
</TableHead>
<TableHead className="min-w-[100px] px-4 text-right sm:px-6">
<Trans i18nKey="cart:table.total" />
</TableHead>
<TableHead className="px-4 sm:px-6"></TableHead>
<TableHead className="px-4 sm:px-6"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{items
.sort((a, b) =>
(a.created_at ?? '') > (b.created_at ?? '') ? -1 : 1,
)
.map((item) => (
<CartServiceItem
key={item.id}
item={item}
currencyCode={cart.currency_code}
isUnavailable={unavailableLineItemIds?.includes(item.id)}
/>
))}
</TableBody>
</Table>
);
}

View File

@@ -2,9 +2,7 @@
import { useState } from 'react'; import { useState } from 'react';
import { handleNavigateToPayment } from '@/lib/services/medusaCart.service';
import { formatCurrency } from '@/packages/shared/src/utils'; import { formatCurrency } from '@/packages/shared/src/utils';
import { initiatePaymentSession } from '@lib/data/cart';
import { StoreCart, StoreCartLineItem } from '@medusajs/types'; import { StoreCart, StoreCartLineItem } from '@medusajs/types';
import { Loader2 } from 'lucide-react'; import { Loader2 } from 'lucide-react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
@@ -15,28 +13,41 @@ import { Trans } from '@kit/ui/trans';
import AnalysisLocation from './analysis-location'; import AnalysisLocation from './analysis-location';
import CartItems from './cart-items'; import CartItems from './cart-items';
import CartServiceItems from './cart-service-items';
import DiscountCode from './discount-code'; import DiscountCode from './discount-code';
import { initiatePayment } from '../../_lib/server/cart-actions';
import { useRouter } from 'next/navigation';
import { AccountBalanceSummary } from '@kit/accounts/services/account-balance.service';
import { EnrichedCartItem } from './types';
const IS_DISCOUNT_SHOWN = true as boolean; const IS_DISCOUNT_SHOWN = true as boolean;
export default function Cart({ export default function Cart({
accountId,
cart, cart,
synlabAnalyses, synlabAnalyses,
ttoServiceItems, ttoServiceItems,
balanceSummary,
}: { }: {
accountId: string;
cart: StoreCart | null; cart: StoreCart | null;
synlabAnalyses: StoreCartLineItem[]; synlabAnalyses: StoreCartLineItem[];
ttoServiceItems: StoreCartLineItem[]; ttoServiceItems: EnrichedCartItem[];
balanceSummary: AccountBalanceSummary | null;
}) { }) {
const { const {
i18n: { language }, i18n: { language },
} = useTranslation(); } = useTranslation();
const [isInitiatingSession, setIsInitiatingSession] = useState(false); const [isInitiatingSession, setIsInitiatingSession] = useState(false);
const router = useRouter();
const [unavailableLineItemIds, setUnavailableLineItemIds] =
useState<string[]>();
const items = cart?.items ?? []; const items = cart?.items ?? [];
const hasCartItems = cart && Array.isArray(items) && items.length > 0;
if (!cart || items.length === 0) { if (!hasCartItems) {
return ( return (
<div className="content-container py-5 lg:px-4"> <div className="content-container py-5 lg:px-4">
<div> <div>
@@ -56,24 +67,38 @@ export default function Cart({
); );
} }
async function initiatePayment() { async function initiateSession() {
setIsInitiatingSession(true); setIsInitiatingSession(true);
const response = await initiatePaymentSession(cart!, {
provider_id: 'pp_montonio_montonio', try {
}); const { url, isFullyPaidByBenefits, orderId, unavailableLineItemIds } = await initiatePayment({
if (response.payment_collection) { accountId,
const { payment_sessions } = response.payment_collection; balanceSummary: balanceSummary!,
const paymentSessionId = payment_sessions![0]!.id; cart: cart!,
const url = await handleNavigateToPayment({ language, paymentSessionId }); language,
window.location.href = url; });
} else { if (unavailableLineItemIds) {
setUnavailableLineItemIds(unavailableLineItemIds);
}
if (url) {
window.location.href = url;
} else if (isFullyPaidByBenefits) {
if (typeof orderId !== 'number') {
throw new Error('Order ID is missing');
}
router.push(`/home/order/${orderId}/confirmed`);
}
} catch (error) {
console.error('Failed to initiate payment', error);
setIsInitiatingSession(false); setIsInitiatingSession(false);
} }
} }
const hasCartItems = Array.isArray(cart.items) && cart.items.length > 0;
const isLocationsShown = synlabAnalyses.length > 0; const isLocationsShown = synlabAnalyses.length > 0;
const companyBenefitsTotal = balanceSummary?.totalBalance ?? 0;
const montonioTotal = cart && companyBenefitsTotal > 0 ? cart.total - companyBenefitsTotal : cart.total;
return ( return (
<div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 lg:px-4"> <div className="small:grid-cols-[1fr_360px] grid grid-cols-1 gap-x-40 lg:px-4">
<div className="flex flex-col gap-y-6 bg-white"> <div className="flex flex-col gap-y-6 bg-white">
@@ -82,10 +107,11 @@ export default function Cart({
items={synlabAnalyses} items={synlabAnalyses}
productColumnLabelKey="cart:items.synlabAnalyses.productColumnLabel" productColumnLabelKey="cart:items.synlabAnalyses.productColumnLabel"
/> />
<CartItems <CartServiceItems
cart={cart} cart={cart}
items={ttoServiceItems} items={ttoServiceItems}
productColumnLabelKey="cart:items.ttoServices.productColumnLabel" productColumnLabelKey="cart:items.ttoServices.productColumnLabel"
unavailableLineItemIds={unavailableLineItemIds}
/> />
</div> </div>
{hasCartItems && ( {hasCartItems && (
@@ -106,7 +132,7 @@ export default function Cart({
</p> </p>
</div> </div>
</div> </div>
<div className="flex gap-x-4 px-4 py-2 sm:justify-end sm:px-6 sm:py-4"> <div className="flex gap-x-4 px-4 pt-2 sm:justify-end sm:px-6 sm:pt-4">
<div className="w-full sm:mr-[42px] sm:w-auto"> <div className="w-full sm:mr-[42px] sm:w-auto">
<p className="text-muted-foreground ml-0 text-sm font-bold"> <p className="text-muted-foreground ml-0 text-sm font-bold">
<Trans i18nKey="cart:order.promotionsTotal" /> <Trans i18nKey="cart:order.promotionsTotal" />
@@ -122,6 +148,24 @@ export default function Cart({
</p> </p>
</div> </div>
</div> </div>
{companyBenefitsTotal > 0 && (
<div className="flex gap-x-4 px-4 py-2 sm:justify-end sm:px-6 sm:py-4">
<div className="w-full sm:mr-[42px] sm:w-auto">
<p className="text-muted-foreground ml-0 text-sm font-bold">
<Trans i18nKey="cart:order.companyBenefitsTotal" />
</p>
</div>
<div className={`sm:mr-[112px] sm:w-[50px]`}>
<p className="text-right text-sm">
{formatCurrency({
value: (companyBenefitsTotal > cart.total ? cart.total : companyBenefitsTotal),
currencyCode: cart.currency_code,
locale: language,
})}
</p>
</div>
</div>
)}
<div className="flex gap-x-4 px-4 sm:justify-end sm:px-6"> <div className="flex gap-x-4 px-4 sm:justify-end sm:px-6">
<div className="w-full sm:mr-[42px] sm:w-auto"> <div className="w-full sm:mr-[42px] sm:w-auto">
<p className="ml-0 text-sm font-bold"> <p className="ml-0 text-sm font-bold">
@@ -131,7 +175,7 @@ export default function Cart({
<div className={`sm:mr-[112px] sm:w-[50px]`}> <div className={`sm:mr-[112px] sm:w-[50px]`}>
<p className="text-right text-sm"> <p className="text-right text-sm">
{formatCurrency({ {formatCurrency({
value: cart.total, value: montonioTotal < 0 ? 0 : montonioTotal,
currencyCode: cart.currency_code, currencyCode: cart.currency_code,
locale: language, locale: language,
})} })}
@@ -175,7 +219,7 @@ export default function Cart({
<div> <div>
<Button <Button
className="h-10" className="h-10"
onClick={initiatePayment} onClick={initiateSession}
disabled={isInitiatingSession} disabled={isInitiatingSession}
> >
{isInitiatingSession && ( {isInitiatingSession && (

View File

@@ -1,3 +1,6 @@
import { StoreCartLineItem } from "@medusajs/types";
import { Reservation } from "~/lib/types/reservation";
export interface MontonioOrderToken { export interface MontonioOrderToken {
uuid: string; uuid: string;
accessKey: string; accessKey: string;
@@ -10,6 +13,12 @@ export interface MontonioOrderToken {
| 'PENDING' | 'PENDING'
| 'EXPIRED' | 'EXPIRED'
| 'REFUNDED'; | 'REFUNDED';
| 'PAID'
| 'FAILED'
| 'CANCELLED'
| 'PENDING'
| 'EXPIRED'
| 'REFUNDED';
paymentMethod: string; paymentMethod: string;
grandTotal: number; grandTotal: number;
currency: string; currency: string;
@@ -20,3 +29,10 @@ export interface MontonioOrderToken {
iat: number; iat: number;
exp: number; exp: number;
} }
export enum CartItemType {
analysisOrders = 'analysisOrders',
ttoServices = 'ttoServices',
}
export type EnrichedCartItem = StoreCartLineItem & { reservation: Reservation };

View File

@@ -3,12 +3,33 @@ import Link from 'next/link';
import { ChevronRight, HeartPulse } from 'lucide-react'; import { ChevronRight, HeartPulse } from 'lucide-react';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
import { Card, CardDescription, CardFooter, CardHeader } from '@kit/ui/card'; import {
Card,
CardContent,
CardDescription,
CardFooter,
CardHeader,
CardTitle,
} from '@kit/ui/card';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { formatCurrency } from '@/packages/shared/src/utils';
import { createI18nServerInstance } from '@/lib/i18n/i18n.server';
import { cn } from '@kit/ui/lib/utils';
import { loadCurrentUserAccount } from '../_lib/server/load-user-account';
import { getAccountBalanceSummary } from '../_lib/server/balance-actions';
export default async function DashboardCards() {
const { language } = await createI18nServerInstance();
const { account } = await loadCurrentUserAccount();
const balanceSummary = account ? await getAccountBalanceSummary(account.id) : null;
export default function DashboardCards() {
return ( return (
<div className="flex gap-4"> <div
className={cn(
'grid grid-cols-1 gap-4',
'md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4',
)}>
<Card <Card
variant="gradient-success" variant="gradient-success"
className="xs:w-1/2 flex w-full flex-col justify-between sm:w-auto" className="xs:w-1/2 flex w-full flex-col justify-between sm:w-auto"
@@ -38,6 +59,34 @@ export default function DashboardCards() {
</CardDescription> </CardDescription>
</CardFooter> </CardFooter>
</Card> </Card>
<Card className="flex flex-col justify-center">
<CardHeader>
<CardTitle size="h5">
<Trans i18nKey="dashboard:heroCard.benefits.title" />
</CardTitle>
</CardHeader>
<CardContent>
<Figure>
{formatCurrency({
value: balanceSummary?.totalBalance || 0,
locale: language,
currencyCode: 'EUR',
})}
</Figure>
<CardDescription>
<Trans
i18nKey="dashboard:heroCard.benefits.validUntil"
values={{ date: '31.12.2025' }}
/>
</CardDescription>
</CardContent>
</Card>
</div> </div>
); );
} }
function Figure(props: React.PropsWithChildren) {
return <div className={'text-3xl font-bold'}>{props.children}</div>;
}

View File

@@ -2,7 +2,6 @@
import Link from 'next/link'; import Link from 'next/link';
import { Database } from '@/packages/supabase/src/database.types';
import { BlendingModeIcon, RulerHorizontalIcon } from '@radix-ui/react-icons'; import { BlendingModeIcon, RulerHorizontalIcon } from '@radix-ui/react-icons';
import { isNil } from 'lodash'; import { isNil } from 'lodash';
import { import {
@@ -15,7 +14,7 @@ import {
User, User,
} from 'lucide-react'; } from 'lucide-react';
import type { AccountWithParams } from '@kit/accounts/types/accounts'; import type { AccountWithParams, BmiThresholds } from '@kit/accounts/types/accounts';
import { pathsConfig } from '@kit/shared/config'; import { pathsConfig } from '@kit/shared/config';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
import { import {
@@ -138,10 +137,7 @@ export default function Dashboard({
bmiThresholds, bmiThresholds,
}: { }: {
account: AccountWithParams; account: AccountWithParams;
bmiThresholds: Omit< bmiThresholds: Omit<BmiThresholds, 'id'>[];
Database['medreport']['Tables']['bmi_thresholds']['Row'],
'id'
>[];
}) { }) {
const height = account.accountParams?.height || 0; const height = account.accountParams?.height || 0;
const weight = account.accountParams?.weight || 0; const weight = account.accountParams?.weight || 0;

View File

@@ -103,7 +103,6 @@ export default function OrderAnalysesCards({
{title} {title}
{description && ( {description && (
<> <>
{' '}
<InfoTooltip <InfoTooltip
content={ content={
<div className="flex flex-col gap-2"> <div className="flex flex-col gap-2">

View File

@@ -8,6 +8,11 @@ import { useTranslation } from 'react-i18next';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
const PaymentProviderIds = {
COMPANY_BENEFITS: "pp_company-benefits_company-benefits",
MONTONIO: "pp_montonio_montonio",
};
export default function CartTotals({ export default function CartTotals({
medusaOrder, medusaOrder,
}: { }: {
@@ -20,11 +25,16 @@ export default function CartTotals({
currency_code, currency_code,
total, total,
subtotal, subtotal,
tax_total,
discount_total, discount_total,
gift_card_total, gift_card_total,
payment_collections,
} = medusaOrder; } = medusaOrder;
const montonioPayment = payment_collections?.[0]?.payments
?.find(({ provider_id }) => provider_id === PaymentProviderIds.MONTONIO);
const companyBenefitsPayment = payment_collections?.[0]?.payments
?.find(({ provider_id }) => provider_id === PaymentProviderIds.COMPANY_BENEFITS);
return ( return (
<div> <div>
<div className="txt-medium text-ui-fg-subtle flex flex-col gap-y-2"> <div className="txt-medium text-ui-fg-subtle flex flex-col gap-y-2">
@@ -86,8 +96,11 @@ export default function CartTotals({
</span> </span>
</div> </div>
)} )}
</div> </div>
<div className="my-4 h-px w-full border-b border-gray-200" /> <div className="my-4 h-px w-full border-b border-gray-200" />
<div className="text-ui-fg-base txt-medium mb-2 flex items-center justify-between"> <div className="text-ui-fg-base txt-medium mb-2 flex items-center justify-between">
<span className="font-bold"> <span className="font-bold">
<Trans i18nKey="cart:order.total" /> <Trans i18nKey="cart:order.total" />
@@ -104,7 +117,42 @@ export default function CartTotals({
})} })}
</span> </span>
</div> </div>
<div className="mt-4 h-px w-full border-b border-gray-200" />
<div className="my-4 h-px w-full border-b border-gray-200" />
<div className="txt-medium text-ui-fg-subtle flex flex-col gap-y-2">
{companyBenefitsPayment && (
<div className="flex items-center justify-between">
<span className="flex items-center gap-x-1">
<Trans i18nKey="cart:order.benefitsTotal" />
</span>
<span data-testid="cart-subtotal" data-value={companyBenefitsPayment.amount || 0}>
-{' '}
{formatCurrency({
value: companyBenefitsPayment.amount ?? 0,
currencyCode: currency_code,
locale: language,
})}
</span>
</div>
)}
{montonioPayment && (
<div className="flex items-center justify-between">
<span className="flex items-center gap-x-1">
<Trans i18nKey="cart:order.montonioTotal" />
</span>
<span data-testid="cart-subtotal" data-value={montonioPayment.amount || 0}>
-{' '}
{formatCurrency({
value: montonioPayment.amount ?? 0,
currencyCode: currency_code,
locale: language,
})}
</span>
</div>
)}
</div>
</div> </div>
); );
} }

View File

@@ -2,16 +2,18 @@ import { formatDate } from 'date-fns';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import type { AnalysisOrder } from '~/lib/types/analysis-order'; export default function OrderDetails({
order,
export default function OrderDetails({ order }: { order: AnalysisOrder }) { }: {
order: { id: string; created_at: string | Date };
}) {
return ( return (
<div className="flex flex-col gap-y-2"> <div className="flex flex-col gap-y-2">
<div> <div>
<span className="font-bold"> <span className="font-bold">
<Trans i18nKey="cart:orderConfirmed.orderNumber" />:{' '} <Trans i18nKey="cart:orderConfirmed.orderNumber" />:{' '}
</span> </span>
<span>{order.medusa_order_id}</span> <span className="break-all">{order.id}</span>
</div> </div>
<div> <div>

View File

@@ -5,51 +5,77 @@ import { Eye } from 'lucide-react';
import { Trans } from '@kit/ui/makerkit/trans'; import { Trans } from '@kit/ui/makerkit/trans';
import type { AnalysisOrder } from '~/lib/types/analysis-order'; import type { AnalysisOrder } from '~/lib/types/order';
import OrderItemsTable from './order-items-table'; import OrderItemsTable from './order-items-table';
export default function OrderBlock({ export default function OrderBlock({
analysisOrder, analysisOrder,
medusaOrderStatus,
itemsAnalysisPackage, itemsAnalysisPackage,
itemsTtoService,
itemsOther, itemsOther,
medusaOrderId,
}: { }: {
analysisOrder: AnalysisOrder; analysisOrder?: AnalysisOrder;
medusaOrderStatus: string;
itemsAnalysisPackage: StoreOrderLineItem[]; itemsAnalysisPackage: StoreOrderLineItem[];
itemsTtoService: StoreOrderLineItem[];
itemsOther: StoreOrderLineItem[]; itemsOther: StoreOrderLineItem[];
medusaOrderId: string;
}) { }) {
return ( return (
<div className="flex flex-col gap-4"> <div className="flex flex-col gap-4">
<h4> <h4>
<Trans <Trans
i18nKey="analysis-results:orderTitle" i18nKey="analysis-results:orderTitle"
values={{ orderNumber: analysisOrder.medusa_order_id }} values={{ orderNumber: medusaOrderId }}
/> />
{` (${analysisOrder.id})`}
</h4> </h4>
<div className="flex gap-2"> {analysisOrder && (
<h5> <div className="flex gap-2">
<Trans i18nKey={`orders:status.${analysisOrder.status}`} /> <h5>
</h5> <Trans i18nKey={`orders:status.${analysisOrder.status}`} />
<Link </h5>
href={`/home/order/${analysisOrder.id}`} <Link
className="text-small-regular flex items-center justify-between" href={`/home/order/${analysisOrder.id}`}
> className="text-small-regular flex items-center justify-between"
<button className="text-ui-fg-subtle hover:text-ui-fg-base flex cursor-pointer gap-x-1"> >
<Eye /> <button className="text-ui-fg-subtle hover:text-ui-fg-base flex cursor-pointer gap-x-1">
</button> <Eye />
</Link> </button>
</div> </Link>
</div>
)}
<div className="flex flex-col gap-4"> <div className="flex flex-col gap-4">
<OrderItemsTable {analysisOrder && (
items={itemsAnalysisPackage} <OrderItemsTable
title="orders:table.analysisPackage" items={itemsAnalysisPackage}
analysisOrder={analysisOrder} title="orders:table.analysisPackage"
/> order={{
medusaOrderId: analysisOrder.medusa_order_id,
id: analysisOrder.id,
status: analysisOrder.status,
}}
/>
)}
{itemsTtoService && (
<OrderItemsTable
items={itemsTtoService}
title="orders:table.ttoService"
type="ttoService"
order={{
status: medusaOrderStatus.toUpperCase(),
medusaOrderId,
}}
/>
)}
<OrderItemsTable <OrderItemsTable
items={itemsOther} items={itemsOther}
title="orders:table.otherOrders" title="orders:table.otherOrders"
analysisOrder={analysisOrder} order={{
status: analysisOrder?.status,
}}
/> />
</div> </div>
</div> </div>

View File

@@ -4,7 +4,6 @@ import { useRouter } from 'next/navigation';
import { StoreOrderLineItem } from '@medusajs/types'; import { StoreOrderLineItem } from '@medusajs/types';
import { formatDate } from 'date-fns'; import { formatDate } from 'date-fns';
import { Eye } from 'lucide-react';
import { pathsConfig } from '@kit/shared/config'; import { pathsConfig } from '@kit/shared/config';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
@@ -18,18 +17,22 @@ import {
} from '@kit/ui/table'; } from '@kit/ui/table';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import type { AnalysisOrder } from '~/lib/types/analysis-order'; import type { Order } from '~/lib/types/order';
import { logAnalysisResultsNavigateAction } from './actions'; import { logAnalysisResultsNavigateAction } from './actions';
export type OrderItemType = 'analysisOrder' | 'ttoService';
export default function OrderItemsTable({ export default function OrderItemsTable({
items, items,
title, title,
analysisOrder, order,
type = 'analysisOrder',
}: { }: {
items: StoreOrderLineItem[]; items: StoreOrderLineItem[];
title: string; title: string;
analysisOrder: AnalysisOrder; order: Order;
type?: OrderItemType;
}) { }) {
const router = useRouter(); const router = useRouter();
@@ -37,9 +40,15 @@ export default function OrderItemsTable({
return null; return null;
} }
const openAnalysisResults = async () => { const isAnalysisOrder = type === 'analysisOrder';
await logAnalysisResultsNavigateAction(analysisOrder.medusa_order_id);
router.push(`${pathsConfig.app.analysisResults}/${analysisOrder.id}`); const openDetailedView = async () => {
if (isAnalysisOrder && order?.medusaOrderId && order?.id) {
await logAnalysisResultsNavigateAction(order.medusaOrderId);
router.push(`${pathsConfig.app.analysisResults}/${order.id}`);
} else {
router.push(`${pathsConfig.app.myOrders}/${order.medusaOrderId}`);
}
}; };
return ( return (
@@ -55,7 +64,7 @@ export default function OrderItemsTable({
<TableHead className="px-6"> <TableHead className="px-6">
<Trans i18nKey="orders:table.status" /> <Trans i18nKey="orders:table.status" />
</TableHead> </TableHead>
<TableHead className="px-6"></TableHead> {isAnalysisOrder && <TableHead className="px-6"></TableHead>}
</TableRow> </TableRow>
</TableHeader> </TableHeader>
<TableBody> <TableBody>
@@ -76,11 +85,13 @@ export default function OrderItemsTable({
</TableCell> </TableCell>
<TableCell className="min-w-[180px] px-6"> <TableCell className="min-w-[180px] px-6">
<Trans i18nKey={`orders:status.${analysisOrder.status}`} /> <Trans
i18nKey={`orders:status.${type}.${order?.status ?? 'CONFIRMED'}`}
/>
</TableCell> </TableCell>
<TableCell className="px-6 text-right"> <TableCell className="px-6 text-right">
<Button size="sm" onClick={openAnalysisResults}> <Button size="sm" onClick={openDetailedView}>
<Trans i18nKey="analysis-results:view" /> <Trans i18nKey="analysis-results:view" />
</Button> </Button>
</TableCell> </TableCell>

View File

@@ -4,17 +4,20 @@ import React from 'react';
import { redirect } from 'next/navigation'; import { redirect } from 'next/navigation';
import { createPath, pathsConfig } from '@/packages/shared/src/config'; import { pathsConfig } from '@/packages/shared/src/config';
import { StoreProduct } from '@medusajs/types';
import { ComponentInstanceIcon } from '@radix-ui/react-icons'; import { ComponentInstanceIcon } from '@radix-ui/react-icons';
import { cn } from '@kit/ui/shadcn'; import { cn } from '@kit/ui/shadcn';
import { Card, CardDescription, CardTitle } from '@kit/ui/shadcn/card'; import { Card, CardDescription } from '@kit/ui/shadcn/card';
export interface ServiceCategory { export interface ServiceCategory {
name: string; name: string;
handle: string; handle: string;
color: string; color: string;
description: string; description: string;
products: StoreProduct[];
countryCode: string;
} }
const ServiceCategories = ({ const ServiceCategories = ({

View File

@@ -0,0 +1,43 @@
'use server';
import { updateLineItem } from '@lib/data/cart';
import { StoreProductVariant } from '@medusajs/types';
import { handleAddToCart } from '~/lib/services/medusaCart.service';
import { createInitialReservation } from '~/lib/services/reservation.service';
export async function createInitialReservationAction(
selectedVariant: Pick<StoreProductVariant, 'id'>,
countryCode: string,
serviceId: number,
clinicId: number,
appointmentUserId: number,
syncUserId: number,
startTime: Date,
locationId: number | null,
comments?: string,
) {
const { addedItem } = await handleAddToCart({
selectedVariant,
countryCode,
});
if (addedItem) {
const reservation = await createInitialReservation({
serviceId,
clinicId,
appointmentUserId,
syncUserID: syncUserId,
startTime,
medusaLineItemId: addedItem.id,
locationId,
comments,
});
await updateLineItem({
lineId: addedItem.id,
quantity: addedItem.quantity,
metadata: { connectedOnlineReservationId: reservation.id },
});
}
}

View File

@@ -0,0 +1,13 @@
'use server';
import { AccountBalanceService, AccountBalanceSummary } from '@kit/accounts/services/account-balance.service';
export async function getAccountBalanceSummary(accountId: string): Promise<AccountBalanceSummary | null> {
try {
const service = new AccountBalanceService();
return await service.getBalanceSummary(accountId);
} catch (error) {
console.error('Error getting account balance summary:', error);
return null;
}
}

View File

@@ -0,0 +1,347 @@
'use server';
import { z } from 'zod';
import jwt from 'jsonwebtoken';
import type { StoreCart, StoreOrder } from "@medusajs/types";
import { initiateMultiPaymentSession, placeOrder } from "@lib/data/cart";
import type { AccountBalanceSummary } from "@kit/accounts/services/account-balance.service";
import { handleNavigateToPayment } from "~/lib/services/medusaCart.service";
import { loadCurrentUserAccount } from "./load-user-account";
import { getOrderedAnalysisIds } from "~/lib/services/medusaOrder.service";
import { createAnalysisOrder, getAnalysisOrder } from "~/lib/services/order.service";
import { listProductTypes } from "@lib/data";
import { sendOrderToMedipost } from "~/lib/services/medipost/medipostPrivateMessage.service";
import { AccountWithParams } from "@/packages/features/accounts/src/types/accounts";
import { createI18nServerInstance } from "~/lib/i18n/i18n.server";
import { getSupabaseServerAdminClient } from "@/packages/supabase/src/clients/server-admin-client";
import { createNotificationsApi } from "@/packages/features/notifications/src/server/api";
import { FailureReason } from '~/lib/types/connected-online';
import { getOrderedTtoServices } from '~/lib/services/reservation.service';
import { bookAppointment } from '~/lib/services/connected-online.service';
const ANALYSIS_PACKAGES_TYPE_HANDLE = 'analysis-packages';
const ANALYSIS_TYPE_HANDLE = 'synlab-analysis';
const env = () =>
z
.object({
emailSender: z
.string({
error: 'EMAIL_SENDER is required',
})
.min(1),
siteUrl: z
.string({
error: 'NEXT_PUBLIC_SITE_URL is required',
})
.min(1),
isEnabledDispatchOnMontonioCallback: z.boolean({
error: 'MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK is required',
}),
medusaBackendPublicUrl: z.string({
error: 'MEDUSA_BACKEND_PUBLIC_URL is required',
}).min(1),
companyBenefitsPaymentSecretKey: z.string({
error: 'COMPANY_BENEFITS_PAYMENT_SECRET_KEY is required',
}).min(1),
})
.parse({
emailSender: process.env.EMAIL_SENDER,
siteUrl: process.env.NEXT_PUBLIC_SITE_URL!,
isEnabledDispatchOnMontonioCallback:
process.env.MEDIPOST_ENABLE_DISPATCH_ON_MONTONIO_CALLBACK === 'true',
medusaBackendPublicUrl: process.env.MEDUSA_BACKEND_PUBLIC_URL!,
companyBenefitsPaymentSecretKey: process.env.COMPANY_BENEFITS_PAYMENT_SECRET_KEY!,
});
export const initiatePayment = async ({
accountId,
balanceSummary,
cart,
language,
}: {
accountId: string;
balanceSummary: AccountBalanceSummary;
cart: StoreCart;
language: string;
}) => {
try {
const {
montonioPaymentSessionId,
companyBenefitsPaymentSessionId,
totalByMontonio,
totalByBenefits,
isFullyPaidByBenefits,
} = await initiateMultiPaymentSession(cart, balanceSummary.totalBalance);
if (!isFullyPaidByBenefits) {
if (!montonioPaymentSessionId) {
throw new Error('Montonio payment session ID is missing');
}
const props = await handleNavigateToPayment({
language,
paymentSessionId: montonioPaymentSessionId,
amount: totalByMontonio,
currencyCode: cart.currency_code,
cartId: cart.id,
});
return { ...props, isFullyPaidByBenefits };
} else {
// place order if all paid already
const { orderId } = await handlePlaceOrder({ cart });
const companyBenefitsOrderToken = jwt.sign({
accountId,
companyBenefitsPaymentSessionId,
orderId,
totalByBenefits,
}, env().companyBenefitsPaymentSecretKey, {
algorithm: 'HS256',
});
const webhookResponse = await fetch(`${env().medusaBackendPublicUrl}/hooks/payment/company-benefits_company-benefits`, {
method: 'POST',
body: JSON.stringify({
orderToken: companyBenefitsOrderToken,
}),
headers: {
'Content-Type': 'application/json',
},
});
if (!webhookResponse.ok) {
throw new Error('Failed to send company benefits webhook');
}
return { isFullyPaidByBenefits, orderId, unavailableLineItemIds: [] };
}
} catch (error) {
console.error('Error initiating payment', error);
}
return { url: null, isFullyPaidByBenefits: false, orderId: null, unavailableLineItemIds: [] };
}
export async function handlePlaceOrder({
cart,
}: {
cart: StoreCart;
}) {
const { account } = await loadCurrentUserAccount();
if (!account) {
throw new Error('Account not found in context');
}
try {
const medusaOrder = await placeOrder(cart.id, {
revalidateCacheTags: false,
});
const orderedAnalysisElements = await getOrderedAnalysisIds({
medusaOrder,
});
const orderContainsSynlabItems = !!orderedAnalysisElements?.length;
try {
const existingAnalysisOrder = await getAnalysisOrder({
medusaOrderId: medusaOrder.id,
});
console.info(
`Analysis order already exists for medusaOrderId=${medusaOrder.id}, orderId=${existingAnalysisOrder.id}`,
);
return { success: true, orderId: existingAnalysisOrder.id };
} catch {
// ignored
}
let orderId: number | undefined = undefined;
if (orderContainsSynlabItems) {
orderId = await createAnalysisOrder({
medusaOrder,
orderedAnalysisElements,
});
}
const orderResult = await getOrderResultParameters(medusaOrder);
const { medusaOrderId, email, analysisPackageOrder, analysisItemsOrder } =
orderResult;
const orderedTtoServices = await getOrderedTtoServices({ medusaOrder });
let bookServiceResults: {
success: boolean;
reason?: FailureReason;
serviceId?: number;
}[] = [];
if (orderedTtoServices?.length) {
const bookingPromises = orderedTtoServices.map((service) =>
bookAppointment(
service.service_id,
service.clinic_id,
service.service_user_id,
service.sync_user_id,
service.start_time,
),
);
bookServiceResults = await Promise.all(bookingPromises);
}
if (email) {
if (analysisPackageOrder) {
await sendAnalysisPackageOrderEmail({
account,
email,
analysisPackageOrder,
});
} else {
console.info(`Order has no analysis package, skipping email.`);
}
if (analysisItemsOrder) {
// @TODO send email for separate analyses
console.warn(
`Order has analysis items, but no email template exists yet`,
);
} else {
console.info(`Order has no analysis items, skipping email.`);
}
} else {
console.error('Missing email to send order result email', orderResult);
}
if (env().isEnabledDispatchOnMontonioCallback) {
await sendOrderToMedipost({ medusaOrderId, orderedAnalysisElements });
}
if (bookServiceResults.some(({ success }) => success === false)) {
const failedServiceBookings = bookServiceResults.filter(
({ success }) => success === false,
);
return {
success: false,
failedServiceBookings,
orderId,
};
}
return { success: true, orderId };
} catch (error) {
console.error('Failed to place order', error);
throw new Error(`Failed to place order, message=${error}`);
}
}
async function sendAnalysisPackageOrderEmail({
account,
email,
analysisPackageOrder,
}: {
account: AccountWithParams;
email: string;
analysisPackageOrder: {
partnerLocationName: string;
analysisPackageName: string;
};
}) {
const { language } = await createI18nServerInstance();
const { analysisPackageName, partnerLocationName } = analysisPackageOrder;
try {
await sendEmail({
account: { id: account.id, name: account.name },
email,
analysisPackageName,
partnerLocationName,
language,
});
console.info(`Successfully sent analysis package order email to ${email}`);
} catch (error) {
console.error(
`Failed to send analysis package order email to ${email}`,
error,
);
}
}
async function getOrderResultParameters(medusaOrder: StoreOrder) {
const { productTypes } = await listProductTypes();
const analysisPackagesType = productTypes.find(
({ metadata }) => metadata?.handle === ANALYSIS_PACKAGES_TYPE_HANDLE,
);
const analysisType = productTypes.find(
({ metadata }) => metadata?.handle === ANALYSIS_TYPE_HANDLE,
);
const analysisPackageOrderItem = medusaOrder.items?.find(
({ product_type_id }) => product_type_id === analysisPackagesType?.id,
);
const analysisItems = medusaOrder.items?.filter(
({ product_type_id }) => product_type_id === analysisType?.id,
);
return {
medusaOrderId: medusaOrder.id,
email: medusaOrder.email,
analysisPackageOrder: analysisPackageOrderItem
? {
partnerLocationName:
(analysisPackageOrderItem?.metadata
?.partner_location_name as string) ?? '',
analysisPackageName: analysisPackageOrderItem?.title ?? '',
}
: null,
analysisItemsOrder:
Array.isArray(analysisItems) && analysisItems.length > 0
? analysisItems.map(({ product }) => ({
analysisName: product?.title ?? '',
analysisId: (product?.metadata?.analysisIdOriginal as string) ?? '',
}))
: null,
};
}
const sendEmail = async ({
account,
email,
analysisPackageName,
partnerLocationName,
language,
}: {
account: Pick<AccountWithParams, 'name' | 'id'>;
email: string;
analysisPackageName: string;
partnerLocationName: string;
language: string;
}) => {
const client = getSupabaseServerAdminClient();
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: account.name,
partnerLocationName,
language,
});
await mailer
.sendEmail({
from: env().emailSender,
to: email,
subject,
html,
})
.catch((error) => {
throw new Error(`Failed to send email, message=${error}`);
});
await createNotificationsApi(client).createNotification({
account_id: account.id,
body: html,
});
} catch (error) {
throw new Error(`Failed to send email, message=${error}`);
}
};

View File

@@ -0,0 +1,12 @@
import OpenAI from 'openai';
export const isValidOpenAiEnv = async () => {
try {
const client = new OpenAI();
await client.models.list();
return true;
} catch (e) {
console.log('No openAI env');
return false;
}
};

View File

@@ -45,10 +45,6 @@ async function analysesLoader() {
}) })
: null; : null;
const serviceCategories = productCategories.filter(
({ parent_category }) => parent_category?.handle === 'tto-categories',
);
return { return {
analyses: analyses:
categoryProducts?.response.products categoryProducts?.response.products

View File

@@ -1,20 +1,30 @@
import { cache } from 'react'; import { cache } from 'react';
import { getProductCategories } from '@lib/data'; import { getProductCategories, listProducts } from '@lib/data';
import { ServiceCategory } from '../../_components/service-categories'; import { loadCountryCodes } from './load-analyses';
async function categoryLoader({
handle,
}: {
handle: string;
}): Promise<{ category: ServiceCategory | null }> {
const response = await getProductCategories({
handle,
fields: '*products, is_active, metadata',
});
async function categoryLoader({ handle }: { handle: string }) {
const [response, countryCodes] = await Promise.all([
getProductCategories({
handle,
limit: 1,
}),
loadCountryCodes(),
]);
const category = response.product_categories[0]; const category = response.product_categories[0];
const countryCode = countryCodes[0]!;
if (!response.product_categories?.[0]?.id) {
return { category: null };
}
const {
response: { products: categoryProducts },
} = await listProducts({
countryCode,
queryParams: { limit: 100, category_id: response.product_categories[0].id },
});
return { return {
category: { category: {
@@ -25,6 +35,8 @@ async function categoryLoader({
description: category?.description || '', description: category?.description || '',
handle: category?.handle || '', handle: category?.handle || '',
name: category?.name || '', name: category?.name || '',
countryCode,
products: categoryProducts,
}, },
}; };
} }

View File

@@ -10,38 +10,36 @@ async function ttoServicesLoader() {
}); });
const heroCategories = response.product_categories?.filter( const heroCategories = response.product_categories?.filter(
({ parent_category, is_active, metadata }) => ({ parent_category, metadata }) =>
parent_category?.handle === 'tto-categories' && parent_category?.handle === 'tto-categories' && metadata?.isHero,
is_active &&
metadata?.isHero,
); );
const ttoCategories = response.product_categories?.filter( const ttoCategories = response.product_categories?.filter(
({ parent_category, is_active, metadata }) => ({ parent_category, metadata }) =>
parent_category?.handle === 'tto-categories' && parent_category?.handle === 'tto-categories' && !metadata?.isHero,
is_active &&
!metadata?.isHero,
); );
return { return {
heroCategories: heroCategories:
heroCategories.map<ServiceCategory>( heroCategories.map<Omit<ServiceCategory, 'countryCode'>>(
({ name, handle, metadata, description }) => ({ ({ name, handle, metadata, description, products }) => ({
name, name,
handle, handle,
color: color:
typeof metadata?.color === 'string' ? metadata.color : 'primary', typeof metadata?.color === 'string' ? metadata.color : 'primary',
description, description,
products: products ?? [],
}), }),
) ?? [], ) ?? [],
ttoCategories: ttoCategories:
ttoCategories.map<ServiceCategory>( ttoCategories.map<Omit<ServiceCategory, 'countryCode'>>(
({ name, handle, metadata, description }) => ({ ({ name, handle, metadata, description, products }) => ({
name, name,
handle, handle,
color: color:
typeof metadata?.color === 'string' ? metadata.color : 'primary', typeof metadata?.color === 'string' ? metadata.color : 'primary',
description, description,
products: products ?? [],
}), }),
) ?? [], ) ?? [],
}; };

View File

@@ -1,23 +1,14 @@
import React from 'react'; import React from 'react';
import { redirect } from 'next/navigation';
import { formatCurrency } from '@/packages/shared/src/utils'; import { formatCurrency } from '@/packages/shared/src/utils';
import { Database } from '@/packages/supabase/src/database.types';
import { PiggyBankIcon, Settings } from 'lucide-react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { createPath, pathsConfig } from '@kit/shared/config';
import { Card, CardTitle } from '@kit/ui/card'; import { Card, CardTitle } from '@kit/ui/card';
import { cn } from '@kit/ui/lib/utils'; import { cn } from '@kit/ui/lib/utils';
import { Button } from '@kit/ui/shadcn/button';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
interface TeamAccountBenefitStatisticsProps { import { TeamAccountBenefitExpensesOverview } from '../_lib/server/load-team-account-benefit-expenses-overview';
employeeCount: number; import { AccountBenefitStatistics } from '../_lib/server/load-team-account-benefit-statistics';
accountSlug: string;
companyParams: Database['medreport']['Tables']['company_params']['Row'];
}
const StatisticsCard = ({ children }: { children: React.ReactNode }) => { const StatisticsCard = ({ children }: { children: React.ReactNode }) => {
return <Card className="p-4">{children}</Card>; return <Card className="p-4">{children}</Card>;
@@ -46,126 +37,90 @@ const StatisticsValue = ({ children }: { children: React.ReactNode }) => {
}; };
const TeamAccountBenefitStatistics = ({ const TeamAccountBenefitStatistics = ({
employeeCount, accountBenefitStatistics,
accountSlug, expensesOverview,
companyParams, }: {
}: TeamAccountBenefitStatisticsProps) => { accountBenefitStatistics: AccountBenefitStatistics;
expensesOverview: TeamAccountBenefitExpensesOverview;
}) => {
const { const {
i18n: { language }, i18n: { language },
} = useTranslation(); } = useTranslation();
return ( return (
<div className="flex h-full w-full flex-col gap-2 sm:flex-row"> <div className="flex h-full w-full flex-col gap-2 sm:flex-row">
<Card className="relative flex flex-row">
<div className="p-6">
<Button
onClick={() =>
redirect(createPath(pathsConfig.app.accountBilling, accountSlug))
}
variant="outline"
className="absolute top-1 right-1 p-3"
>
<Settings />
</Button>
<div className="flex h-16 w-16 items-center justify-center rounded-full bg-orange-100">
<PiggyBankIcon className="h-[32px] w-[32px] stroke-orange-400 stroke-2" />
</div>
<p className="mt-4 text-sm font-medium">
<Trans i18nKey="teams:benefitStatistics.budget.title" />
</p>
<h3 className="text-2xl">
<Trans
i18nKey="teams:benefitStatistics.budget.balance"
values={{
balance: formatCurrency({
value: 11800,
locale: language,
currencyCode: 'EUR',
}),
}}
/>
</h3>
<StatisticsDescription>
<Trans
i18nKey="teams:benefitStatistics.budget.volume"
values={{
volume: formatCurrency({
value:
(Number(companyParams.benefit_amount) || 0) * employeeCount,
locale: language,
currencyCode: 'EUR',
}),
}}
/>
</StatisticsDescription>
</div>
</Card>
<div className="grid flex-2 grid-cols-2 gap-2 sm:grid-cols-3 sm:grid-rows-2"> <div className="grid flex-2 grid-cols-2 gap-2 sm:grid-cols-3 sm:grid-rows-2">
<StatisticsCard> <StatisticsCard>
<StatisticsCardTitle className="text-lg font-bold"> <StatisticsCardTitle className="text-lg font-bold">
<Trans i18nKey="teams:benefitStatistics.data.serviceSum" /> <Trans i18nKey="teams:benefitStatistics.budget.membersCount" />
</StatisticsCardTitle> </StatisticsCardTitle>
<StatisticsValue>1800 </StatisticsValue> <StatisticsValue>
{accountBenefitStatistics.companyAccountsCount}
</StatisticsValue>
</StatisticsCard> </StatisticsCard>
<StatisticsCard>
<StatisticsCardTitle className="text-lg font-bold">
<Trans i18nKey="teams:benefitStatistics.data.totalSum" />
</StatisticsCardTitle>
<StatisticsValue>
{formatCurrency({
value: accountBenefitStatistics.orders.totalSum,
locale: language,
currencyCode: 'EUR',
})}
</StatisticsValue>
</StatisticsCard>
<StatisticsCard>
<StatisticsCardTitle className="text-lg font-bold">
<Trans i18nKey="teams:benefitStatistics.data.currentMonthUsageTotal" />
</StatisticsCardTitle>
<StatisticsValue>
{formatCurrency({
value: expensesOverview.currentMonthUsageTotal,
locale: language,
currencyCode: 'EUR',
})}
</StatisticsValue>
</StatisticsCard>
<StatisticsCard> <StatisticsCard>
<StatisticsCardTitle> <StatisticsCardTitle>
<Trans i18nKey="teams:benefitStatistics.data.analysis" /> <Trans i18nKey="teams:benefitStatistics.data.analysis" />
</StatisticsCardTitle> </StatisticsCardTitle>
<StatisticsValue>200 </StatisticsValue> <StatisticsValue>
{formatCurrency({
value: accountBenefitStatistics.orders.analysesSum,
locale: language,
currencyCode: 'EUR',
})}
</StatisticsValue>
<StatisticsDescription> <StatisticsDescription>
<Trans <Trans
i18nKey="teams:benefitStatistics.data.reservations" i18nKey="teams:benefitStatistics.data.reservations"
values={{ value: 36 }} values={{ value: accountBenefitStatistics.orders.analysesCount }}
/>
</StatisticsDescription>
</StatisticsCard>
<StatisticsCard>
<StatisticsCardTitle>
<Trans i18nKey="teams:benefitStatistics.data.doctorsAndSpecialists" />
</StatisticsCardTitle>
<StatisticsValue>200 </StatisticsValue>
<StatisticsDescription>
<Trans
i18nKey="teams:benefitStatistics.data.reservations"
values={{ value: 44 }}
/>
</StatisticsDescription>
</StatisticsCard>
<StatisticsCard>
<StatisticsCardTitle>
<Trans i18nKey="teams:benefitStatistics.data.researches" />
</StatisticsCardTitle>
<StatisticsValue>200 </StatisticsValue>
<StatisticsDescription>
<Trans
i18nKey="teams:benefitStatistics.data.reservations"
values={{ value: 40 }}
/>
</StatisticsDescription>
</StatisticsCard>
<StatisticsCard>
<StatisticsCardTitle>
<Trans i18nKey="teams:benefitStatistics.data.eclinic" />
</StatisticsCardTitle>
<StatisticsValue>200 </StatisticsValue>
<StatisticsDescription>
<Trans
i18nKey="teams:benefitStatistics.data.reservations"
values={{ value: 34 }}
/> />
</StatisticsDescription> </StatisticsDescription>
</StatisticsCard> </StatisticsCard>
<StatisticsCard> <StatisticsCard>
<StatisticsCardTitle> <StatisticsCardTitle>
<Trans i18nKey="teams:benefitStatistics.data.healthResearchPlans" /> <Trans i18nKey="teams:benefitStatistics.data.analysisPackages" />
</StatisticsCardTitle> </StatisticsCardTitle>
<StatisticsValue>200 </StatisticsValue> <StatisticsValue>
{formatCurrency({
value: accountBenefitStatistics.orders.analysisPackagesSum,
locale: language,
currencyCode: 'EUR',
})}
</StatisticsValue>
<StatisticsDescription> <StatisticsDescription>
<Trans <Trans
i18nKey="teams:benefitStatistics.data.serviceUsage" i18nKey="teams:benefitStatistics.data.analysisPackagesCount"
values={{ value: 46 }} values={{
value: accountBenefitStatistics.orders.analysisPackagesCount,
}}
/> />
</StatisticsDescription> </StatisticsDescription>
</StatisticsCard> </StatisticsCard>

View File

@@ -5,6 +5,7 @@ import { Database } from '@/packages/supabase/src/database.types';
import { Card } from '@kit/ui/card'; import { Card } from '@kit/ui/card';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { cn } from '@kit/ui/utils'; import { cn } from '@kit/ui/utils';
import type { BmiThresholds } from '@kit/accounts/types/accounts';
import { getAccountHealthDetailsFields } from '../_lib/server/load-team-account-health-details'; import { getAccountHealthDetailsFields } from '../_lib/server/load-team-account-health-details';
import { TeamAccountStatisticsProps } from './team-account-statistics'; import { TeamAccountStatisticsProps } from './team-account-statistics';
@@ -15,10 +16,7 @@ const TeamAccountHealthDetails = ({
members, members,
}: { }: {
memberParams: TeamAccountStatisticsProps['memberParams']; memberParams: TeamAccountStatisticsProps['memberParams'];
bmiThresholds: Omit< bmiThresholds: Omit<BmiThresholds, 'id'>[];
Database['medreport']['Tables']['bmi_thresholds']['Row'],
'id'
>[];
members: Database['medreport']['Functions']['get_account_members']['Returns']; members: Database['medreport']['Functions']['get_account_members']['Returns'];
}) => { }) => {
const accountHealthDetailsFields = getAccountHealthDetailsFields( const accountHealthDetailsFields = getAccountHealthDetailsFields(

View File

@@ -52,7 +52,6 @@ function SidebarContainer(props: {
<SidebarContent> <SidebarContent>
<SidebarNavigation config={config} /> <SidebarNavigation config={config} />
</SidebarContent> </SidebarContent>
</Sidebar> </Sidebar>
); );
} }

View File

@@ -14,28 +14,21 @@ import { createPath, pathsConfig } from '@kit/shared/config';
import { Card } from '@kit/ui/card'; import { Card } from '@kit/ui/card';
import { Trans } from '@kit/ui/makerkit/trans'; import { Trans } from '@kit/ui/makerkit/trans';
import { Button } from '@kit/ui/shadcn/button'; import { Button } from '@kit/ui/shadcn/button';
import { Calendar, DateRange } from '@kit/ui/shadcn/calendar'; import { DateRange } from '@kit/ui/shadcn/calendar';
import {
Popover,
PopoverContent,
PopoverTrigger,
} from '@kit/ui/shadcn/popover';
import { AccountBenefitStatistics } from '../_lib/server/load-team-account-benefit-statistics';
import TeamAccountBenefitStatistics from './team-account-benefit-statistics'; import TeamAccountBenefitStatistics from './team-account-benefit-statistics';
import TeamAccountHealthDetails from './team-account-health-details'; import TeamAccountHealthDetails from './team-account-health-details';
import type { Account, AccountParams, BmiThresholds } from '@/packages/features/accounts/src/types/accounts';
import { TeamAccountBenefitExpensesOverview } from '../_lib/server/load-team-account-benefit-expenses-overview';
export interface TeamAccountStatisticsProps { export interface TeamAccountStatisticsProps {
teamAccount: Database['medreport']['Tables']['accounts']['Row']; teamAccount: Account;
memberParams: Pick< memberParams: Pick<AccountParams, 'weight' | 'height'>[];
Database['medreport']['Tables']['account_params']['Row'], bmiThresholds: Omit<BmiThresholds, 'id'>[];
'weight' | 'height'
>[];
bmiThresholds: Omit<
Database['medreport']['Tables']['bmi_thresholds']['Row'],
'id'
>[];
members: Database['medreport']['Functions']['get_account_members']['Returns']; members: Database['medreport']['Functions']['get_account_members']['Returns'];
companyParams: Database['medreport']['Tables']['company_params']['Row']; accountBenefitStatistics: AccountBenefitStatistics;
expensesOverview: TeamAccountBenefitExpensesOverview;
} }
export default function TeamAccountStatistics({ export default function TeamAccountStatistics({
@@ -43,11 +36,13 @@ export default function TeamAccountStatistics({
memberParams, memberParams,
bmiThresholds, bmiThresholds,
members, members,
companyParams, accountBenefitStatistics,
expensesOverview,
}: TeamAccountStatisticsProps) { }: TeamAccountStatisticsProps) {
const currentDate = new Date();
const [date, setDate] = useState<DateRange | undefined>({ const [date, setDate] = useState<DateRange | undefined>({
from: new Date(), from: new Date(currentDate.getFullYear(), currentDate.getMonth(), 1),
to: new Date(), to: new Date(currentDate.getFullYear(), currentDate.getMonth() + 1, 0),
}); });
const { const {
i18n: { language }, i18n: { language },
@@ -58,7 +53,7 @@ export default function TeamAccountStatistics({
return ( return (
<> <>
<div className="mt-4 flex items-center justify-between"> <div className="mt-4 flex flex-col gap-4 sm:gap-0 sm:flex-row items-center justify-between">
<h4 className="font-bold"> <h4 className="font-bold">
<Trans <Trans
i18nKey={'teams:home.headerTitle'} i18nKey={'teams:home.headerTitle'}
@@ -66,28 +61,16 @@ export default function TeamAccountStatistics({
/> />
</h4> </h4>
<Popover> <Button variant="outline" data-empty={!date}>
<PopoverTrigger asChild> <CalendarIcon />
<Button variant="outline" data-empty={!date}> {date?.from && date?.to ? (
<CalendarIcon /> `${format(date.from, 'd MMMM yyyy', dateFormatOptions)} - ${format(date.to, 'd MMMM yyyy', dateFormatOptions)}`
{date?.from && date?.to ? ( ) : (
`${format(date.from, 'd MMMM yyyy', dateFormatOptions)} - ${format(date.to, 'd MMMM yyyy', dateFormatOptions)}` <span>
) : ( <Trans i18nKey="common:formField.selectDate" />
<span> </span>
<Trans i18nKey="common:formField.selectDate" /> )}
</span> </Button>
)}
</Button>
</PopoverTrigger>
<PopoverContent className="w-auto p-0">
<Calendar
mode="range"
selected={date}
onSelect={setDate}
locale={language === 'et' ? et : enGB}
/>
</PopoverContent>
</Popover>
</div> </div>
<div <div
@@ -95,11 +78,7 @@ export default function TeamAccountStatistics({
'animate-in fade-in flex flex-col space-y-4 pb-36 duration-500' 'animate-in fade-in flex flex-col space-y-4 pb-36 duration-500'
} }
> >
<TeamAccountBenefitStatistics <TeamAccountBenefitStatistics accountBenefitStatistics={accountBenefitStatistics} expensesOverview={expensesOverview} />
employeeCount={members.length}
accountSlug={teamAccount.slug || ''}
companyParams={companyParams}
/>
<h5 className="mt-4 mb-2"> <h5 className="mt-4 mb-2">
<Trans i18nKey="teams:home.healthDetails" /> <Trans i18nKey="teams:home.healthDetails" />
@@ -148,7 +127,7 @@ export default function TeamAccountStatistics({
redirect( redirect(
createPath( createPath(
pathsConfig.app.accountBilling, pathsConfig.app.accountBilling,
teamAccount.slug || '', teamAccount.slug!,
), ),
) )
} }

View File

@@ -0,0 +1,75 @@
import { getSupabaseServerClient } from "@/packages/supabase/src/clients/server-client";
import { loadCompanyPersonalAccountsBalanceEntries } from "./load-team-account-benefit-statistics";
export interface TeamAccountBenefitExpensesOverview {
benefitAmount: number | null;
benefitOccurrence: 'yearly' | 'monthly' | 'quarterly' | null;
currentMonthUsageTotal: number;
managementFee: number;
managementFeeTotal: number;
total: number;
}
const MANAGEMENT_FEE = 5.50;
const MONTHS = 12;
const QUARTERS = 4;
export async function loadTeamAccountBenefitExpensesOverview({
companyId,
employeeCount,
}: {
companyId: string;
employeeCount: number;
}): Promise<TeamAccountBenefitExpensesOverview> {
const supabase = getSupabaseServerClient();
const { data, error } = await supabase
.schema('medreport')
.from('benefit_distribution_schedule')
.select('*')
.eq('company_id', companyId)
.eq('is_active', true)
.single();
let benefitAmount: TeamAccountBenefitExpensesOverview['benefitAmount'] = null;
let benefitOccurrence: TeamAccountBenefitExpensesOverview['benefitOccurrence'] = null;
if (error) {
console.warn('Failed to load team account benefit expenses overview');
} else {
benefitAmount = data.benefit_amount as TeamAccountBenefitExpensesOverview['benefitAmount'];
benefitOccurrence = data.benefit_occurrence as TeamAccountBenefitExpensesOverview['benefitOccurrence'];
}
const { purchaseEntriesTotal } = await loadCompanyPersonalAccountsBalanceEntries({ accountId: companyId });
return {
benefitAmount,
benefitOccurrence,
currentMonthUsageTotal: purchaseEntriesTotal,
managementFee: MANAGEMENT_FEE,
managementFeeTotal: MANAGEMENT_FEE * employeeCount,
total: (() => {
if (typeof benefitAmount !== 'number') {
return 0;
}
const currentDate = new Date();
const createdAt = new Date(data.created_at);
const isCreatedThisYear = createdAt.getFullYear() === currentDate.getFullYear();
if (benefitOccurrence === 'yearly') {
return benefitAmount * employeeCount;
} else if (benefitOccurrence === 'monthly') {
const monthsLeft = isCreatedThisYear
? MONTHS - createdAt.getMonth()
: MONTHS;
return benefitAmount * employeeCount * monthsLeft;
} else if (benefitOccurrence === 'quarterly') {
const quartersLeft = isCreatedThisYear
? QUARTERS - Math.ceil(createdAt.getMonth() / 3)
: QUARTERS;
return benefitAmount * employeeCount * quartersLeft;
}
return 0;
})(),
}
}

View File

@@ -0,0 +1,95 @@
'use server';
import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client';
export interface AccountBenefitStatistics {
benefitDistributionSchedule: {
amount: number;
};
companyAccountsCount: number;
periodTotal: number;
orders: {
totalSum: number;
analysesCount: number;
analysesSum: number;
analysisPackagesCount: number;
analysisPackagesSum: number;
}
}
export const loadCompanyPersonalAccountsBalanceEntries = async ({
accountId,
}: {
accountId: string;
}) => {
const supabase = getSupabaseServerAdminClient();
const { count, data: accountMemberships } = await supabase
.schema('medreport')
.from('accounts_memberships')
.select('user_id', { count: 'exact' })
.eq('account_id', accountId)
.throwOnError();
const { data: accountBalanceEntries } = await supabase
.schema('medreport')
.from('account_balance_entries')
.select('*')
.eq('is_active', true)
.in('account_id', accountMemberships.map(({ user_id }) => user_id))
.throwOnError();
const purchaseEntries = accountBalanceEntries.filter(({ entry_type }) => entry_type === 'purchase');
const analysesEntries = purchaseEntries.filter(({ is_analysis_order }) => is_analysis_order);
const analysisPackagesEntries = purchaseEntries.filter(({ is_analysis_package_order }) => is_analysis_package_order);
return {
accountBalanceEntries,
analysesEntries,
analysisPackagesEntries,
companyAccountsCount: count || 0,
purchaseEntries,
purchaseEntriesTotal: purchaseEntries.reduce((acc, { amount }) => acc + Math.abs(amount || 0), 0),
};
}
export const loadAccountBenefitStatistics = async (
accountId: string,
): Promise<AccountBenefitStatistics> => {
const supabase = getSupabaseServerAdminClient();
const {
analysesEntries,
analysisPackagesEntries,
companyAccountsCount,
purchaseEntriesTotal,
} = await loadCompanyPersonalAccountsBalanceEntries({ accountId });
const { data: benefitDistributionSchedule } = await supabase
.schema('medreport')
.from('benefit_distribution_schedule')
.select('*')
.eq('company_id', accountId)
.eq('is_active', true)
.single();
const scheduleAmount = benefitDistributionSchedule?.benefit_amount || 0;
return {
companyAccountsCount,
benefitDistributionSchedule: {
amount: benefitDistributionSchedule?.benefit_amount || 0,
},
periodTotal: scheduleAmount * companyAccountsCount,
orders: {
totalSum: purchaseEntriesTotal,
analysesCount: analysesEntries.length,
analysesSum: analysesEntries.reduce((acc, { amount }) => acc + Math.abs(amount || 0), 0),
analysisPackagesCount: analysisPackagesEntries.length,
analysisPackagesSum: analysisPackagesEntries.reduce((acc, { amount }) => acc + Math.abs(amount || 0), 0),
},
};
};

View File

@@ -11,6 +11,7 @@ import {
} from '~/lib/utils'; } from '~/lib/utils';
import { TeamAccountStatisticsProps } from '../../_components/team-account-statistics'; import { TeamAccountStatisticsProps } from '../../_components/team-account-statistics';
import type { BmiThresholds } from '@kit/accounts/types/accounts';
interface AccountHealthDetailsField { interface AccountHealthDetailsField {
title: string; title: string;
@@ -25,10 +26,7 @@ interface AccountHealthDetailsField {
export const getAccountHealthDetailsFields = ( export const getAccountHealthDetailsFields = (
memberParams: TeamAccountStatisticsProps['memberParams'], memberParams: TeamAccountStatisticsProps['memberParams'],
bmiThresholds: Omit< bmiThresholds: Omit<BmiThresholds, 'id'>[],
Database['medreport']['Tables']['bmi_thresholds']['Row'],
'id'
>[],
members: Database['medreport']['Functions']['get_account_members']['Returns'], members: Database['medreport']['Functions']['get_account_members']['Returns'],
): AccountHealthDetailsField[] => { ): AccountHealthDetailsField[] => {
const averageWeight = const averageWeight =
@@ -82,7 +80,7 @@ export const getAccountHealthDetailsFields = (
}, },
{ {
title: 'teams:healthDetails.bmi', title: 'teams:healthDetails.bmi',
value: averageBMI, value: averageBMI!,
Icon: TrendingUp, Icon: TrendingUp,
iconBg: getBmiBackgroundColor(bmiStatus), iconBg: getBmiBackgroundColor(bmiStatus),
}, },

View File

@@ -30,7 +30,9 @@ const HealthBenefitFields = () => {
<Select {...field} onValueChange={field.onChange}> <Select {...field} onValueChange={field.onChange}>
<SelectTrigger> <SelectTrigger>
<SelectValue <SelectValue
placeholder={<Trans i18nKey="common:formField:occurrence" />} placeholder={
<Trans i18nKey="common:formField:occurrence" />
}
/> />
</SelectTrigger> </SelectTrigger>

View File

@@ -0,0 +1,100 @@
'use client';
import { useState } from 'react';
import { useRouter } from 'next/navigation';
import { UpdateHealthBenefitSchema } from '@/packages/billing/core/src/schema';
import { zodResolver } from '@hookform/resolvers/zod';
import { useForm } from 'react-hook-form';
import { Button } from '@kit/ui/button';
import { Form } from '@kit/ui/form';
import { Spinner } from '@kit/ui/makerkit/spinner';
import { toast } from '@kit/ui/shadcn/sonner';
import { Trans } from '@kit/ui/trans';
import { cn } from '~/lib/utils';
import { updateHealthBenefit } from '../_lib/server/server-actions';
import HealthBenefitFields from './health-benefit-fields';
import { Account, CompanyParams } from '@/packages/features/accounts/src/types/accounts';
import { useTranslation } from 'react-i18next';
const HealthBenefitFormClient = ({
account,
companyParams,
}: {
account: Account;
companyParams: CompanyParams;
}) => {
const { t } = useTranslation('account');
const router = useRouter();
const [currentCompanyParams, setCurrentCompanyParams] =
useState<CompanyParams>(companyParams);
const [isLoading, setIsLoading] = useState<boolean>(false);
const form = useForm({
resolver: zodResolver(UpdateHealthBenefitSchema),
mode: 'onChange',
defaultValues: {
occurrence: currentCompanyParams.benefit_occurance || 'yearly',
amount: currentCompanyParams.benefit_amount || 0,
},
});
const isDirty = form.formState.isDirty;
const onSubmit = (data: { occurrence: string; amount: number }) => {
const promise = async () => {
setIsLoading(true);
try {
await updateHealthBenefit({ ...data, accountId: account.id });
setCurrentCompanyParams((prev) => ({
...prev,
benefit_amount: data.amount,
benefit_occurance: data.occurrence,
}));
} finally {
form.reset(data);
setIsLoading(false);
router.refresh();
}
};
toast.promise(promise, {
success: t('account:healthBenefitForm.updateSuccess'),
error: 'error',
});
};
return (
<Form {...form}>
<form
className="flex flex-col gap-6"
onSubmit={form.handleSubmit(onSubmit)}
>
<HealthBenefitFields />
<Button
type="submit"
className="relative"
disabled={!isDirty || isLoading}
>
{isLoading && (
<div className="absolute inset-0 flex items-center justify-center">
<Spinner />
</div>
)}
<div className={cn({ invisible: isLoading })}>
<Trans i18nKey="account:saveChanges" />
</div>
</Button>
</form>
</Form>
);
};
export default HealthBenefitFormClient;

View File

@@ -1,138 +1,70 @@
'use client';
import { useState } from 'react';
import { UpdateHealthBenefitSchema } from '@/packages/billing/core/src/schema';
import { Database } from '@/packages/supabase/src/database.types';
import { zodResolver } from '@hookform/resolvers/zod';
import { PiggyBankIcon } from 'lucide-react'; import { PiggyBankIcon } from 'lucide-react';
import { useForm } from 'react-hook-form';
import { Button } from '@kit/ui/button';
import { Form } from '@kit/ui/form';
import { Spinner } from '@kit/ui/makerkit/spinner';
import { Separator } from '@kit/ui/shadcn/separator'; import { Separator } from '@kit/ui/shadcn/separator';
import { toast } from '@kit/ui/shadcn/sonner';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { cn } from '~/lib/utils'; import HealthBenefitFormClient from './health-benefit-form-client';
import { updateHealthBenefit } from '../_lib/server/server-actions';
import HealthBenefitFields from './health-benefit-fields';
import YearlyExpensesOverview from './yearly-expenses-overview'; import YearlyExpensesOverview from './yearly-expenses-overview';
import { TeamAccountBenefitExpensesOverview } from '../../_lib/server/load-team-account-benefit-expenses-overview';
import { Account, CompanyParams } from '@/packages/features/accounts/src/types/accounts';
const HealthBenefitForm = ({ const HealthBenefitForm = async ({
account, account,
companyParams, companyParams,
employeeCount, employeeCount,
expensesOverview,
}: { }: {
account: Database['medreport']['Tables']['accounts']['Row']; account: Account;
companyParams: Database['medreport']['Tables']['company_params']['Row']; companyParams: CompanyParams;
employeeCount: number; employeeCount: number;
expensesOverview: TeamAccountBenefitExpensesOverview;
}) => { }) => {
const [currentCompanyParams, setCurrentCompanyParams] =
useState<Database['medreport']['Tables']['company_params']['Row']>(
companyParams,
);
const [isLoading, setIsLoading] = useState<boolean>(false);
const form = useForm({
resolver: zodResolver(UpdateHealthBenefitSchema),
mode: 'onChange',
defaultValues: {
occurrence: currentCompanyParams.benefit_occurance || 'yearly',
amount: currentCompanyParams.benefit_amount || 0,
},
});
const isDirty = form.formState.isDirty;
const onSubmit = (data: { occurrence: string; amount: number }) => {
const promise = async () => {
setIsLoading(true);
try {
await updateHealthBenefit({ ...data, accountId: account.id });
setCurrentCompanyParams((prev) => ({
...prev,
benefit_amount: data.amount,
benefit_occurance: data.occurrence,
}));
} finally {
form.reset(data);
setIsLoading(false);
}
};
toast.promise(promise, {
success: 'Andmed uuendatud',
error: 'error',
});
};
return ( return (
<Form {...form}> <div className="flex flex-col gap-6 px-6 text-left">
<form <div className="mt-8 flex items-center justify-between">
className="flex flex-col gap-6 px-6 text-left" <div>
onSubmit={form.handleSubmit(onSubmit)} <h4>
> <Trans
<div className="mt-8 flex items-center justify-between"> i18nKey="billing:pageTitle"
<div> values={{ companyName: account.name }}
<h4>
<Trans
i18nKey="billing:pageTitle"
values={{ companyName: account.name }}
/>
</h4>
<p className="text-muted-foreground text-sm">
<Trans i18nKey="billing:description" />
</p>
</div>
<Button
type="submit"
className="relative"
disabled={!isDirty || isLoading}
>
{isLoading && (
<div className="absolute inset-0 flex items-center justify-center">
<Spinner />
</div>
)}
<div className={cn({ invisible: isLoading })}>
<Trans i18nKey="account:saveChanges" />
</div>
</Button>
</div>
<div className="flex flex-row gap-6">
<div className="border-border w-1/3 rounded-lg border">
<div className="p-6">
<div className="flex h-16 w-16 items-center justify-center rounded-full bg-orange-100">
<PiggyBankIcon className="h-[32px] w-[32px] stroke-orange-400 stroke-2" />
</div>
<p className="mt-4 text-sm font-medium">
<Trans i18nKey="billing:healthBenefitForm.description" />
</p>
<p className="pt-2 text-2xl font-semibold">
{currentCompanyParams.benefit_amount || 0}
</p>
</div>
<Separator />
<div className="p-6">
<HealthBenefitFields />
</div>
</div>
<div className="flex-1">
<YearlyExpensesOverview
employeeCount={employeeCount}
companyParams={currentCompanyParams}
/> />
<p className="text-muted-foreground mt-2 text-sm"> </h4>
<Trans i18nKey="billing:healthBenefitForm.info" /> </div>
</div>
<div className="flex flex-col-reverse sm:flex-row gap-6">
<div className="border-border w-full sm:w-1/3 rounded-lg border">
<div className="p-6">
<div className="flex h-16 w-16 items-center justify-center rounded-full bg-orange-100">
<PiggyBankIcon className="h-[32px] w-[32px] stroke-orange-400 stroke-2" />
</div>
<p className="mt-4 text-sm font-medium">
<Trans i18nKey="billing:healthBenefitForm.description" />
</p> </p>
</div> </div>
<Separator />
<div className="p-6">
<HealthBenefitFormClient
account={account}
companyParams={companyParams}
/>
</div>
</div> </div>
</form>
</Form> <div className="flex-1 space-y-6">
<YearlyExpensesOverview
employeeCount={employeeCount}
expensesOverview={expensesOverview}
/>
<p className="text-muted-foreground text-sm">
<Trans i18nKey="billing:healthBenefitForm.info" />
</p>
</div>
</div>
</div>
); );
}; };

View File

@@ -1,50 +1,19 @@
import { useMemo } from 'react'; 'use client';
import { Database } from '@/packages/supabase/src/database.types';
import { Trans } from '@kit/ui/makerkit/trans'; import { Trans } from '@kit/ui/makerkit/trans';
import { Separator } from '@kit/ui/separator'; import { Separator } from '@kit/ui/separator';
import { formatCurrency } from '@/packages/shared/src/utils';
import { useTranslation } from 'react-i18next';
import { TeamAccountBenefitExpensesOverview } from '../../_lib/server/load-team-account-benefit-expenses-overview';
const YearlyExpensesOverview = ({ const YearlyExpensesOverview = ({
employeeCount = 0, employeeCount = 0,
companyParams, expensesOverview,
}: { }: {
employeeCount?: number; employeeCount?: number;
companyParams: Database['medreport']['Tables']['company_params']['Row']; expensesOverview: TeamAccountBenefitExpensesOverview;
}) => { }) => {
const monthlyExpensePerEmployee = useMemo(() => { const { i18n: { language } } = useTranslation();
if (!companyParams.benefit_amount) {
return '0.00';
}
switch (companyParams.benefit_occurance) {
case 'yearly':
return (companyParams.benefit_amount / 12).toFixed(2);
case 'quarterly':
return (companyParams.benefit_amount / 3).toFixed(2);
case 'monthly':
return companyParams.benefit_amount.toFixed(2);
default:
return '0.00';
}
}, [companyParams]);
const maxYearlyExpensePerEmployee = useMemo(() => {
if (!companyParams.benefit_amount) {
return '0.00';
}
switch (companyParams.benefit_occurance) {
case 'yearly':
return companyParams.benefit_amount.toFixed(2);
case 'quarterly':
return (companyParams.benefit_amount * 3).toFixed(2);
case 'monthly':
return (companyParams.benefit_amount * 12).toFixed(2);
default:
return '0.00';
}
}, [companyParams]);
return ( return (
<div className="border-border rounded-lg border p-6"> <div className="border-border rounded-lg border p-6">
@@ -53,41 +22,56 @@ const YearlyExpensesOverview = ({
</h5> </h5>
<div className="mt-5 flex justify-between"> <div className="mt-5 flex justify-between">
<p className="text-sm font-medium"> <p className="text-sm font-medium">
<Trans i18nKey="billing:expensesOverview.monthly" /> <Trans i18nKey="billing:expensesOverview.employeeCount" />
</p> </p>
<span className="text-primary text-sm font-bold"> <span className="text-primary text-sm font-bold">
{monthlyExpensePerEmployee} {employeeCount}
</span> </span>
</div> </div>
<div className="mt-3 flex justify-between"> <div className="mt-3 flex justify-between">
<p className="text-sm font-medium">
<Trans i18nKey="billing:expensesOverview.yearly" />
</p>
<span className="text-sm font-medium">
{maxYearlyExpensePerEmployee}
</span>
</div>
<div className="mt-5 mb-3 flex justify-between">
<p className="text-sm font-medium"> <p className="text-sm font-medium">
<Trans <Trans
i18nKey="billing:expensesOverview.total" i18nKey="billing:expensesOverview.managementFeeTotal"
values={{ employeeCount: employeeCount || 0 }} values={{
managementFee: formatCurrency({
value: expensesOverview.managementFee,
locale: language,
currencyCode: 'EUR',
}),
}}
/> />
</p> </p>
<span className="text-sm font-medium"> <span className="text-sm font-medium">
{(Number(maxYearlyExpensePerEmployee) * employeeCount).toFixed(2)} {formatCurrency({
value: expensesOverview.managementFeeTotal,
locale: language,
currencyCode: 'EUR',
})}
</span>
</div>
<div className="mt-3 mb-4 flex justify-between">
<p className="text-sm font-medium">
<Trans i18nKey="billing:expensesOverview.currentMonthUsageTotal" />
</p>
<span className="text-sm font-medium">
{formatCurrency({
value: expensesOverview.currentMonthUsageTotal,
locale: language,
currencyCode: 'EUR',
})}
</span> </span>
</div> </div>
<Separator /> <Separator />
<div className="mt-4 flex justify-between"> <div className="mt-4 flex justify-between">
<p className="font-semibold"> <p className="font-semibold">
<Trans i18nKey="billing:expensesOverview.sum" /> <Trans i18nKey="billing:expensesOverview.total" />
</p> </p>
<span className="font-semibold"> <span className="font-semibold">
{companyParams.benefit_amount {formatCurrency({
? companyParams.benefit_amount * employeeCount value: expensesOverview.total,
: 0}{' '} locale: language,
currencyCode: 'EUR',
})}
</span> </span>
</div> </div>
</div> </div>

View File

@@ -7,6 +7,7 @@ import { createI18nServerInstance } from '~/lib/i18n/i18n.server';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import HealthBenefitForm from './_components/health-benefit-form'; import HealthBenefitForm from './_components/health-benefit-form';
import { loadTeamAccountBenefitExpensesOverview } from '../_lib/server/load-team-account-benefit-expenses-overview';
interface TeamAccountBillingPageProps { interface TeamAccountBillingPageProps {
params: Promise<{ account: string }>; params: Promise<{ account: string }>;
@@ -27,8 +28,14 @@ async function TeamAccountBillingPage({ params }: TeamAccountBillingPageProps) {
const api = createTeamAccountsApi(client); const api = createTeamAccountsApi(client);
const account = await api.getTeamAccount(accountSlug); const account = await api.getTeamAccount(accountSlug);
const companyParams = await api.getTeamAccountParams(account.id);
const { members } = await api.getMembers(accountSlug); const { members } = await api.getMembers(accountSlug);
const [expensesOverview, companyParams] = await Promise.all([
loadTeamAccountBenefitExpensesOverview({
companyId: account.id,
employeeCount: members.length,
}),
api.getTeamAccountParams(account.id),
]);
return ( return (
<PageBody> <PageBody>
@@ -36,6 +43,7 @@ async function TeamAccountBillingPage({ params }: TeamAccountBillingPageProps) {
account={account} account={account}
companyParams={companyParams} companyParams={companyParams}
employeeCount={members.length} employeeCount={members.length}
expensesOverview={expensesOverview}
/> />
</PageBody> </PageBody>
); );

View File

@@ -5,6 +5,7 @@ import { SupabaseClient } from '@supabase/supabase-js';
import { Database } from '@/packages/supabase/src/database.types'; import { Database } from '@/packages/supabase/src/database.types';
import { loadTeamWorkspace } from '~/home/[account]/_lib/server/team-account-workspace.loader'; import { loadTeamWorkspace } from '~/home/[account]/_lib/server/team-account-workspace.loader';
import { getSupabaseServerAdminClient } from '@/packages/supabase/src/clients/server-admin-client';
/** /**
* Load data for the members page * Load data for the members page
@@ -15,11 +16,13 @@ export async function loadMembersPageData(
client: SupabaseClient<Database>, client: SupabaseClient<Database>,
slug: string, slug: string,
) { ) {
const workspace = await loadTeamWorkspace(slug);
return Promise.all([ return Promise.all([
loadAccountMembers(client, slug), loadAccountMembers(client, slug),
loadInvitations(client, slug), loadInvitations(client, slug),
canAddMember, canAddMember,
loadTeamWorkspace(slug), workspace,
loadAccountMembersBenefitsUsage(getSupabaseServerAdminClient(), workspace.account.id),
]); ]);
} }
@@ -60,6 +63,27 @@ async function loadAccountMembers(
return data ?? []; return data ?? [];
} }
export async function loadAccountMembersBenefitsUsage(
client: SupabaseClient<Database>,
accountId: string,
): Promise<{
personal_account_id: string;
benefit_amount: number;
}[]> {
const { data, error } = await client
.schema('medreport')
.rpc('get_benefits_usages_for_company_members', {
p_account_id: accountId,
});
if (error) {
console.error('Failed to load account members benefits usage', error);
return [];
}
return data ?? [];
}
/** /**
* Load account invitations * Load account invitations
* @param client * @param client

View File

@@ -42,7 +42,7 @@ async function TeamAccountMembersPage({ params }: TeamAccountMembersPageProps) {
const client = getSupabaseServerClient(); const client = getSupabaseServerClient();
const slug = (await params).account; const slug = (await params).account;
const [members, invitations, canAddMember, { user, account }] = const [members, invitations, canAddMember, { user, account }, membersBenefitsUsage] =
await loadMembersPageData(client, slug); await loadMembersPageData(client, slug);
const canManageRoles = account.permissions.includes('roles.manage'); const canManageRoles = account.permissions.includes('roles.manage');
@@ -54,8 +54,10 @@ async function TeamAccountMembersPage({ params }: TeamAccountMembersPageProps) {
return ( return (
<> <>
<TeamAccountLayoutPageHeader <TeamAccountLayoutPageHeader
title={<Trans i18nKey={'common:routes.members'} />} title={<Trans i18nKey={'common:routes.companyMembers'} />}
description={<AppBreadcrumbs values={{ [account.slug]: account.name }}/>} description={
<AppBreadcrumbs values={{ [account.slug]: account.name }} />
}
/> />
<PageBody> <PageBody>
@@ -96,6 +98,7 @@ async function TeamAccountMembersPage({ params }: TeamAccountMembersPageProps) {
members={members} members={members}
isPrimaryOwner={isPrimaryOwner} isPrimaryOwner={isPrimaryOwner}
canManageRoles={canManageRoles} canManageRoles={canManageRoles}
membersBenefitsUsage={membersBenefitsUsage}
/> />
</CardContent> </CardContent>
</Card> </Card>

View File

@@ -17,6 +17,8 @@ import {
} from '~/lib/services/audit/pageView.service'; } from '~/lib/services/audit/pageView.service';
import { Dashboard } from './_components/dashboard'; import { Dashboard } from './_components/dashboard';
import { loadAccountBenefitStatistics } from './_lib/server/load-team-account-benefit-statistics';
import { loadTeamAccountBenefitExpensesOverview } from './_lib/server/load-team-account-benefit-expenses-overview';
interface TeamAccountHomePageProps { interface TeamAccountHomePageProps {
params: Promise<{ account: string }>; params: Promise<{ account: string }>;
@@ -39,9 +41,11 @@ function TeamAccountHomePage({ params }: TeamAccountHomePageProps) {
const teamAccount = use(teamAccountsApi.getTeamAccount(account)); const teamAccount = use(teamAccountsApi.getTeamAccount(account));
const { memberParams, members } = use(teamAccountsApi.getMembers(account)); const { memberParams, members } = use(teamAccountsApi.getMembers(account));
const bmiThresholds = use(userAnalysesApi.fetchBmiThresholds()); const bmiThresholds = use(userAnalysesApi.fetchBmiThresholds());
const companyParams = use( const accountBenefitStatistics = use(loadAccountBenefitStatistics(teamAccount.id));
teamAccountsApi.getTeamAccountParams(teamAccount.id), const expensesOverview = use(loadTeamAccountBenefitExpensesOverview({
); companyId: teamAccount.id,
employeeCount: members.length,
}));
use( use(
createPageViewLog({ createPageViewLog({
@@ -57,7 +61,8 @@ function TeamAccountHomePage({ params }: TeamAccountHomePageProps) {
memberParams={memberParams} memberParams={memberParams}
bmiThresholds={bmiThresholds} bmiThresholds={bmiThresholds}
members={members} members={members}
companyParams={companyParams} accountBenefitStatistics={accountBenefitStatistics}
expensesOverview={expensesOverview}
/> />
</PageBody> </PageBody>
); );

View File

@@ -48,7 +48,9 @@ async function TeamAccountSettingsPage(props: TeamAccountSettingsPageProps) {
<> <>
<TeamAccountLayoutPageHeader <TeamAccountLayoutPageHeader
title={<Trans i18nKey={'teams:settings.pageTitle'} />} title={<Trans i18nKey={'teams:settings.pageTitle'} />}
description={<AppBreadcrumbs values={{ [account.slug]: account.name }} />} description={
<AppBreadcrumbs values={{ [account.slug]: account.name }} />
}
/> />
<PageBody> <PageBody>

View File

@@ -17,6 +17,7 @@ import { Trans } from '@kit/ui/trans';
import { createI18nServerInstance } from '~/lib/i18n/i18n.server'; import { createI18nServerInstance } from '~/lib/i18n/i18n.server';
import { withI18n } from '~/lib/i18n/with-i18n'; import { withI18n } from '~/lib/i18n/with-i18n';
import { toTitleCase } from '~/lib/utils';
interface JoinTeamAccountPageProps { interface JoinTeamAccountPageProps {
searchParams: Promise<{ searchParams: Promise<{
@@ -110,12 +111,12 @@ async function JoinTeamAccountPage(props: JoinTeamAccountPageProps) {
// once the user accepts the invitation, we redirect them to the account home page // once the user accepts the invitation, we redirect them to the account home page
const membershipConfirmation = pathsConfig.auth.membershipConfirmation; const membershipConfirmation = pathsConfig.auth.membershipConfirmation;
const email = auth.data.email ?? ''; const fullName = toTitleCase(auth.data.user_metadata.full_name ?? '');
return ( return (
<AuthLayoutShell Logo={AppLogo}> <AuthLayoutShell Logo={AppLogo}>
<AcceptInvitationContainer <AcceptInvitationContainer
email={email} fullName={fullName || auth.data.email || ''}
inviteToken={token} inviteToken={token}
invitation={invitation} invitation={invitation}
paths={{ paths={{

View File

@@ -1,14 +1,13 @@
'use server'; 'use server';
import { RequestStatus } from '@/lib/types/audit'; import { RequestStatus, SyncStatus } from '@/lib/types/audit';
import { ConnectedOnlineMethodName } from '@/lib/types/connected-online'; import { ConnectedOnlineMethodName } from '@/lib/types/connected-online';
import { ExternalApi } from '@/lib/types/external'; import { ExternalApi } from '@/lib/types/external';
import { MedipostAction } from '@/lib/types/medipost'; import { MedipostAction } from '@/lib/types/medipost';
import { getSupabaseServerAdminClient } from '@/packages/supabase/src/clients/server-admin-client'; import { getSupabaseServerAdminClient } from '@/packages/supabase/src/clients/server-admin-client';
import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client';
export default async function logRequestResult( export default async function logRequestResult(
/* personalCode: string, */ requestApi: keyof typeof ExternalApi, requestApi: keyof typeof ExternalApi,
requestApiMethod: `${ConnectedOnlineMethodName}` | `${MedipostAction}`, requestApiMethod: `${ConnectedOnlineMethodName}` | `${MedipostAction}`,
status: RequestStatus, status: RequestStatus,
comment?: string, comment?: string,
@@ -16,11 +15,10 @@ export default async function logRequestResult(
serviceId?: number, serviceId?: number,
serviceProviderId?: number, serviceProviderId?: number,
) { ) {
const { error } = await getSupabaseServerClient() const { error } = await getSupabaseServerAdminClient()
.schema('audit') .schema('audit')
.from('request_entries') .from('request_entries')
.insert({ .insert({
/* personal_code: personalCode, */
request_api: requestApi, request_api: requestApi,
request_api_method: requestApiMethod, request_api_method: requestApiMethod,
requested_start_date: startTime, requested_start_date: startTime,
@@ -69,3 +67,29 @@ export async function getMedipostDispatchTries(medusaOrderId: string) {
return data; return data;
} }
export async function logSyncResult({
operation,
comment,
status,
changed_by_role,
}: {
operation: string;
comment?: string;
status: SyncStatus;
changed_by_role: string;
}) {
const { error } = await getSupabaseServerAdminClient()
.schema('audit')
.from('sync_entries')
.insert({
operation,
comment,
status,
changed_by_role,
});
if (error) {
throw new Error('Failed to insert log entry, error: ' + error.message);
}
}

View File

@@ -0,0 +1,40 @@
import { getSupabaseServerClient } from '@kit/supabase/server-client';
export const createCartEntriesLog = async ({
operation,
accountId,
cartId,
variantId,
comment,
}: {
operation: string;
accountId: string;
cartId: string;
variantId?: string;
comment?: string;
}) => {
try {
const supabase = getSupabaseServerClient();
const {
data: { user },
error: userError,
} = await supabase.auth.getUser();
if (userError || !user) {
console.error('No authenticated user found; skipping audit insert');
return;
}
return supabase.schema('audit').from('cart_entries').insert({
operation,
account_id: accountId,
cart_id: cartId,
changed_by: user.id,
variant_id: variantId,
comment,
});
} catch (error) {
console.error('Failed to insert doctor page view log', error);
}
};

View File

@@ -6,6 +6,7 @@ export enum PageViewAction {
REGISTRATION_SUCCESS = 'REGISTRATION_SUCCESS', REGISTRATION_SUCCESS = 'REGISTRATION_SUCCESS',
VIEW_ORDER_ANALYSIS = 'VIEW_ORDER_ANALYSIS', VIEW_ORDER_ANALYSIS = 'VIEW_ORDER_ANALYSIS',
VIEW_TEAM_ACCOUNT_DASHBOARD = 'VIEW_TEAM_ACCOUNT_DASHBOARD', VIEW_TEAM_ACCOUNT_DASHBOARD = 'VIEW_TEAM_ACCOUNT_DASHBOARD',
VIEW_TTO_SERVICE_BOOKING = 'VIEW_TTO_SERVICE_BOOKING',
} }
export const createPageViewLog = async ({ export const createPageViewLog = async ({
@@ -37,6 +38,7 @@ export const createPageViewLog = async ({
account_id: accountId, account_id: accountId,
action, action,
changed_by: user.id, changed_by: user.id,
extra_data: extraData,
}) })
.throwOnError(); .throwOnError();
} catch (error) { } catch (error) {

View File

@@ -7,19 +7,30 @@ import {
BookTimeResponse, BookTimeResponse,
ConfirmedLoadResponse, ConfirmedLoadResponse,
ConnectedOnlineMethodName, ConnectedOnlineMethodName,
FailureReason,
} from '@/lib/types/connected-online'; } from '@/lib/types/connected-online';
import { ExternalApi } from '@/lib/types/external'; import { ExternalApi } from '@/lib/types/external';
import { Tables } from '@/packages/supabase/src/database.types'; import { Tables } from '@/packages/supabase/src/database.types';
import { createClient } from '@/utils/supabase/server';
import axios from 'axios'; import axios from 'axios';
import { uniq, uniqBy } from 'lodash';
import { renderBookTimeFailedEmail } from '@kit/email-templates';
import { getLogger } from '@kit/shared/logger';
import { getSupabaseServerClient } from '@kit/supabase/server-client';
import { TimeSlotResponse } from '~/home/(user)/_components/booking/booking.context';
import { sendEmailFromTemplate } from './mailer.service';
export async function getAvailableAppointmentsForService( export async function getAvailableAppointmentsForService(
serviceId: number, serviceId: number,
key: string,
locationId: number | null,
startTime?: Date, startTime?: Date,
maxDays?: number,
) { ) {
try { try {
const showTimesFrom = startTime ? { StartTime: startTime } : {}; const start = startTime ? { StartTime: startTime } : {};
const response = await axios.post( const response = await axios.post(
`${process.env.CONNECTED_ONLINE_URL!}/${ConnectedOnlineMethodName.GetAvailabilities}`, `${process.env.CONNECTED_ONLINE_URL!}/${ConnectedOnlineMethodName.GetAvailabilities}`,
{ {
@@ -28,9 +39,11 @@ export async function getAvailableAppointmentsForService(
}, },
param: JSON.stringify({ param: JSON.stringify({
ServiceID: serviceId, ServiceID: serviceId,
Key: '7T624nlu', Key: key,
Lang: 'et', Lang: 'et',
...showTimesFrom, MaxDays: maxDays ?? 120,
LocationId: locationId ?? -1,
...start,
}), }),
}, },
); );
@@ -80,157 +93,210 @@ export async function getAvailableAppointmentsForService(
} }
export async function bookAppointment( export async function bookAppointment(
serviceSyncId: number, serviceId: number,
clinicId: number, clinicId: number,
appointmentUserId: number, appointmentUserId: number,
syncUserID: number, syncUserID: number,
startTime: string, startTime: string,
locationId = 0,
comments = '', comments = '',
isEarlierTimeRequested = false,
earlierTimeRequestComment = '',
) { ) {
const supabase = await createClient(); const logger = await getLogger();
const supabase = getSupabaseServerClient();
let reason = FailureReason.BOOKING_FAILED;
try { try {
const { const {
data: { user }, data: { user },
} = await supabase.auth.getUser(); } = await supabase.auth.getUser();
logger.info(
`Booking time slot ${JSON.stringify({ serviceId, clinicId, startTime, userId: user?.id })}`,
);
if (!user?.id) { if (!user?.id) {
throw new Error('User not authenticated'); throw new Error('User not authenticated');
} }
const formattedStartTime = startTime.replace('T', ' ');
const [ const [
{ data: dbClinic, error: clinicError }, { data: dbClinic, error: clinicError },
{ data: dbService, error: serviceError }, { data: dbService, error: serviceError },
{ data: account, error: accountError },
{ data: dbReservation, error: dbReservationError },
] = await Promise.all([ ] = await Promise.all([
supabase supabase
.schema('medreport') .schema('medreport')
.from('connected_online_providers') .from('connected_online_providers')
.select('*') .select('*')
.eq('id', clinicId) .eq('id', clinicId)
.limit(1), .single(),
supabase supabase
.schema('medreport') .schema('medreport')
.from('connected_online_services') .from('connected_online_services')
.select('*') .select('*')
.eq('sync_id', serviceSyncId) .eq('id', serviceId)
.eq('clinic_id', clinicId) .eq('clinic_id', clinicId)
.limit(1), .single(),
supabase
.schema('medreport')
.from('accounts')
.select('name, last_name, personal_code, phone, email')
.eq('is_personal_account', true)
.eq('primary_owner_user_id', user.id)
.single(),
supabase
.schema('medreport')
.from('connected_online_reservation')
.select('id')
.eq('clinic_id', clinicId)
.eq('service_id', serviceId)
.eq('start_time', formattedStartTime)
.eq('user_id', user.id)
.eq('status', 'PENDING')
.single(),
]); ]);
if (!dbClinic?.length || !dbService?.length) { if (!dbClinic || !dbService) {
return logRequestResult( const errorMessage = dbClinic
ExternalApi.ConnectedOnline, ? `Could not find clinic with id ${clinicId}, error: ${JSON.stringify(clinicError)}`
ConnectedOnlineMethodName.BookTime, : `Could not find service with sync id ${serviceId} and clinic id ${clinicId}, error: ${JSON.stringify(serviceError)}`;
RequestStatus.Fail, logger.error(errorMessage);
dbClinic?.length
? `Could not find clinic with id ${clinicId}, error: ${JSON.stringify(clinicError)}` throw new Error(errorMessage);
: `Could not find service with sync id ${serviceSyncId} and clinic id ${clinicId}, error: ${JSON.stringify(serviceError)}`, }
startTime,
serviceSyncId, if (clinicError || serviceError || accountError) {
clinicId, const stringifiedErrors = JSON.stringify({
); clinicError,
serviceError,
accountError,
});
const errorMessage = `Failed to book time, error: ${stringifiedErrors}`;
logger.error(errorMessage);
throw new Error(errorMessage);
}
if (!dbReservation) {
const errorMessage = `No reservation found in db with data ${JSON.stringify({ clinicId, serviceId, startTime, userId: user.id })}, got error ${JSON.stringify(dbReservationError)}`;
logger.error(errorMessage);
throw new Error(errorMessage);
} }
const clinic: Tables< const clinic: Tables<
{ schema: 'medreport' }, { schema: 'medreport' },
'connected_online_providers' 'connected_online_providers'
> = dbClinic![0]; > = dbClinic;
const service: Tables< const service: Tables<
{ schema: 'medreport' }, { schema: 'medreport' },
'connected_online_services' 'connected_online_services'
> = dbService![0]; > = dbService;
// TODO the dummy data needs to be replaced with real values once they're present on the user/account const connectedOnlineBookingResponse = await axios.post(
const response = await axios.post(
`${process.env.CONNECTED_ONLINE_URL!}/${ConnectedOnlineMethodName.BookTime}`, `${process.env.CONNECTED_ONLINE_URL!}/${ConnectedOnlineMethodName.BookTime}`,
{ {
headers: { headers: {
'Content-Type': 'application/json; charset=utf-8', 'Content-Type': 'application/json; charset=utf-8',
}, },
param: JSON.stringify({ param: JSON.stringify({
EarlierTime: isEarlierTimeRequested, // once we have the e-shop functionality we can let the user select if she would like to be offered earlier time slots if they become available
EarlierTimeComment: earlierTimeRequestComment,
ClinicID: clinic.id, ClinicID: clinic.id,
ServiceID: service.id, ServiceID: service.sync_id,
ClinicServiceID: service.sync_id, ClinicServiceID: service.id,
UserID: appointmentUserId, UserID: appointmentUserId,
SyncUserID: syncUserID, SyncUserID: syncUserID,
StartTime: startTime, StartTime: startTime,
FirstName: 'Test', FirstName: account.name,
LastName: 'User', LastName: account.last_name,
PersonalCode: '4', PersonalCode: account.personal_code,
Email: user.email, Email: account.email ?? user.email,
Phone: 'phone', Phone: account.phone,
Comments: comments, Comments: comments,
Location: locationId,
FreeCode: '',
AddToBasket: false, AddToBasket: false,
Key: '7T624nlu', Key: dbClinic.key,
Lang: 'et', // update when integrated into app, if needed Lang: 'et',
}), }),
}, },
); );
const responseData: BookTimeResponse = JSON.parse(response.data.d); const connectedOnlineBookingResponseData: BookTimeResponse = JSON.parse(
connectedOnlineBookingResponse.data.d,
);
if (responseData?.ErrorCode !== 0 || !responseData.Value) { const errorCode = connectedOnlineBookingResponseData?.ErrorCode;
return logRequestResult( if (errorCode !== 0 || !connectedOnlineBookingResponseData.Value) {
ExternalApi.ConnectedOnline, const errorMessage = `Received invalid result from external api, error: ${JSON.stringify(connectedOnlineBookingResponseData)}`;
ConnectedOnlineMethodName.BookTime, logger.error(errorMessage);
RequestStatus.Fail, if (process.env.SUPPORT_EMAIL) {
JSON.stringify(responseData), await sendEmailFromTemplate(
startTime, renderBookTimeFailedEmail,
service.id, { reservationId: dbReservation.id, error: errorMessage },
clinicId, process.env.SUPPORT_EMAIL,
);
}
await supabase
.schema('medreport')
.from('connected_online_reservation')
.update({
status: 'REJECTED',
})
.eq('id', dbReservation.id)
.throwOnError();
if (errorCode === 1) {
reason = FailureReason.TIME_SLOT_UNAVAILABLE;
}
throw new Error(errorMessage);
}
const responseParts = connectedOnlineBookingResponseData.Value.split(',');
const { data: updatedReservation, error } = await supabase
.schema('medreport')
.from('connected_online_reservation')
.update({
booking_code: responseParts[1],
requires_payment: !!responseParts[0],
status: 'CONFIRMED',
})
.eq('id', dbReservation.id)
.select('id')
.single();
if (error) {
throw new Error(
JSON.stringify({ connectedOnlineBookingResponseData, error }),
); );
} }
const responseParts = responseData.Value.split(','); logger.info(
'Booked time, updated reservation with id ' + updatedReservation?.id,
const { error } = await supabase );
.schema('medreport')
.from('connected_online_reservation')
.insert({
booking_code: responseParts[1],
clinic_id: clinic.id,
comments,
lang: 'et', // change later, if needed
service_id: service.id,
service_user_id: appointmentUserId,
start_time: startTime,
sync_user_id: syncUserID,
requires_payment: !!responseParts[0],
user_id: user.id,
});
await logRequestResult( await logRequestResult(
ExternalApi.ConnectedOnline, ExternalApi.ConnectedOnline,
ConnectedOnlineMethodName.BookTime, ConnectedOnlineMethodName.BookTime,
RequestStatus.Success, RequestStatus.Success,
JSON.stringify(responseData), JSON.stringify(connectedOnlineBookingResponseData),
startTime, startTime.toString(),
service.id, service.id,
clinicId, clinicId,
); );
return { success: true };
if (error) {
throw new Error(error.message);
}
return responseData.Value;
} catch (error) { } catch (error) {
return logRequestResult( logger.error(`Failed to book time, error: ${JSON.stringify(error)}`);
await logRequestResult(
ExternalApi.ConnectedOnline, ExternalApi.ConnectedOnline,
ConnectedOnlineMethodName.BookTime, ConnectedOnlineMethodName.BookTime,
RequestStatus.Fail, RequestStatus.Fail,
JSON.stringify(error), JSON.stringify(error),
startTime, startTime.toString(),
serviceSyncId, serviceId,
clinicId, clinicId,
); );
return { success: false, reason };
} }
} }
@@ -270,8 +336,83 @@ export async function getConfirmedService(reservationCode: string) {
ExternalApi.ConnectedOnline, ExternalApi.ConnectedOnline,
ConnectedOnlineMethodName.ConfirmedLoad, ConnectedOnlineMethodName.ConfirmedLoad,
RequestStatus.Fail, RequestStatus.Fail,
JSON.stringify(error), error?.toString(),
); );
return null; return null;
} }
} }
export async function getAvailableTimeSlotsForDisplay(
serviceIds: number[],
locationId: number | null,
date?: Date,
): Promise<TimeSlotResponse> {
const supabase = getSupabaseServerClient();
const { data: syncedServices } = await supabase
.schema('medreport')
.from('connected_online_services')
.select(
'*, providerClinic:clinic_id(*,locations:connected_online_locations(*))',
)
.in('id', serviceIds)
.throwOnError();
const timeSlotPromises = [];
for (const syncedService of syncedServices) {
const timeSlotsPromise = getAvailableAppointmentsForService(
syncedService.id,
syncedService.providerClinic.key,
locationId,
date,
);
timeSlotPromises.push(timeSlotsPromise);
}
const timeSlots = await Promise.all(timeSlotPromises);
const mappedTimeSlots = [];
for (const timeSlotGroup of timeSlots) {
const { data: serviceProviders } = await supabase
.schema('medreport')
.from('connected_online_service_providers')
.select(
'name, id, jobTitleEn: job_title_en, jobTitleEt: job_title_et, jobTitleRu: job_title_ru, clinicId: clinic_id',
)
.in(
'clinic_id',
uniq(timeSlotGroup?.T_Booking.map(({ ClinicID }) => ClinicID)),
)
.throwOnError();
const timeSlots =
timeSlotGroup?.T_Booking?.map((item) => {
return {
...item,
serviceProvider: serviceProviders.find(
({ id }) => id === item.UserID,
),
syncedService: syncedServices.find(
(syncedService) => syncedService.sync_id === item.ServiceID,
),
location: syncedServices
.find(
({ providerClinic }) =>
providerClinic.id === Number(item.ClinicID),
)
?.providerClinic?.locations?.find(
(location) => location.sync_id === item.LocationID,
),
};
}) ?? [];
mappedTimeSlots.push(...timeSlots);
}
return {
timeSlots: mappedTimeSlots,
locations: uniqBy(
syncedServices.flatMap(({ providerClinic }) => providerClinic.locations),
'id',
),
};
}

View File

@@ -16,8 +16,8 @@ import axios from 'axios';
import { toArray } from '@kit/shared/utils'; import { toArray } from '@kit/shared/utils';
import { Tables } from '@kit/supabase/database'; import { Tables } from '@kit/supabase/database';
import type { AnalysisOrder } from '~/lib/types/analysis-order';
import type { AnalysisResponseElement } from '~/lib/types/analysis-response-element'; import type { AnalysisResponseElement } from '~/lib/types/analysis-response-element';
import type { AnalysisOrder } from '~/lib/types/order';
import { getAccountAdmin } from '../account.service'; import { getAccountAdmin } from '../account.service';
import { getAnalyses } from '../analyses.service'; import { getAnalyses } from '../analyses.service';
@@ -28,7 +28,7 @@ import {
upsertAnalysisResponseElement, upsertAnalysisResponseElement,
} from '../analysis-order.service'; } from '../analysis-order.service';
import { logMedipostDispatch } from '../audit.service'; import { logMedipostDispatch } from '../audit.service';
import { getAnalysisOrder, updateAnalysisOrderStatus } from '../order.service'; import { getAnalysisOrder } from '../order.service';
import { parseXML } from '../util/xml.service'; import { parseXML } from '../util/xml.service';
import { MedipostValidationError } from './MedipostValidationError'; import { MedipostValidationError } from './MedipostValidationError';
import { import {
@@ -430,17 +430,19 @@ export async function readPrivateMessageResponse({
medipostExternalOrderId, medipostExternalOrderId,
}); });
if (status.isPartial) { if (status.isPartial) {
await updateAnalysisOrderStatus({ await createUserAnalysesApi(getSupabaseServerAdminClient())
medusaOrderId, .updateAnalysisOrderStatus({
orderStatus: 'PARTIAL_ANALYSIS_RESPONSE', medusaOrderId,
}); orderStatus: 'PARTIAL_ANALYSIS_RESPONSE',
});
hasAnalysisResponse = true; hasAnalysisResponse = true;
hasPartialAnalysisResponse = true; hasPartialAnalysisResponse = true;
} else if (status.isCompleted) { } else if (status.isCompleted) {
await updateAnalysisOrderStatus({ await createUserAnalysesApi(getSupabaseServerAdminClient())
medusaOrderId, .updateAnalysisOrderStatus({
orderStatus: 'FULL_ANALYSIS_RESPONSE', medusaOrderId,
}); orderStatus: 'FULL_ANALYSIS_RESPONSE',
});
if (IS_ENABLED_DELETE_PRIVATE_MESSAGE) { if (IS_ENABLED_DELETE_PRIVATE_MESSAGE) {
await deletePrivateMessage(privateMessageId); await deletePrivateMessage(privateMessageId);
} }
@@ -622,5 +624,9 @@ export async function sendOrderToMedipost({
hasAnalysisResults: false, hasAnalysisResults: false,
medusaOrderId, medusaOrderId,
}); });
await updateAnalysisOrderStatus({ medusaOrderId, orderStatus: 'PROCESSING' }); await createUserAnalysesApi(getSupabaseServerAdminClient())
.updateAnalysisOrderStatus({
medusaOrderId,
orderStatus: 'PROCESSING',
});
} }

View File

@@ -5,9 +5,16 @@ import { MontonioOrderHandlerService } from '@/packages/billing/montonio/src';
import { addToCart, deleteLineItem, retrieveCart } from '@lib/data/cart'; import { addToCart, deleteLineItem, retrieveCart } from '@lib/data/cart';
import { getCartId } from '@lib/data/cookies'; import { getCartId } from '@lib/data/cookies';
import { StoreCartLineItem, StoreProductVariant } from '@medusajs/types'; import { StoreCartLineItem, StoreProductVariant } from '@medusajs/types';
import { isSameMinute } from 'date-fns';
import { z } from 'zod'; import { z } from 'zod';
import { getSupabaseServerClient } from '@kit/supabase/server-client'; import {
cancelReservation,
getOrderedTtoServices,
} from '~/lib/services/reservation.service';
import { createCartEntriesLog } from './audit/cartEntries';
import { getAvailableAppointmentsForService } from './connected-online.service';
const env = () => const env = () =>
z z
@@ -35,64 +42,60 @@ export async function handleAddToCart({
selectedVariant: Pick<StoreProductVariant, 'id'>; selectedVariant: Pick<StoreProductVariant, 'id'>;
countryCode: string; countryCode: string;
}) { }) {
const supabase = getSupabaseServerClient(); const { account } = await loadCurrentUserAccount();
const { account, user } = await loadCurrentUserAccount();
if (!account) { if (!account) {
throw new Error('Account not found'); throw new Error('Account not found');
} }
const quantity = 1; const quantity = 1;
const cart = await addToCart({ const { newCart, addedItem } = await addToCart({
variantId: selectedVariant.id, variantId: selectedVariant.id,
quantity, quantity,
countryCode, countryCode,
}); });
const { error } = await supabase.schema('audit').from('cart_entries').insert({ await createCartEntriesLog({
variant_id: selectedVariant.id, variantId: selectedVariant.id,
operation: 'ADD_TO_CART', operation: 'ADD_TO_CART',
account_id: account.id, accountId: account.id,
cart_id: cart.id, cartId: newCart.id,
changed_by: user.id,
}); });
if (error) {
throw new Error('Error logging cart entry: ' + error.message);
}
return cart; return { cart: newCart, addedItem };
} }
export async function handleDeleteCartItem({ lineId }: { lineId: string }) { export async function handleDeleteCartItem({ lineId }: { lineId: string }) {
await deleteLineItem(lineId); await deleteLineItem(lineId);
await cancelReservation(lineId);
const supabase = getSupabaseServerClient();
const cartId = await getCartId(); const cartId = await getCartId();
const { account, user } = await loadCurrentUserAccount(); const { account } = await loadCurrentUserAccount();
if (!account) { if (!account) {
throw new Error('Account not found'); throw new Error('Account not found');
} }
const { error } = await supabase.schema('audit').from('cart_entries').insert({ await createCartEntriesLog({
variant_id: lineId, variantId: lineId,
operation: 'REMOVE_FROM_CART', operation: 'REMOVE_FROM_CART',
account_id: account.id, accountId: account.id,
cart_id: cartId!, cartId: cartId!,
changed_by: user.id,
}); });
if (error) {
throw new Error('Error logging cart entry: ' + error.message);
}
} }
export async function handleNavigateToPayment({ export async function handleNavigateToPayment({
language, language,
paymentSessionId, paymentSessionId,
amount,
currencyCode,
cartId,
}: { }: {
language: string; language: string;
paymentSessionId: string; paymentSessionId: string;
amount: number;
currencyCode: string;
cartId: string;
}) { }) {
const supabase = getSupabaseServerClient(); const { account } = await loadCurrentUserAccount();
const { account, user } = await loadCurrentUserAccount();
if (!account) { if (!account) {
throw new Error('Account not found'); throw new Error('Account not found');
} }
@@ -101,29 +104,52 @@ export async function handleNavigateToPayment({
if (!cart) { if (!cart) {
throw new Error('No cart found'); throw new Error('No cart found');
} }
const orderedTtoServices = await getOrderedTtoServices({ cart });
if (orderedTtoServices?.length) {
const unavailableLineItemIds: string[] = [];
for (const ttoService of orderedTtoServices) {
const availabilities = await getAvailableAppointmentsForService(
ttoService.service_id,
ttoService.provider.key,
ttoService.location_sync_id,
new Date(ttoService.start_time),
1,
);
const isAvailable = availabilities?.T_Booking?.length
? availabilities.T_Booking.find((timeSlot) =>
isSameMinute(ttoService.start_time, timeSlot.StartTime),
)
: false;
if (!isAvailable) {
unavailableLineItemIds.push(ttoService.medusa_cart_line_item_id!);
}
}
if (unavailableLineItemIds.length) {
return { unavailableLineItemIds };
}
}
const paymentLink = const paymentLink =
await new MontonioOrderHandlerService().getMontonioPaymentLink({ await new MontonioOrderHandlerService().getMontonioPaymentLink({
notificationUrl: `${env().medusaBackendPublicUrl}/hooks/payment/montonio_montonio`, notificationUrl: `${env().medusaBackendPublicUrl}/hooks/payment/montonio_montonio`,
returnUrl: `${env().siteUrl}/home/cart/montonio-callback`, returnUrl: `${env().siteUrl}/home/cart/montonio-callback`,
amount: cart.total, amount,
currency: cart.currency_code.toUpperCase(), currency: currencyCode.toUpperCase(),
description: `Order from Medreport`, description: `Order from Medreport`,
locale: language, locale: language,
merchantReference: `${account.id}:${paymentSessionId}:${cart.id}`, merchantReference: `${account.id}:${paymentSessionId}:${cartId}`,
}); });
const { error } = await supabase.schema('audit').from('cart_entries').insert({ await createCartEntriesLog({
operation: 'NAVIGATE_TO_PAYMENT', operation: 'NAVIGATE_TO_PAYMENT',
account_id: account.id, accountId: account.id,
cart_id: cart.id, cartId: cart.id,
changed_by: user.id,
}); });
if (error) {
throw new Error('Error logging cart entry: ' + error.message);
}
return paymentLink; return { url: paymentLink };
} }
export async function handleLineItemTimeout({ export async function handleLineItemTimeout({
@@ -131,21 +157,16 @@ export async function handleLineItemTimeout({
}: { }: {
lineItem: StoreCartLineItem; lineItem: StoreCartLineItem;
}) { }) {
const supabase = getSupabaseServerClient(); const { account } = await loadCurrentUserAccount();
const { account, user } = await loadCurrentUserAccount();
if (!account) { if (!account) {
throw new Error('Account not found'); throw new Error('Account not found');
} }
await deleteLineItem(lineItem.id); await deleteLineItem(lineItem.id);
const { error } = await supabase.schema('audit').from('cart_entries').insert({ await createCartEntriesLog({
operation: 'LINE_ITEM_TIMEOUT', operation: 'LINE_ITEM_TIMEOUT',
account_id: account.id, accountId: account.id,
cart_id: lineItem.cart_id, cartId: lineItem.cart_id,
changed_by: user.id,
}); });
if (error) {
throw new Error('Error logging cart entry: ' + error.message);
}
} }

View File

@@ -4,7 +4,7 @@ import type { Tables } from '@kit/supabase/database';
import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client'; import { getSupabaseServerAdminClient } from '@kit/supabase/server-admin-client';
import { getSupabaseServerClient } from '@kit/supabase/server-client'; import { getSupabaseServerClient } from '@kit/supabase/server-client';
import type { AnalysisOrder } from '../types/analysis-order'; import type { AnalysisOrder, TTOOrder } from '../types/order';
export async function createAnalysisOrder({ export async function createAnalysisOrder({
medusaOrder, medusaOrder,
@@ -51,48 +51,6 @@ export async function createAnalysisOrder({
return orderResult.data.id; return orderResult.data.id;
} }
export async function updateAnalysisOrder({
orderId,
orderStatus,
}: {
orderId: number;
orderStatus: Tables<{ schema: 'medreport' }, 'analysis_orders'>['status'];
}) {
console.info(`Updating order id=${orderId} status=${orderStatus}`);
await getSupabaseServerAdminClient()
.schema('medreport')
.from('analysis_orders')
.update({
status: orderStatus,
})
.eq('id', orderId)
.throwOnError();
}
export async function updateAnalysisOrderStatus({
orderId,
medusaOrderId,
orderStatus,
}: {
orderId?: number;
medusaOrderId?: string;
orderStatus: Tables<{ schema: 'medreport' }, 'analysis_orders'>['status'];
}) {
const orderIdParam = orderId;
const medusaOrderIdParam = medusaOrderId;
if (!orderIdParam && !medusaOrderIdParam) {
throw new Error('Either orderId or medusaOrderId must be provided');
}
await getSupabaseServerAdminClient()
.schema('medreport')
.rpc('update_analysis_order_status', {
order_id: orderIdParam ?? -1,
status_param: orderStatus,
medusa_order_id_param: medusaOrderIdParam ?? '',
})
.throwOnError();
}
export async function getAnalysisOrder({ export async function getAnalysisOrder({
medusaOrderId, medusaOrderId,
analysisOrderId, analysisOrderId,
@@ -171,3 +129,39 @@ export async function getAnalysisOrdersAdmin({
.throwOnError(); .throwOnError();
return orders.data; return orders.data;
} }
export async function getTtoOrders({
orderStatus,
lineItemIds,
}: {
orderStatus?: TTOOrder['status'];
lineItemIds?: string[];
} = {}) {
const client = getSupabaseServerClient();
const {
data: { user },
} = await client.auth.getUser();
if (!user) {
throw new Error('Unauthorized');
}
const query = client
.schema('medreport')
.from('connected_online_reservation')
.select('*')
.eq('user_id', user.id);
if (orderStatus) {
query.eq('status', orderStatus);
}
if (lineItemIds?.length) {
query.in('medusa_cart_line_item_id', lineItemIds);
}
const orders = await query
.order('created_at', { ascending: false })
.throwOnError();
return orders.data;
}

View File

@@ -0,0 +1,343 @@
'use server';
import { revalidatePath } from 'next/cache';
import { listRegions } from '@lib/data/regions';
import { StoreCart, StoreOrder } from '@medusajs/types';
import { getLogger } from '@kit/shared/logger';
import { Tables } from '@kit/supabase/database';
import { getSupabaseServerClient } from '@kit/supabase/server-client';
import { EnrichedCartItem } from '../../app/home/(user)/_components/cart/types';
import { loadCurrentUserAccount } from '../../app/home/(user)/_lib/server/load-user-account';
import { createCartEntriesLog } from './audit/cartEntries';
import { handleDeleteCartItem } from './medusaCart.service';
type Locations = Tables<{ schema: 'medreport' }, 'connected_online_locations'>;
type Services = Tables<{ schema: 'medreport' }, 'connected_online_services'>;
type ServiceProviders = Tables<
{ schema: 'medreport' },
'connected_online_service_providers'
>;
export async function getCartReservations(
medusaCart: StoreCart,
): Promise<EnrichedCartItem[]> {
const supabase = getSupabaseServerClient();
const cartLineItemIds = medusaCart.items?.map(({ id }) => id);
if (!cartLineItemIds?.length) {
return [];
}
const { data: reservations } = await supabase
.schema('medreport')
.from('connected_online_reservation')
.select(
'id, startTime:start_time, service:service_id, location:location_sync_id, serviceProvider:service_user_id, medusaCartLineItemId:medusa_cart_line_item_id',
)
.in('medusa_cart_line_item_id', cartLineItemIds)
.throwOnError();
const locationSyncIds: number[] =
reservations
?.filter((reservation) => !!reservation.location)
.map((reservation) => reservation.location!) ?? [];
const serviceIds =
reservations?.map((reservation) => reservation.service) ?? [];
const serviceProviderIds =
reservations.map((reservation) => reservation.serviceProvider) ?? [];
let locations:
| {
syncId: Locations['sync_id'];
name: Locations['name'];
address: Locations['address'];
}[]
| null = null;
if (locationSyncIds.length) {
({ data: locations } = await supabase
.schema('medreport')
.from('connected_online_locations')
.select('syncId:sync_id, name, address')
.in('sync_id', locationSyncIds)
.throwOnError());
}
let services:
| {
id: Services['id'];
name: Services['name'];
}[]
| null = null;
if (serviceIds.length) {
({ data: services } = await supabase
.schema('medreport')
.from('connected_online_services')
.select('name, id')
.in('id', serviceIds)
.throwOnError());
}
let serviceProviders:
| {
id: ServiceProviders['id'];
name: ServiceProviders['name'];
jobTitleEt: ServiceProviders['job_title_et'];
jobTitleEn: ServiceProviders['job_title_en'];
jobTitleRu: ServiceProviders['job_title_ru'];
spokenLanguages: ServiceProviders['spoken_languages'];
}[]
| null = null;
if (serviceProviderIds.length) {
({ data: serviceProviders } = await supabase
.schema('medreport')
.from('connected_online_service_providers')
.select(
'id, name, jobTitleEt:job_title_et, jobTitleEn:job_title_en, jobTitleRu:job_title_ru, spokenLanguages:spoken_languages',
)
.in('id', serviceProviderIds)
.throwOnError());
}
const results = [];
for (const reservation of reservations) {
if (reservation.medusaCartLineItemId === null) {
continue;
}
const cartLineItem = medusaCart.items?.find(
(item) => item.id === reservation.medusaCartLineItemId,
);
if (!cartLineItem) {
continue;
}
const location = locations?.find(
(location) => location.syncId === reservation.location,
);
const service = services?.find(
(service) => service.id === reservation.service,
);
const serviceProvider = serviceProviders?.find(
(serviceProvider) => serviceProvider.id === reservation.serviceProvider,
);
const countryCodes = await listRegions().then((regions) =>
regions?.map((r) => r.countries?.map((c) => c.iso_2)).flat(),
);
const enrichedReservation = {
...reservation,
location,
service,
serviceProvider,
};
results.push({
...cartLineItem,
reservation: {
...enrichedReservation,
medusaCartLineItemId: reservation.medusaCartLineItemId!,
countryCode: countryCodes[0],
},
});
}
return results;
}
export async function createInitialReservation({
serviceId,
clinicId,
appointmentUserId,
syncUserID,
startTime,
medusaLineItemId,
locationId,
comments = '',
}: {
serviceId: number;
clinicId: number;
appointmentUserId: number;
syncUserID: number;
startTime: Date;
medusaLineItemId: string;
locationId?: number | null;
comments?: string;
}) {
const logger = await getLogger();
const supabase = getSupabaseServerClient();
const {
data: { user },
} = await supabase.auth.getUser();
const userId = user?.id;
if (!userId) {
throw new Error('User not authenticated');
}
logger.info(
'Creating reservation' +
JSON.stringify({ serviceId, clinicId, startTime, userId }),
);
try {
const { data: createdReservation } = await supabase
.schema('medreport')
.from('connected_online_reservation')
.insert({
clinic_id: clinicId,
comments,
lang: 'et',
service_id: serviceId,
service_user_id: appointmentUserId,
start_time: startTime.toString(),
sync_user_id: syncUserID,
user_id: userId,
status: 'PENDING',
medusa_cart_line_item_id: medusaLineItemId,
location_sync_id: locationId,
})
.select('id')
.single()
.throwOnError();
logger.info(
`Created reservation ${JSON.stringify({ createdReservation, userId })}`,
);
return createdReservation;
} catch (e) {
logger.error(
`Failed to create initial reservation ${JSON.stringify({ serviceId, clinicId, startTime })} ${e}`,
);
await handleDeleteCartItem({ lineId: medusaLineItemId });
throw e;
}
}
export async function cancelReservation(medusaLineItemId: string) {
const supabase = getSupabaseServerClient();
return supabase
.schema('medreport')
.from('connected_online_reservation')
.update({
status: 'CANCELLED',
})
.eq('medusa_cart_line_item_id', medusaLineItemId)
.throwOnError();
}
export async function getOrderedTtoServices({
cart,
medusaOrder,
}: {
cart?: StoreCart;
medusaOrder?: StoreOrder;
}) {
const supabase = getSupabaseServerClient();
if (!medusaOrder && !cart) {
throw new Error('No cart or medusa order provided');
}
const ttoReservationIds: number[] =
(medusaOrder?.items ?? cart?.items)
?.filter(({ metadata }) => !!metadata?.connectedOnlineReservationId)
.map(({ metadata }) => Number(metadata!.connectedOnlineReservationId)) ??
[];
const { data: orderedTtoServices } = await supabase
.schema('medreport')
.from('connected_online_reservation')
.select('*, provider:connected_online_providers(key)')
.in('id', ttoReservationIds)
.throwOnError();
return orderedTtoServices;
}
export async function updateReservationTime({
reservationId,
newStartTime,
newServiceId,
newAppointmentUserId,
newSyncUserId,
newLocationId, // TODO stop allowing null when Connected starts returning the correct ids instead of -1
cartId,
}: {
reservationId: number;
newStartTime: Date;
newServiceId: number;
newAppointmentUserId: number;
newSyncUserId: number;
newLocationId: number | null;
cartId: string;
}) {
const logger = await getLogger();
const supabase = getSupabaseServerClient();
const {
data: { user },
} = await supabase.auth.getUser();
const userId = user?.id;
const { account } = await loadCurrentUserAccount();
if (!userId || !account) {
throw new Error('User not authenticated');
}
const reservationData = JSON.stringify({
reservationId,
newStartTime,
newServiceId,
newAppointmentUserId,
newSyncUserId,
newLocationId,
userId,
cartId,
});
logger.info('Updating reservation' + reservationData);
try {
await supabase
.schema('medreport')
.from('connected_online_reservation')
.update({
service_id: newServiceId,
service_user_id: newAppointmentUserId,
sync_user_id: newSyncUserId,
start_time: newStartTime.toString(),
location_sync_id: newLocationId,
})
.eq('id', reservationId)
.eq('user_id', user.id)
.throwOnError();
logger.info(`Successfully updated reservation ${reservationData}`);
await createCartEntriesLog({
operation: 'CHANGE_RESERVATION',
accountId: account.id,
cartId: cartId,
comment: `${reservationData}`,
});
revalidatePath('/home/cart', 'layout');
} catch (e) {
logger.error(`Failed to update reservation ${reservationData}`);
await createCartEntriesLog({
operation: 'CHANGE_RESERVATION',
accountId: account.id,
cartId: cartId,
comment: `${e}`,
});
throw e;
}
}

View File

@@ -1,3 +0,0 @@
import type { Tables } from '@kit/supabase/database';
export type AnalysisOrder = Tables<{ schema: 'medreport' }, 'analysis_orders'>;

View File

@@ -10,13 +10,15 @@ export type BookTimeResponse = z.infer<typeof BookTimeResponseSchema>;
export enum ConnectedOnlineMethodName { export enum ConnectedOnlineMethodName {
SearchLoad = 'Search_Load', SearchLoad = 'Search_Load',
DefaultLoad = 'Default_Load',
ConfirmedCancel = 'Confirmed_Cancel',
GetAvailabilities = 'GetAvailabilities', GetAvailabilities = 'GetAvailabilities',
BookTime = 'BookTime', BookTime = 'BookTime',
ConfirmedLoad = 'Confirmed_Load', ConfirmedLoad = 'Confirmed_Load',
} }
export const AvailableAppointmentTBookingSchema = z.object({ export const AvailableAppointmentTBookingSchema = z.object({
ClinicID: z.string(), ClinicID: z.number(),
LocationID: z.number(), LocationID: z.number(),
UserID: z.number(), UserID: z.number(),
SyncUserID: z.number(), SyncUserID: z.number(),
@@ -225,6 +227,18 @@ export const ConfirmedLoadResponseSchema = z.object({
}); });
export type ConfirmedLoadResponse = z.infer<typeof ConfirmedLoadResponseSchema>; export type ConfirmedLoadResponse = z.infer<typeof ConfirmedLoadResponseSchema>;
export type P_JobTitleTranslation = {
ID: number;
SyncID: number;
TextEN: string;
TextET: string;
TextFI: string;
TextRU: string;
TextLT: string;
ClinicID: number;
Deleted: number;
};
export interface ISearchLoadResponse { export interface ISearchLoadResponse {
Value: string; Value: string;
Data: { Data: {
@@ -232,9 +246,11 @@ export interface ISearchLoadResponse {
ID: number; ID: number;
Name: string; Name: string;
OnlineCanSelectWorker: boolean; OnlineCanSelectWorker: boolean;
Address: string;
Email: string | null; Email: string | null;
PersonalCodeRequired: boolean; PersonalCodeRequired: boolean;
Phone: string | null; Phone: string | null;
Key: string;
}[]; }[];
T_Service: { T_Service: {
ID: number; ID: number;
@@ -253,7 +269,14 @@ export interface ISearchLoadResponse {
RequiresPayment: boolean; RequiresPayment: boolean;
SyncID: string; SyncID: string;
}[]; }[];
T_Doctor: TDoctor[];
P_JobTitleTranslations: P_JobTitleTranslation[];
}; };
ErrorCode: number; ErrorCode: number;
ErrorMessage: string; ErrorMessage: string;
} }
export enum FailureReason {
BOOKING_FAILED = 'BOOKING_FAILED',
TIME_SLOT_UNAVAILABLE = 'TIME_SLOT_UNAVAILABLE',
}

12
lib/types/order.ts Normal file
View File

@@ -0,0 +1,12 @@
import type { Tables } from '@kit/supabase/database';
export type AnalysisOrder = Tables<{ schema: 'medreport' }, 'analysis_orders'>;
export type TTOOrder = Tables<
{ schema: 'medreport' },
'connected_online_reservation'
>;
export type Order = {
medusaOrderId?: string;
id?: number;
status?: string;
};

35
lib/types/reservation.ts Normal file
View File

@@ -0,0 +1,35 @@
import z from 'zod';
export const LocationSchema = z.object({
syncId: z.number(),
name: z.string(),
address: z.string().nullable(),
});
export type Location = z.infer<typeof LocationSchema>;
export const ServiceSchema = z.object({
name: z.string(),
id: z.number(),
});
export type Service = z.infer<typeof ServiceSchema>;
export const ServiceProviderSchema = z.object({
id: z.number(),
name: z.string(),
jobTitleEt: z.string().nullable(),
jobTitleEn: z.string().nullable(),
jobTitleRu: z.string().nullable(),
spokenLanguages: z.array(z.string()).nullable(),
});
export type ServiceProvider = z.infer<typeof ServiceProviderSchema>;
export const ReservationSchema = z.object({
startTime: z.string(),
service: ServiceSchema.optional(),
location: LocationSchema.optional(),
serviceProvider: ServiceProviderSchema.optional(),
medusaCartLineItemId: z.string(),
id: z.number(),
countryCode: z.string().optional(),
});
export type Reservation = z.infer<typeof ReservationSchema>;

View File

@@ -1,9 +1,9 @@
import { Database } from '@/packages/supabase/src/database.types';
import { type ClassValue, clsx } from 'clsx'; import { type ClassValue, clsx } from 'clsx';
import Isikukood, { Gender } from 'isikukood'; import Isikukood, { Gender } from 'isikukood';
import { twMerge } from 'tailwind-merge'; import { twMerge } from 'tailwind-merge';
import { BmiCategory } from './types/bmi'; import { BmiCategory } from './types/bmi';
import type { BmiThresholds } from '@kit/accounts/types/accounts';
export function cn(...inputs: ClassValue[]) { export function cn(...inputs: ClassValue[]) {
return twMerge(clsx(inputs)); return twMerge(clsx(inputs));
@@ -45,10 +45,7 @@ export const bmiFromMetric = (kg: number, cm: number) => {
}; };
export function getBmiStatus( export function getBmiStatus(
thresholds: Omit< thresholds: Omit<BmiThresholds, 'id'>[],
Database['medreport']['Tables']['bmi_thresholds']['Row'],
'id'
>[],
params: { age: number; height: number; weight: number }, params: { age: number; height: number; weight: number },
): BmiCategory | null { ): BmiCategory | null {
const age = params.age; const age = params.age;
@@ -143,3 +140,10 @@ export default class PersonalCode {
}; };
} }
} }
export const findProductTypeIdByHandle = (
productTypes: { metadata?: Record<string, unknown> | null; id: string }[],
handle: string,
) => {
return productTypes.find(({ metadata }) => metadata?.handle === handle)?.id;
};

View File

@@ -37,10 +37,11 @@ interface MontonioOrderToken {
exp: number; exp: number;
} }
const { secretKey } = MontonioServerEnvSchema.parse({ const env = () =>
apiUrl: process.env.MONTONIO_API_URL, MontonioServerEnvSchema.parse({
secretKey: process.env.MONTONIO_SECRET_KEY, apiUrl: process.env.MONTONIO_API_URL,
}); secretKey: process.env.MONTONIO_SECRET_KEY,
});
export class MontonioWebhookHandlerService export class MontonioWebhookHandlerService
implements BillingWebhookHandlerService implements BillingWebhookHandlerService
@@ -50,6 +51,7 @@ export class MontonioWebhookHandlerService
async verifyWebhookSignature(request: Request) { async verifyWebhookSignature(request: Request) {
const logger = await getLogger(); const logger = await getLogger();
const { secretKey } = env();
let token: string; let token: string;
try { try {

View File

@@ -0,0 +1,61 @@
import {
Body,
Head,
Html,
Preview,
Tailwind,
Text,
render,
} from '@react-email/components';
import { BodyStyle } from '../components/body-style';
import { EmailContent } from '../components/content';
import { EmailHeader } from '../components/header';
import { EmailHeading } from '../components/heading';
import { EmailWrapper } from '../components/wrapper';
export async function renderBookTimeFailedEmail({
reservationId,
error,
}: {
reservationId: number;
error: string;
}) {
const subject = 'Aja broneerimine ei õnnestunud';
const html = await render(
<Html>
<Head>
<BodyStyle />
</Head>
<Preview>{subject}</Preview>
<Tailwind>
<Body>
<EmailWrapper>
<EmailContent>
<EmailHeader>
<EmailHeading>{subject}</EmailHeading>
</EmailHeader>
<Text className="text-[16px] leading-[24px] text-[#242424]">
Tere
</Text>
<Text>
Broneeringu {reservationId} Connected Online'i saatmine ei
õnnestunud, kliendile tuleb teha tagasimakse.
</Text>
<Text>Saadud error: {error}</Text>
</EmailContent>
</EmailWrapper>
</Body>
</Tailwind>
</Html>,
);
return {
html,
subject,
};
}

View File

@@ -10,3 +10,4 @@ export * from './emails/all-results-received.email';
export * from './emails/order-processing.email'; export * from './emails/order-processing.email';
export * from './emails/patient-first-results-received.email'; export * from './emails/patient-first-results-received.email';
export * from './emails/patient-full-results-received.email'; export * from './emails/patient-full-results-received.email';
export * from './emails/book-time-failed.email';

View File

@@ -12,6 +12,7 @@
"./personal-account-settings": "./src/components/personal-account-settings/index.ts", "./personal-account-settings": "./src/components/personal-account-settings/index.ts",
"./components": "./src/components/index.ts", "./components": "./src/components/index.ts",
"./hooks/*": "./src/hooks/*.ts", "./hooks/*": "./src/hooks/*.ts",
"./services/*": "./src/server/services/*.ts",
"./api": "./src/server/api.ts", "./api": "./src/server/api.ts",
"./types/*": "./src/types/*.ts" "./types/*": "./src/types/*.ts"
}, },

View File

@@ -0,0 +1,127 @@
import { getSupabaseServerClient } from '@kit/supabase/server-client';
import type { AccountBalanceEntry } from '../../types/account-balance-entry';
export type AccountBalanceSummary = {
totalBalance: number;
expiringSoon: number;
recentEntries: AccountBalanceEntry[];
};
export class AccountBalanceService {
private supabase: ReturnType<typeof getSupabaseServerClient>;
constructor() {
this.supabase = getSupabaseServerClient();
}
/**
* Get the current balance for a specific account
*/
async getAccountBalance(accountId: string): Promise<number> {
const { data, error } = await this.supabase
.schema('medreport')
.rpc('get_account_balance', {
p_account_id: accountId,
});
if (error) {
console.error('Error getting account balance:', error);
throw new Error('Failed to get account balance');
}
return data || 0;
}
/**
* Get balance entries for an account with pagination
*/
async getAccountBalanceEntries(
accountId: string,
options: {
limit?: number;
offset?: number;
entryType?: string;
includeInactive?: boolean;
} = {}
): Promise<{
entries: AccountBalanceEntry[];
total: number;
}> {
const { limit = 50, offset = 0, entryType, includeInactive = false } = options;
let query = this.supabase
.schema('medreport')
.from('account_balance_entries')
.select('*', { count: 'exact' })
.eq('account_id', accountId)
.order('created_at', { ascending: false })
.range(offset, offset + limit - 1);
if (!includeInactive) {
query = query.eq('is_active', true);
}
if (entryType) {
query = query.eq('entry_type', entryType);
}
const { data, error, count } = await query;
if (error) {
console.error('Error getting account balance entries:', error);
throw new Error('Failed to get account balance entries');
}
return {
entries: data || [],
total: count || 0,
};
}
/**
* Get balance summary for dashboard display
*/
async getBalanceSummary(accountId: string): Promise<AccountBalanceSummary> {
const [balance, entries] = await Promise.all([
this.getAccountBalance(accountId),
this.getAccountBalanceEntries(accountId, { limit: 5 }),
]);
// Calculate expiring balance (next 30 days)
const thirtyDaysFromNow = new Date();
thirtyDaysFromNow.setDate(thirtyDaysFromNow.getDate() + 30);
const { data: expiringData, error: expiringError } = await this.supabase
.schema('medreport')
.from('account_balance_entries')
.select('amount')
.eq('account_id', accountId)
.eq('is_active', true)
.not('expires_at', 'is', null)
.lte('expires_at', thirtyDaysFromNow.toISOString());
if (expiringError) {
console.error('Error getting expiring balance:', expiringError);
}
const expiringSoon = expiringData?.reduce((sum, entry) => sum + (entry.amount || 0), 0) || 0;
return {
totalBalance: balance,
expiringSoon,
recentEntries: entries.entries,
};
}
async processPeriodicBenefitDistributions(): Promise<void> {
console.info('Processing periodic benefit distributions...');
const { error } = await this.supabase.schema('medreport').rpc('process_periodic_benefit_distributions');
if (error) {
console.error('Error processing periodic benefit distributions:', error);
throw new Error('Failed to process periodic benefit distributions');
}
console.info('Periodic benefit distributions processed successfully');
}
}

View File

@@ -0,0 +1,3 @@
import type { Database } from '@kit/supabase/database';
export type AccountBalanceEntry = Database['medreport']['Tables']['account_balance_entries']['Row'];

View File

@@ -1,23 +1,25 @@
import { Database } from '@kit/supabase/database'; import { Database } from '@kit/supabase/database';
export type ApplicationRole = export type ApplicationRole = Account['application_role'];
Database['medreport']['Tables']['accounts']['Row']['application_role'];
export enum ApplicationRoleEnum { export enum ApplicationRoleEnum {
User = 'user', User = 'user',
Doctor = 'doctor', Doctor = 'doctor',
SuperAdmin = 'super_admin', SuperAdmin = 'super_admin',
} }
export type AccountWithParams = export type AccountParams =
Database['medreport']['Tables']['accounts']['Row'] & { Database['medreport']['Tables']['account_params']['Row'];
accountParams:
| (Pick< export type Account = Database['medreport']['Tables']['accounts']['Row'];
Database['medreport']['Tables']['account_params']['Row'], export type AccountWithParams = Account & {
'weight' | 'height' accountParams:
> & { | (Pick<AccountParams, 'weight' | 'height'> & {
isSmoker: isSmoker: AccountParams['is_smoker'] | null;
| Database['medreport']['Tables']['account_params']['Row']['is_smoker'] })
| null; | null;
}) };
| null;
}; export type CompanyParams =
Database['medreport']['Tables']['company_params']['Row'];
export type BmiThresholds = Database['medreport']['Tables']['bmi_thresholds']['Row'];

View File

@@ -9,6 +9,7 @@
"devDependencies": { "devDependencies": {
"@hookform/resolvers": "^5.0.1", "@hookform/resolvers": "^5.0.1",
"@kit/next": "workspace:*", "@kit/next": "workspace:*",
"@kit/accounts": "workspace:*",
"@kit/shared": "workspace:*", "@kit/shared": "workspace:*",
"@kit/supabase": "workspace:*", "@kit/supabase": "workspace:*",
"@kit/tsconfig": "workspace:*", "@kit/tsconfig": "workspace:*",

View File

@@ -11,7 +11,7 @@ import { EllipsisVertical } from 'lucide-react';
import { useForm } from 'react-hook-form'; import { useForm } from 'react-hook-form';
import { z } from 'zod'; import { z } from 'zod';
import { Database } from '@kit/supabase/database'; import type { Account } from '@kit/accounts/types/accounts';
import { Button } from '@kit/ui/button'; import { Button } from '@kit/ui/button';
import { Checkbox } from '@kit/ui/checkbox'; import { Checkbox } from '@kit/ui/checkbox';
import { import {
@@ -44,8 +44,6 @@ import { AdminDeleteUserDialog } from './admin-delete-user-dialog';
import { AdminImpersonateUserDialog } from './admin-impersonate-user-dialog'; import { AdminImpersonateUserDialog } from './admin-impersonate-user-dialog';
import { AdminResetPasswordDialog } from './admin-reset-password-dialog'; import { AdminResetPasswordDialog } from './admin-reset-password-dialog';
type Account = Database['medreport']['Tables']['accounts']['Row'];
const FiltersSchema = z.object({ const FiltersSchema = z.object({
type: z.enum(['all', 'team', 'personal']), type: z.enum(['all', 'team', 'personal']),
query: z.string().optional(), query: z.string().optional(),

View File

@@ -4,10 +4,10 @@ import Link from 'next/link';
import { ColumnDef } from '@tanstack/react-table'; import { ColumnDef } from '@tanstack/react-table';
import { formatDateAndTime } from '@kit/shared/utils';
import { Database } from '@kit/supabase/database'; import { Database } from '@kit/supabase/database';
import { DataTable } from '@kit/ui/enhanced-data-table'; import { DataTable } from '@kit/ui/enhanced-data-table';
import { ProfileAvatar } from '@kit/ui/profile-avatar'; import { ProfileAvatar } from '@kit/ui/profile-avatar';
import { formatDateAndTime } from '@kit/shared/utils';
type Memberships = type Memberships =
Database['medreport']['Functions']['get_account_members']['Returns'][number]; Database['medreport']['Functions']['get_account_members']['Returns'][number];

View File

@@ -1,6 +1,6 @@
import { z } from 'zod'; import { z } from 'zod';
import { Database } from '@kit/supabase/database'; import { ApplicationRole } from '@kit/accounts/types/accounts';
const ConfirmationSchema = z.object({ const ConfirmationSchema = z.object({
confirmation: z.custom<string>((value) => value === 'CONFIRM'), confirmation: z.custom<string>((value) => value === 'CONFIRM'),
@@ -19,9 +19,7 @@ export const DeleteAccountSchema = ConfirmationSchema.extend({
accountId: z.string().uuid(), accountId: z.string().uuid(),
}); });
type ApplicationRoleType =
Database['medreport']['Tables']['accounts']['Row']['application_role'];
export const UpdateAccountRoleSchema = z.object({ export const UpdateAccountRoleSchema = z.object({
accountId: z.string().uuid(), accountId: z.string().uuid(),
role: z.string() as z.ZodType<ApplicationRoleType>, role: z.string() as z.ZodType<ApplicationRole>,
}); });

View File

@@ -3,6 +3,7 @@ import 'server-only';
import { SupabaseClient } from '@supabase/supabase-js'; import { SupabaseClient } from '@supabase/supabase-js';
import { Database } from '@kit/supabase/database'; import { Database } from '@kit/supabase/database';
import type { ApplicationRole } from '@kit/accounts/types/accounts';
export function createAdminAccountsService(client: SupabaseClient<Database>) { export function createAdminAccountsService(client: SupabaseClient<Database>) {
return new AdminAccountsService(client); return new AdminAccountsService(client);
@@ -25,7 +26,7 @@ class AdminAccountsService {
async updateRole( async updateRole(
accountId: string, accountId: string,
role: Database['medreport']['Tables']['accounts']['Row']['application_role'], role: ApplicationRole,
) { ) {
const { error } = await this.adminClient const { error } = await this.adminClient
.schema('medreport') .schema('medreport')

View File

@@ -13,6 +13,7 @@
"@kit/supabase": "workspace:*", "@kit/supabase": "workspace:*",
"@kit/tsconfig": "workspace:*", "@kit/tsconfig": "workspace:*",
"@kit/ui": "workspace:*", "@kit/ui": "workspace:*",
"@kit/user-analyses": "workspace:*",
"@makerkit/data-loader-supabase-core": "^0.0.10", "@makerkit/data-loader-supabase-core": "^0.0.10",
"@makerkit/data-loader-supabase-nextjs": "^1.2.5", "@makerkit/data-loader-supabase-nextjs": "^1.2.5",
"@supabase/supabase-js": "2.49.4", "@supabase/supabase-js": "2.49.4",

View File

@@ -5,6 +5,7 @@ import { isBefore } from 'date-fns';
import { renderDoctorSummaryReceivedEmail } from '@kit/email-templates'; import { renderDoctorSummaryReceivedEmail } from '@kit/email-templates';
import { getFullName } from '@kit/shared/utils'; import { getFullName } from '@kit/shared/utils';
import { getSupabaseServerClient } from '@kit/supabase/server-client'; import { getSupabaseServerClient } from '@kit/supabase/server-client';
import { createUserAnalysesApi } from '@kit/user-analyses/api';
import { sendEmailFromTemplate } from '../../../../../../../lib/services/mailer.service'; import { sendEmailFromTemplate } from '../../../../../../../lib/services/mailer.service';
import { AnalysisResultDetails } from '../schema/doctor-analysis-detail-view.schema'; import { AnalysisResultDetails } from '../schema/doctor-analysis-detail-view.schema';
@@ -641,7 +642,14 @@ export async function submitFeedback(
} }
if (status === 'COMPLETED') { if (status === 'COMPLETED') {
const [{ data: recipient }, { data: analysisOrder }] = await Promise.all([ const { data: analysisOrder } = await supabase
.schema('medreport')
.from('analysis_orders')
.select('medusa_order_id, id')
.eq('id', analysisOrderId)
.limit(1)
.throwOnError();
const [{ data: recipient }] = await Promise.all([
supabase supabase
.schema('medreport') .schema('medreport')
.from('accounts') .from('accounts')
@@ -649,19 +657,10 @@ export async function submitFeedback(
.eq('is_personal_account', true) .eq('is_personal_account', true)
.eq('primary_owner_user_id', userId) .eq('primary_owner_user_id', userId)
.throwOnError(), .throwOnError(),
supabase createUserAnalysesApi(supabase).updateAnalysisOrderStatus({
.schema('medreport') orderId: analysisOrderId,
.from('analysis_orders') orderStatus: 'COMPLETED',
.select('medusa_order_id, id') }),
.eq('id', analysisOrderId)
.limit(1)
.throwOnError(),
supabase
.schema('medreport')
.from('analysis_orders')
.update({ status: 'COMPLETED' })
.eq('id', analysisOrderId)
.throwOnError(),
]); ]);
if (!recipient?.[0]?.email) { if (!recipient?.[0]?.email) {

View File

@@ -168,7 +168,12 @@ export async function addToCart({
}) })
.catch(medusaError); .catch(medusaError);
return cart; const newCart = await getOrSetCart(countryCode);
const addedItem = newCart.items?.filter(
(item) => !cart.items?.some((oldCartItem) => oldCartItem.id === item.id),
)?.[0];
return { newCart, addedItem };
} }
export async function updateLineItem({ export async function updateLineItem({
@@ -253,6 +258,37 @@ export async function setShippingMethod({
.catch(medusaError); .catch(medusaError);
} }
export async function initiateMultiPaymentSession(
cart: HttpTypes.StoreCart,
benefitsAmount: number,
) {
const headers = {
...(await getAuthHeaders()),
};
return sdk.client.fetch<unknown>(`/store/multi-payment`, {
method: 'POST',
body: { cartId: cart.id, benefitsAmount },
headers,
})
.then(async (response) => {
console.info('Payment session initiated:', response);
const cartCacheTag = await getCacheTag('carts');
revalidateTag(cartCacheTag);
return response as {
montonioPaymentSessionId: string | null;
companyBenefitsPaymentSessionId: string | null;
totalByBenefits: number;
totalByMontonio: number;
isFullyPaidByBenefits: boolean;
};
})
.catch((e) => {
console.error('Error initiating payment session:', e, JSON.stringify(Object.keys(e)));
return medusaError(e);
});
}
export async function initiatePaymentSession( export async function initiatePaymentSession(
cart: HttpTypes.StoreCart, cart: HttpTypes.StoreCart,
data: HttpTypes.StoreInitializePaymentSession, data: HttpTypes.StoreInitializePaymentSession,

View File

@@ -21,7 +21,6 @@ export const listCategories = async (query?: Record<string, any>) => {
...query, ...query,
}, },
next, next,
cache: 'force-cache',
}, },
) )
.then(({ product_categories }) => product_categories); .then(({ product_categories }) => product_categories);
@@ -57,7 +56,6 @@ export const getProductCategories = async ({
limit, limit,
}, },
next, next,
//cache: "force-cache",
}, },
); );
}; };

View File

@@ -6,7 +6,7 @@ import { HttpTypes } from '@medusajs/types';
import { getAuthHeaders, getCacheOptions } from './cookies'; import { getAuthHeaders, getCacheOptions } from './cookies';
export const retrieveOrder = async (id: string) => { export const retrieveOrder = async (id: string, allowCache = true) => {
const headers = { const headers = {
...(await getAuthHeaders()), ...(await getAuthHeaders()),
}; };
@@ -24,7 +24,7 @@ export const retrieveOrder = async (id: string) => {
}, },
headers, headers,
next, next,
cache: 'force-cache', ...(allowCache ? { cache: 'force-cache' } : {}),
}) })
.then(({ order }) => order) .then(({ order }) => order)
.catch((err) => medusaError(err)); .catch((err) => medusaError(err));
@@ -61,11 +61,6 @@ export const listOrders = async (
}; };
export const createTransferRequest = async ( export const createTransferRequest = async (
state: {
success: boolean;
error: string | null;
order: HttpTypes.StoreOrder | null;
},
formData: FormData, formData: FormData,
): Promise<{ ): Promise<{
success: boolean; success: boolean;

View File

@@ -15,7 +15,7 @@ export const listRegions = async () => {
.fetch<{ regions: HttpTypes.StoreRegion[] }>(`/store/regions`, { .fetch<{ regions: HttpTypes.StoreRegion[] }>(`/store/regions`, {
method: 'GET', method: 'GET',
next, next,
cache: 'force-cache', // cache: 'force-cache',
}) })
.then(({ regions }) => regions) .then(({ regions }) => regions)
.catch(medusaError); .catch(medusaError);

View File

@@ -1,6 +1,5 @@
import Image from 'next/image'; import Image from 'next/image';
import { useDismissNotification } from '@kit/notifications/hooks';
import { Heading } from '@kit/ui/heading'; import { Heading } from '@kit/ui/heading';
import { If } from '@kit/ui/if'; import { If } from '@kit/ui/if';
import { Separator } from '@kit/ui/separator'; import { Separator } from '@kit/ui/separator';
@@ -12,7 +11,7 @@ import { SignOutInvitationButton } from './sign-out-invitation-button';
export function AcceptInvitationContainer(props: { export function AcceptInvitationContainer(props: {
inviteToken: string; inviteToken: string;
email: string; fullName: string;
invitation: { invitation: {
id: string; id: string;
@@ -76,7 +75,7 @@ export function AcceptInvitationContainer(props: {
/> />
<InvitationSubmitButton <InvitationSubmitButton
email={props.email} fullName={props.fullName}
accountName={props.invitation.account.name} accountName={props.invitation.account.name}
/> />
</form> </form>

View File

@@ -7,7 +7,7 @@ import { Trans } from '@kit/ui/trans';
export function InvitationSubmitButton(props: { export function InvitationSubmitButton(props: {
accountName: string; accountName: string;
email: string; fullName: string;
}) { }) {
const { pending } = useFormStatus(); const { pending } = useFormStatus();
@@ -17,7 +17,7 @@ export function InvitationSubmitButton(props: {
i18nKey={pending ? 'teams:joiningTeam' : 'teams:continueAs'} i18nKey={pending ? 'teams:joiningTeam' : 'teams:continueAs'}
values={{ values={{
accountName: props.accountName, accountName: props.accountName,
email: props.email, fullName: props.fullName,
}} }}
/> />
</Button> </Button>

View File

@@ -20,6 +20,7 @@ import { If } from '@kit/ui/if';
import { Input } from '@kit/ui/input'; import { Input } from '@kit/ui/input';
import { ProfileAvatar } from '@kit/ui/profile-avatar'; import { ProfileAvatar } from '@kit/ui/profile-avatar';
import { Trans } from '@kit/ui/trans'; import { Trans } from '@kit/ui/trans';
import { formatCurrency } from '@kit/shared/utils';
import { RemoveMemberDialog } from './remove-member-dialog'; import { RemoveMemberDialog } from './remove-member-dialog';
import { RoleBadge } from './role-badge'; import { RoleBadge } from './role-badge';
@@ -42,6 +43,10 @@ type AccountMembersTableProps = {
userRoleHierarchy: number; userRoleHierarchy: number;
isPrimaryOwner: boolean; isPrimaryOwner: boolean;
canManageRoles: boolean; canManageRoles: boolean;
membersBenefitsUsage: {
personal_account_id: string;
benefit_amount: number;
}[];
}; };
export function AccountMembersTable({ export function AccountMembersTable({
@@ -51,6 +56,7 @@ export function AccountMembersTable({
isPrimaryOwner, isPrimaryOwner,
userRoleHierarchy, userRoleHierarchy,
canManageRoles, canManageRoles,
membersBenefitsUsage,
}: AccountMembersTableProps) { }: AccountMembersTableProps) {
const [search, setSearch] = useState(''); const [search, setSearch] = useState('');
const { t } = useTranslation('teams'); const { t } = useTranslation('teams');
@@ -73,6 +79,7 @@ export function AccountMembersTable({
currentUserId, currentUserId,
currentAccountId, currentAccountId,
currentRoleHierarchy: userRoleHierarchy, currentRoleHierarchy: userRoleHierarchy,
membersBenefitsUsage,
}); });
const filteredMembers = members const filteredMembers = members
@@ -122,9 +129,13 @@ function useGetColumns(
currentUserId: string; currentUserId: string;
currentAccountId: string; currentAccountId: string;
currentRoleHierarchy: number; currentRoleHierarchy: number;
membersBenefitsUsage: {
personal_account_id: string;
benefit_amount: number;
}[];
}, },
): ColumnDef<Members[0]>[] { ): ColumnDef<Members[0]>[] {
const { t } = useTranslation('teams'); const { t, i18n: { language } } = useTranslation('teams');
return useMemo( return useMemo(
() => [ () => [
@@ -168,6 +179,23 @@ function useGetColumns(
return row.original.personal_code ?? '-'; return row.original.personal_code ?? '-';
}, },
}, },
{
header: t('distributedBenefitsAmount'),
cell: ({ row }) => {
const benefitAmount = params.membersBenefitsUsage.find(
(usage) => usage.personal_account_id === row.original.id
)?.benefit_amount;
if (typeof benefitAmount !== 'number') {
return '-';
}
return formatCurrency({
currencyCode: 'EUR',
locale: language,
value: benefitAmount,
});
},
},
{ {
header: t('roleLabel'), header: t('roleLabel'),
cell: ({ row }) => { cell: ({ row }) => {
@@ -175,7 +203,7 @@ function useGetColumns(
const isPrimaryOwner = primary_owner_user_id === user_id; const isPrimaryOwner = primary_owner_user_id === user_id;
return ( return (
<span className={'flex items-center space-x-1'}> <span className={'flex items-center space-x-1 flex-wrap space-y-1 sm:space-y-0 sm:flex-nowrap'}>
<RoleBadge role={role} /> <RoleBadge role={role} />
<If condition={isPrimaryOwner}> <If condition={isPrimaryOwner}>

View File

@@ -13,6 +13,8 @@ import { TeamAccountDangerZone } from './team-account-danger-zone';
import { UpdateTeamAccountImage } from './update-team-account-image-container'; import { UpdateTeamAccountImage } from './update-team-account-image-container';
import { UpdateTeamAccountNameForm } from './update-team-account-name-form'; import { UpdateTeamAccountNameForm } from './update-team-account-name-form';
const SHOW_TEAM_LOGO = false as boolean;
export function TeamAccountSettingsContainer(props: { export function TeamAccountSettingsContainer(props: {
account: { account: {
name: string; name: string;
@@ -32,21 +34,23 @@ export function TeamAccountSettingsContainer(props: {
}) { }) {
return ( return (
<div className={'flex w-full flex-col space-y-4'}> <div className={'flex w-full flex-col space-y-4'}>
<Card> {SHOW_TEAM_LOGO && (
<CardHeader> <Card>
<CardTitle> <CardHeader>
<Trans i18nKey={'teams:settings.teamLogo'} /> <CardTitle>
</CardTitle> <Trans i18nKey={'teams:settings.teamLogo'} />
</CardTitle>
<CardDescription> <CardDescription>
<Trans i18nKey={'teams:settings.teamLogoDescription'} /> <Trans i18nKey={'teams:settings.teamLogoDescription'} />
</CardDescription> </CardDescription>
</CardHeader> </CardHeader>
<CardContent> <CardContent>
<UpdateTeamAccountImage account={props.account} /> <UpdateTeamAccountImage account={props.account} />
</CardContent> </CardContent>
</Card> </Card>
)}
<Card> <Card>
<CardHeader> <CardHeader>

View File

@@ -18,6 +18,7 @@ import { RenewInvitationSchema } from '../../schema/renew-invitation.schema';
import { UpdateInvitationSchema } from '../../schema/update-invitation.schema'; import { UpdateInvitationSchema } from '../../schema/update-invitation.schema';
import { createAccountInvitationsService } from '../services/account-invitations.service'; import { createAccountInvitationsService } from '../services/account-invitations.service';
import { createAccountPerSeatBillingService } from '../services/account-per-seat-billing.service'; import { createAccountPerSeatBillingService } from '../services/account-per-seat-billing.service';
import { AccountBalanceService } from '@kit/accounts/services/account-balance.service';
/** /**
* @name createInvitationsAction * @name createInvitationsAction
@@ -148,6 +149,7 @@ export const updateInvitationAction = enhanceAction(
export const acceptInvitationAction = enhanceAction( export const acceptInvitationAction = enhanceAction(
async (data: FormData, user) => { async (data: FormData, user) => {
const client = getSupabaseServerClient(); const client = getSupabaseServerClient();
const accountBalanceService = new AccountBalanceService();
const { inviteToken, nextPath } = AcceptInvitationSchema.parse( const { inviteToken, nextPath } = AcceptInvitationSchema.parse(
Object.fromEntries(data), Object.fromEntries(data),
@@ -171,6 +173,9 @@ export const acceptInvitationAction = enhanceAction(
throw new Error('Failed to accept invitation'); throw new Error('Failed to accept invitation');
} }
// Make sure new account gets company benefits added to balance
await accountBalanceService.processPeriodicBenefitDistributions();
// Increase the seats for the account // Increase the seats for the account
await perSeatBillingService.increaseSeats(accountId); await perSeatBillingService.increaseSeats(accountId);

View File

@@ -1,9 +1,7 @@
import { z } from 'zod'; import { z } from 'zod';
import { getLogger } from '@kit/shared/logger'; import { getLogger } from '@kit/shared/logger';
import { Database } from '@kit/supabase/database'; import type { Account } from '@kit/accounts/types/accounts';
type Account = Database['medreport']['Tables']['accounts']['Row'];
export function createAccountWebhooksService() { export function createAccountWebhooksService() {
return new AccountWebhooksService(); return new AccountWebhooksService();

View File

@@ -4,7 +4,7 @@ import type { UuringuVastus } from '@kit/shared/types/medipost-analysis';
import { toArray } from '@kit/shared/utils'; import { toArray } from '@kit/shared/utils';
import { Database } from '@kit/supabase/database'; import { Database } from '@kit/supabase/database';
import type { AnalysisOrder } from '../types/analysis-orders'; import type { AnalysisOrder, AnalysisOrderStatus } from '../types/analysis-orders';
import type { import type {
AnalysisResultDetailsElement, AnalysisResultDetailsElement,
AnalysisResultDetailsMapped, AnalysisResultDetailsMapped,
@@ -450,6 +450,32 @@ class UserAnalysesApi {
return data; return data;
} }
async updateAnalysisOrderStatus({
orderId,
medusaOrderId,
orderStatus,
}: {
orderId?: number;
medusaOrderId?: string;
orderStatus: AnalysisOrderStatus;
}) {
const orderIdParam = orderId;
const medusaOrderIdParam = medusaOrderId;
console.info(`Updating order id=${orderId} medusaOrderId=${medusaOrderId} status=${orderStatus}`);
if (!orderIdParam && !medusaOrderIdParam) {
throw new Error('Either orderId or medusaOrderId must be provided');
}
await this.client
.schema('medreport')
.rpc('update_analysis_order_status', {
order_id: orderIdParam ?? -1,
status_param: orderStatus,
medusa_order_id_param: medusaOrderIdParam ?? '',
})
.throwOnError();
}
} }
export function createUserAnalysesApi(client: SupabaseClient<Database>) { export function createUserAnalysesApi(client: SupabaseClient<Database>) {

View File

@@ -1,3 +1,4 @@
import { Tables } from '@kit/supabase/database'; import { Tables } from '@kit/supabase/database';
export type AnalysisOrder = Tables<{ schema: 'medreport' }, 'analysis_orders'>; export type AnalysisOrder = Tables<{ schema: 'medreport' }, 'analysis_orders'>;
export type AnalysisOrderStatus = AnalysisOrder['status'];

View File

@@ -1,7 +1,7 @@
import { adminNavigationConfig } from './admin-navigation.config';
import appConfig from './app.config'; import appConfig from './app.config';
import authConfig from './auth.config'; import authConfig from './auth.config';
import billingConfig from './billing.config'; import billingConfig from './billing.config';
import { adminNavigationConfig } from './admin-navigation.config';
import { import {
DynamicAuthConfig, DynamicAuthConfig,
getCachedAuthConfig, getCachedAuthConfig,

Some files were not shown because too many files have changed in this diff Show More