feat(MED-131): fix medusa vs medipost vs b2b product ids mixed
This commit is contained in:
@@ -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),
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
@@ -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];
|
||||||
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
Reference in New Issue
Block a user