Files
medusa-store/packages/modules/order/src/services/actions/create-exchange.ts
Carlos R. L. Rodrigues cc73802ab3 chore(order): dml (#10292)
* ../../core/types/src/dml/index.ts

* ../../core/types/src/dml/index.ts

* fix: relationships mapping

* handle nullable foreign keys types

* handle nullable foreign keys types

* handle nullable foreign keys types

* continue to update product category repository

* fix all product category repositories issues

* fix product category service types

* fix product module service types

* fix product module service types

* fix repository template type

* refactor: use a singleton DMLToMikroORM factory instance

Since the MikroORM MetadataStorage is global, we will also have to turn DML
to MikroORM entities conversion use a global bucket as well

* refactor: update product module to use DML in tests

* wip: tests

* WIP product linkable fixes

* continue type fixing and start test fixing

* test: fix more tests

* fix repository

* fix pivot table computaion + fix mikro orm repository

* fix many to many management and configuration

* fix many to many management and configuration

* fix many to many management and configuration

* update product tag relation configuration

* Introduce experimental dml hooks to fix some issues with categories

* more fixes

* fix product tests

* add missing id prefixes

* fix product category handle management

* test: fix more failing tests

* test: make it all green

* test: fix breaking tests

* fix: build issues

* fix: build issues

* fix: more breaking tests

* refactor: fix issues after merge

* refactor: fix issues after merge

* refactor: surpress types error

* test: fix DML failing tests

* improve many to many inference + tests

* Wip fix columns from product entity

* remove product model before create hook and manage handle validation and transformation at the service level

* test: fix breaking unit tests

* fix: product module service to not update handle on product update

* fix define link and joiner config

* test: fix joiner config test

* test: fix joiner config test

* fix joiner config primary keys

* Fix joiner config builder

* Fix joiner config builder

* test: remove only modifier from test

* refactor: remove hooks usage from product collection

* refactor: remove hooks usage from product-option

* refactor: remove hooks usage for computing category handle

* refactor: remove hooks usage from productCategory model

* refactor: remove hooks from DML

* refactor: remove cruft

* order dml

* cleanup

* re add foerign key indexes

* wip

* chore: remove unused types

* wip

* changes

* rm raw

* autoincrement

* wip

* rel

* refactor: cleanup

* migration and models configuration adjustments

* cleanup

* number searchable

* fix random ordering

* fix

* test: fix product-category tests

* test: update breaking DML tests

* test: array assertion to not care about ordering

* fix: temporarily apply id ordering for products

* types

* wip

* WIP type improvements

* update order models

* partially fix types temporarely

* rel

* fix: recursive type issue

* improve type inference breaks

* improve type inference breaks

* update models

* rm nullable

* default value

* repository

* update default value handling

* fix unit tests

* WIP

* toMikroORM

* fix relations

* cascades

* fix

* experimental dml hooks

* rm migration

* serial

* nullable autoincrement

* fix model

* model changes

* fix one to one DML

* order test

* fix addresses

* fix unit tests

* Re align dml entity name inference

* update model table name config

* update model table name config

* revert

* update return relation

* WIP

* hasOne

* models

* fix

* model

* initial commit

* cart service

* order module

* utils unit test

* index engine

* changeset

* merge

* fix hasOne with fk

* update

* free text filter per entity

* tests

* prod category

* property string many to many

* fix big number

* link modules migration set names

* merge

* shipping option rules

* serializer

* unit test

* fix test mikro orm init

* fix test mikro orm init

* Maintain merge object properties

* fix test mikro orm init

* prevent unit test from connecting to db

* wip

* fix test

* fix test

* link test

* schema

* models

* auto increment

* hook

* model hooks

* order

* wip

* orm version

* request return field

* fix return configuration on order model

* workflows

* core flows

* unit test

* test

* base repo

* test

* base repo

* test fix

* inventory move

* locking inventory

* test

* free text fix

* rm timeout mock

* migrate fulfillment values

* v6.4.3

* cleanup

* link-modules update sql

* revert test

* remove fake timers

---------

Co-authored-by: adrien2p <adrien.deperetti@gmail.com>
Co-authored-by: Harminder Virk <virk.officials@gmail.com>
Co-authored-by: Oli Juhl <59018053+olivermrbl@users.noreply.github.com>
2025-01-21 08:04:47 -05:00

319 lines
7.5 KiB
TypeScript

import {
Context,
CreateOrderChangeActionDTO,
OrderTypes,
} from "@medusajs/framework/types"
import {
ChangeActionType,
OrderChangeType,
ReturnStatus,
getShippingMethodsTotals,
isString,
promiseAll,
toMikroORMEntity,
} from "@medusajs/framework/utils"
import { OrderExchange, OrderExchangeItem, Return, ReturnItem } from "@models"
function createExchangeAndReturnEntities(em, data, order) {
const exchangeReference = em.create(toMikroORMEntity(OrderExchange), {
order_id: data.order_id,
order_version: order.version,
no_notification: data.no_notification,
allow_backorder: data.allow_backorder,
difference_due: data.difference_due,
})
const returnReference = em.create(toMikroORMEntity(Return), {
order_id: data.order_id,
order_version: order.version,
status: ReturnStatus.REQUESTED,
exchange_id: exchangeReference.id,
refund_amount: (data.refund_amount as unknown) ?? null,
})
exchangeReference.return = returnReference
return { exchangeReference, returnReference }
}
function createReturnItems(
em,
data,
exchangeReference,
returnReference,
actions
) {
return data.return_items?.map((item) => {
actions.push({
action: ChangeActionType.RETURN_ITEM,
reference: "return",
reference_id: returnReference.id,
details: {
reference_id: item.id,
quantity: item.quantity,
metadata: item.metadata,
},
})
return em.create(toMikroORMEntity(ReturnItem), {
item_id: item.id,
return_id: returnReference.id,
reason: item.reason,
quantity: item.quantity,
note: item.note,
metadata: item.metadata,
})
})
}
async function processAdditionalItems(
em,
service,
data,
order,
exchangeReference,
actions,
sharedContext
) {
const itemsToAdd: any[] = []
const additionalNewItems: any[] = []
const additionalItems: any[] = []
data.additional_items?.forEach((item) => {
const hasItem = item.id
? order.items.find((o) => o.item.id === item.id)
: false
if (hasItem) {
actions.push({
action: ChangeActionType.ITEM_ADD,
exchange_id: exchangeReference.id,
internal_note: item.internal_note,
reference: "exchange",
reference_id: exchangeReference.id,
details: {
reference_id: item.id,
quantity: item.quantity,
unit_price: item.unit_price ?? hasItem.item.unit_price,
metadata: item.metadata,
},
})
additionalItems.push(
em.create(toMikroORMEntity(OrderExchangeItem), {
item_id: item.id,
quantity: item.quantity,
note: item.note,
metadata: item.metadata,
is_additional_item: true,
})
)
} else {
itemsToAdd.push(item)
additionalNewItems.push(
em.create(toMikroORMEntity(OrderExchangeItem), {
quantity: item.quantity,
unit_price: item.unit_price,
note: item.note,
metadata: item.metadata,
is_additional_item: true,
})
)
}
})
const createItems = await service.orderLineItemService_.create(
itemsToAdd,
sharedContext
)
createItems.forEach((item, index) => {
const addedItem = itemsToAdd[index]
additionalNewItems[index].item_id = item.id
actions.push({
action: ChangeActionType.ITEM_ADD,
exchange_id: exchangeReference.id,
internal_note: addedItem.internal_note,
reference: "exchange",
reference_id: exchangeReference.id,
details: {
reference_id: item.id,
exchange_id: exchangeReference.id,
quantity: addedItem.quantity,
unit_price: item.unit_price,
metadata: addedItem.metadata,
},
})
})
return additionalNewItems.concat(additionalItems)
}
async function processShippingMethods(
service,
data,
exchangeReference,
actions,
sharedContext
) {
for (const shippingMethod of data.shipping_methods ?? []) {
let shippingMethodId
if (!isString(shippingMethod)) {
const methods = await service.createOrderShippingMethods(
[
{
...shippingMethod,
order_id: data.order_id,
exchange_id: exchangeReference.id,
},
],
sharedContext
)
shippingMethodId = methods[0].id
} else {
shippingMethodId = shippingMethod
}
const method = await service.retrieveOrderShippingMethod(
shippingMethodId,
{ relations: ["tax_lines", "adjustments"] },
sharedContext
)
const calculatedAmount = getShippingMethodsTotals([method as any], {})[
method.id
]
actions.push({
action: ChangeActionType.SHIPPING_ADD,
reference: "order_shipping_method",
reference_id: shippingMethodId,
exchange_id: exchangeReference.id,
amount: calculatedAmount.total,
})
}
}
async function processReturnShipping(
service,
data,
exchangeReference,
returnReference,
actions,
sharedContext
) {
let returnShippingMethodId
if (!isString(data.return_shipping)) {
const methods = await service.createOrderShippingMethods(
[
{
...data.return_shipping,
order_id: data.order_id,
exchange_id: exchangeReference.id,
return_id: returnReference.id,
},
],
sharedContext
)
returnShippingMethodId = methods[0].id
} else {
returnShippingMethodId = data.return_shipping
}
const method = await service.retrieveOrderShippingMethod(
returnShippingMethodId,
{ relations: ["tax_lines", "adjustments"] },
sharedContext
)
const calculatedAmount = getShippingMethodsTotals([method as any], {})[
method.id
]
actions.push({
action: ChangeActionType.SHIPPING_ADD,
reference: "order_shipping_method",
reference_id: returnShippingMethodId,
return_id: returnReference.id,
exchange_id: exchangeReference.id,
amount: calculatedAmount.total,
})
}
export async function createExchange(
this: any,
data: OrderTypes.CreateOrderExchangeDTO,
sharedContext?: Context
) {
const order = await this.orderService_.retrieve(
data.order_id,
{ relations: ["items"] },
sharedContext
)
const actions: CreateOrderChangeActionDTO[] = []
const em = sharedContext!.transactionManager as any
const { exchangeReference, returnReference } =
createExchangeAndReturnEntities(em, data, order)
returnReference.items = createReturnItems(
em,
data,
exchangeReference,
returnReference,
actions
)
exchangeReference.additional_items = await processAdditionalItems(
em,
this,
data,
order,
exchangeReference,
actions,
sharedContext
)
await processShippingMethods(
this,
data,
exchangeReference,
actions,
sharedContext
)
await processReturnShipping(
this,
data,
exchangeReference,
returnReference,
actions,
sharedContext
)
const change = await this.createOrderChange_(
{
order_id: data.order_id,
exchange_id: exchangeReference.id,
return_id: returnReference.id,
change_type: OrderChangeType.EXCHANGE,
reference: "exchange",
reference_id: exchangeReference.id,
description: data.description,
internal_note: data.internal_note,
created_by: data.created_by,
metadata: data.metadata,
actions,
},
sharedContext
)
await promiseAll([
this.createReturns([returnReference], sharedContext),
this.createOrderExchanges([exchangeReference], sharedContext),
this.confirmOrderChange(change[0].id, sharedContext),
])
return exchangeReference
}