feat(MED-131): fix medusa vs medipost vs b2b product ids mixed

This commit is contained in:
2025-08-04 16:27:54 +03:00
parent 36816cfcd5
commit 8790b515d5
4 changed files with 88 additions and 71 deletions

View File

@@ -3,6 +3,7 @@ import { getOrder } from "~/lib/services/order.service";
import { composeOrderTestResponseXML, sendPrivateMessageTestResponse } from "~/lib/services/medipostTest.service"; import { composeOrderTestResponseXML, sendPrivateMessageTestResponse } from "~/lib/services/medipostTest.service";
import { retrieveOrder } from "@lib/data"; import { retrieveOrder } from "@lib/data";
import { getAccountAdmin } from "~/lib/services/account.service"; import { getAccountAdmin } from "~/lib/services/account.service";
import { getOrderedAnalysisElementsIds } from "~/lib/services/medipost.service";
export async function POST(request: Request) { export async function POST(request: Request) {
const isDev = process.env.NODE_ENV === 'development'; const isDev = process.env.NODE_ENV === 'development';
@@ -16,19 +17,9 @@ export async function POST(request: Request) {
const medreportOrder = await getOrder({ medusaOrderId }); const medreportOrder = await getOrder({ medusaOrderId });
const account = await getAccountAdmin({ primaryOwnerUserId: medreportOrder.user_id }); const account = await getAccountAdmin({ primaryOwnerUserId: medreportOrder.user_id });
const orderedAnalysisElementsIds = await getOrderedAnalysisElementsIds({ medusaOrder });
const ANALYSIS_ELEMENT_HANDLE_PREFIX = 'analysis-element-'; console.info(`Sending test response for order=${medusaOrderId} with ${orderedAnalysisElementsIds.length} ordered analysis elements`);
const orderedAnalysisElementsIds = (medusaOrder?.items ?? [])
.filter((item) => item.product?.handle?.startsWith(ANALYSIS_ELEMENT_HANDLE_PREFIX))
.map((item) => {
const id = Number(item.product?.handle?.replace(ANALYSIS_ELEMENT_HANDLE_PREFIX, ''));
if (Number.isNaN(id)) {
return null;
}
return id;
})
.filter(Boolean) as number[];
const messageXml = await composeOrderTestResponseXML({ const messageXml = await composeOrderTestResponseXML({
person: { person: {
idCode: account.personal_code!, idCode: account.personal_code!,
@@ -36,7 +27,7 @@ export async function POST(request: Request) {
lastName: account.last_name ?? '', lastName: account.last_name ?? '',
phone: account.phone ?? '', phone: account.phone ?? '',
}, },
orderedAnalysisElementsIds, orderedAnalysisElementsIds: orderedAnalysisElementsIds.map(({ analysisElementId }) => analysisElementId),
orderedAnalysesIds: [], orderedAnalysesIds: [],
orderId: medusaOrderId, orderId: medusaOrderId,
orderCreatedAt: new Date(medreportOrder.created_at), orderCreatedAt: new Date(medreportOrder.created_at),

View File

@@ -6,7 +6,7 @@ import { listProductTypes } from "@lib/data/products";
import { placeOrder, retrieveCart } from "@lib/data/cart"; import { placeOrder, retrieveCart } from "@lib/data/cart";
import { createI18nServerInstance } from "~/lib/i18n/i18n.server"; import { createI18nServerInstance } from "~/lib/i18n/i18n.server";
import { createOrder } from '~/lib/services/order.service'; import { createOrder } from '~/lib/services/order.service';
import { sendOrderToMedipost } from '~/lib/services/medipost.service'; import { getOrderedAnalysisElementsIds, sendOrderToMedipost } from '~/lib/services/medipost.service';
const emailSender = process.env.EMAIL_SENDER; const emailSender = process.env.EMAIL_SENDER;
const siteUrl = process.env.NEXT_PUBLIC_SITE_URL!; const siteUrl = process.env.NEXT_PUBLIC_SITE_URL!;
@@ -88,7 +88,8 @@ const handleOrderToken = async (orderToken: string) => {
} }
const medusaOrder = await placeOrder(cartId, { revalidateCacheTags: true }); const medusaOrder = await placeOrder(cartId, { revalidateCacheTags: true });
await createOrder({ medusaOrder: medusaOrder }); const orderedAnalysisElements = await getOrderedAnalysisElementsIds({ medusaOrder });
await createOrder({ medusaOrder, orderedAnalysisElements });
const { productTypes } = await listProductTypes(); const { productTypes } = await listProductTypes();
const analysisPackagesType = productTypes.find(({ metadata }) => metadata?.handle === ANALYSIS_PACKAGES_TYPE_HANDLE); const analysisPackagesType = productTypes.find(({ metadata }) => metadata?.handle === ANALYSIS_PACKAGES_TYPE_HANDLE);
@@ -98,6 +99,7 @@ const handleOrderToken = async (orderToken: string) => {
email: medusaOrder.email, email: medusaOrder.email,
partnerLocationName: analysisPackageOrderItem?.metadata?.partner_location_name as string ?? '', partnerLocationName: analysisPackageOrderItem?.metadata?.partner_location_name as string ?? '',
analysisPackageName: analysisPackageOrderItem?.title ?? '', analysisPackageName: analysisPackageOrderItem?.title ?? '',
orderedAnalysisElements,
}; };
} catch (error) { } catch (error) {
console.error("Failed to place order", error); console.error("Failed to place order", error);
@@ -124,15 +126,19 @@ export async function GET(request: Request) {
throw new Error("Order result is missing"); throw new Error("Order result is missing");
} }
const { medusaOrderId, email, partnerLocationName, analysisPackageName } = orderResult; const { medusaOrderId, email, partnerLocationName, analysisPackageName, orderedAnalysisElements } = orderResult;
const personName = account.name; const personName = account.name;
if (email && analysisPackageName) { if (email && analysisPackageName) {
try {
await sendEmail({ email, analysisPackageName, personName, partnerLocationName, language }); await sendEmail({ email, analysisPackageName, personName, partnerLocationName, language });
} catch (error) {
console.error("Failed to send email", error);
}
} else { } else {
// @TODO send email for separate analyses // @TODO send email for separate analyses
console.error("Missing email or analysisPackageName", orderResult); console.error("Missing email or analysisPackageName", orderResult);
} }
sendOrderToMedipost({ medusaOrderId }) sendOrderToMedipost({ medusaOrderId, orderedAnalysisElements })
return Response.redirect(new URL('/home/order', baseUrl)) return Response.redirect(new URL('/home/order', baseUrl))
} catch (error) { } catch (error) {
console.error("Failed to place order", error); console.error("Failed to place order", error);

View File

@@ -37,11 +37,10 @@ import { Tables } from '@kit/supabase/database';
import { createAnalysisGroup } from './analysis-group.service'; import { createAnalysisGroup } from './analysis-group.service';
import { getSupabaseServerAdminClient } from '@/packages/supabase/src/clients/server-admin-client'; import { getSupabaseServerAdminClient } from '@/packages/supabase/src/clients/server-admin-client';
import { getOrder } from './order.service'; import { getOrder } from './order.service';
import { getAnalysisElementsAdmin } from './analysis-element.service'; import { getAnalysisElements, getAnalysisElementsAdmin } from './analysis-element.service';
import { getAnalyses } from './analyses.service'; import { getAnalyses } from './analyses.service';
import { retrieveOrder } from '@lib/data/orders';
import { getAccountAdmin } from './account.service'; import { getAccountAdmin } from './account.service';
import { StoreProduct } from '@medusajs/types'; import { StoreOrder } from '@medusajs/types';
import { listProducts } from '@lib/data/products'; import { listProducts } from '@lib/data/products';
import { listRegions } from '@lib/data/regions'; import { listRegions } from '@lib/data/regions';
import { getAnalysisElementMedusaProductIds } from '@/utils/medusa-product'; import { getAnalysisElementMedusaProductIds } from '@/utils/medusa-product';
@@ -648,55 +647,14 @@ export async function syncPrivateMessage({
export async function sendOrderToMedipost({ export async function sendOrderToMedipost({
medusaOrderId, medusaOrderId,
orderedAnalysisElements,
}: { }: {
medusaOrderId: string; medusaOrderId: string;
orderedAnalysisElements: { analysisElementId: number }[];
}) { }) {
const [medusaOrder, medreportOrder, countryCodes] = await Promise.all([ const medreportOrder = await getOrder({ medusaOrderId });
retrieveOrder(medusaOrderId),
getOrder({ medusaOrderId }),
listRegions(),
]);
const countryCode = countryCodes[0]!.countries![0]!.iso_2!;
const account = await getAccountAdmin({ primaryOwnerUserId: medreportOrder.user_id }); const account = await getAccountAdmin({ primaryOwnerUserId: medreportOrder.user_id });
let analysisPackageElements: StoreProduct[] = [];
const orderedAnalysisPackages = (medusaOrder?.items ?? []).filter(({ product }) => product?.handle.startsWith(ANALYSIS_PACKAGE_HANDLE_PREFIX));
const orderedAnalysisElements = (medusaOrder?.items ?? []).filter(({ product }) => product?.handle.startsWith(ANALYSIS_ELEMENT_HANDLE_PREFIX));
if (orderedAnalysisPackages.length > 0) {
const { response: { products: analysisPackages } } = await listProducts({
countryCode,
queryParams: { limit: 100, "type_id[0]": orderedAnalysisPackages[0]!.product_type_id! },
});
const analysisElementMedusaProductIds = getAnalysisElementMedusaProductIds(analysisPackages);
const { response: { products } } = await listProducts({
countryCode,
queryParams: {
id: analysisElementMedusaProductIds,
limit: 100,
},
});
analysisPackageElements = products;
}
const analysisPackageElementsIds = analysisPackageElements
.map((product) => {
const id = Number(product?.metadata?.analysisIdOriginal);
if (Number.isNaN(id)) {
return null;
}
return id;
})
.filter(Boolean) as number[];
const orderedAnalysisElementsIds = orderedAnalysisElements
.map((line) => {
const id = Number(line.product?.handle?.replace(ANALYSIS_ELEMENT_HANDLE_PREFIX, ''));
if (Number.isNaN(id)) {
return null;
}
return id;
})
.filter(Boolean) as number[];
const orderXml = await composeOrderXML({ const orderXml = await composeOrderXML({
person: { person: {
idCode: account.personal_code!, idCode: account.personal_code!,
@@ -704,7 +662,7 @@ export async function sendOrderToMedipost({
lastName: account.last_name ?? '', lastName: account.last_name ?? '',
phone: account.phone ?? '', phone: account.phone ?? '',
}, },
orderedAnalysisElementsIds: [...analysisPackageElementsIds, ...orderedAnalysisElementsIds], orderedAnalysisElementsIds: orderedAnalysisElements.map(({ analysisElementId }) => analysisElementId),
orderedAnalysesIds: [], orderedAnalysesIds: [],
orderId: medusaOrderId, orderId: medusaOrderId,
orderCreatedAt: new Date(medreportOrder.created_at), orderCreatedAt: new Date(medreportOrder.created_at),
@@ -713,3 +671,67 @@ export async function sendOrderToMedipost({
await sendPrivateMessage(orderXml); await sendPrivateMessage(orderXml);
} }
export async function getOrderedAnalysisElementsIds({
medusaOrder,
}: {
medusaOrder: StoreOrder;
}): Promise<{
analysisElementId: number;
}[]> {
const countryCodes = await listRegions();
const countryCode = countryCodes[0]!.countries![0]!.iso_2!;
function getOrderedAnalysisElements(medusaOrder: StoreOrder) {
return (medusaOrder?.items ?? [])
.filter(({ product }) => product?.handle.startsWith(ANALYSIS_ELEMENT_HANDLE_PREFIX))
.map((line) => {
const analysisElementId = Number(line.product?.handle?.replace(ANALYSIS_ELEMENT_HANDLE_PREFIX, ''));
if (Number.isNaN(analysisElementId)) {
return null;
}
return { analysisElementId };
}) as { analysisElementId: number }[];
}
async function getOrderedAnalysisPackages(medusaOrder: StoreOrder) {
const orderedPackages = (medusaOrder?.items ?? []).filter(({ product }) => product?.handle.startsWith(ANALYSIS_PACKAGE_HANDLE_PREFIX));
const orderedPackageIds = orderedPackages.map(({ product }) => product?.id).filter(Boolean) as string[];
if (orderedPackageIds.length === 0) {
return [];
}
console.info(`Order has ${orderedPackageIds.length} packages`);
const { response: { products: orderedPackagesProducts } } = await listProducts({
countryCode,
queryParams: { limit: 100, id: orderedPackageIds },
});
console.info(`Order has ${orderedPackagesProducts.length} packages`);
if (orderedPackagesProducts.length !== orderedPackageIds.length) {
throw new Error(`Got ${orderedPackagesProducts.length} ordered packages products, expected ${orderedPackageIds.length}`);
}
const ids = getAnalysisElementMedusaProductIds(orderedPackagesProducts);
const { response: { products: analysisPackagesProducts } } = await listProducts({
countryCode,
queryParams: { limit: 100, id: ids },
});
if (analysisPackagesProducts.length !== ids.length) {
throw new Error(`Got ${analysisPackagesProducts.length} analysis packages products, expected ${ids.length}`);
}
const originalIds = analysisPackagesProducts
.map(({ metadata }) => metadata?.analysisIdOriginal)
.filter((id) => typeof id === 'string');
if (originalIds.length !== ids.length) {
throw new Error(`Got ${originalIds.length} analysis packages products with analysisIdOriginal, expected ${ids.length}`);
}
const analysisElements = await getAnalysisElements({ originalIds });
return analysisElements.map(({ id }) => ({ analysisElementId: id }));
}
const analysisPackageElements = await getOrderedAnalysisPackages(medusaOrder);
const orderedAnalysisElements = getOrderedAnalysisElements(medusaOrder);
return [...analysisPackageElements, ...orderedAnalysisElements];
}

View File

@@ -5,15 +5,13 @@ import type { StoreOrder } from '@medusajs/types';
export async function createOrder({ export async function createOrder({
medusaOrder, medusaOrder,
orderedAnalysisElements,
}: { }: {
medusaOrder: StoreOrder; medusaOrder: StoreOrder;
orderedAnalysisElements: { analysisElementId: number }[];
}) { }) {
const supabase = getSupabaseServerClient(); const supabase = getSupabaseServerClient();
const analysisElementIds = medusaOrder.items
?.filter(({ product }) => product?.handle?.startsWith('analysis-element-'))
.map(({ product }) => Number(product?.handle.replace('analysis-element-', '')))
.filter((id) => !Number.isNaN(id)) as number[];
const { data: { user } } = await supabase.auth.getUser(); const { data: { user } } = await supabase.auth.getUser();
if (!user) { if (!user) {
throw new Error('User not found'); throw new Error('User not found');
@@ -21,7 +19,7 @@ export async function createOrder({
const orderResult = await supabase.schema('medreport') const orderResult = await supabase.schema('medreport')
.from('analysis_orders') .from('analysis_orders')
.insert({ .insert({
analysis_element_ids: analysisElementIds, analysis_element_ids: orderedAnalysisElements.map(({ analysisElementId }) => analysisElementId),
analysis_ids: [], analysis_ids: [],
status: 'QUEUED', status: 'QUEUED',
user_id: user.id, user_id: user.id,