fix(core-flows): fulfilment cancelation with shared inventory kit item (#12503)

* fix(core-flows): fulfilment cancelation with shared inventory kit item

* fix: typos, check if iitem exists

* chore: typo
This commit is contained in:
Frane Polić
2025-05-16 10:50:10 +02:00
committed by GitHub
parent e149a99886
commit 9f4d32b220
3 changed files with 616 additions and 33 deletions

View File

@@ -1375,7 +1375,7 @@ medusaIntegrationTestRunner({
])
)
// create a fulfillment for the entiretablet order
// create a fulfillment for the entire tablet order
const fulOrder = (
await api.post(
`/admin/orders/${tabletOrder.id}/fulfillments?fields=*fulfillments,*fulfillments.items`,
@@ -1394,7 +1394,7 @@ medusaIntegrationTestRunner({
).data.reservations
// no more reservations since everything is fuliflled
expect(reservations).toEqual(expect.arrayContaining([]))
expect(reservations.length).toEqual(0)
// cancel the fulfillment
await api.post(
@@ -1421,7 +1421,7 @@ medusaIntegrationTestRunner({
])
)
// create a fulfillment for the entiretablet order again
// create a fulfillment for the entire tablet order again
const fulOrder2 = (
await api.post(
`/admin/orders/${tabletOrder.id}/fulfillments?fields=*fulfillments,*fulfillments.items`,
@@ -1440,11 +1440,13 @@ medusaIntegrationTestRunner({
).data.reservations
// no more reservations since everything is fuliflled again
expect(reservations).toEqual(expect.arrayContaining([]))
expect(reservations.length).toEqual(0)
// cancel the fulfillment
await api.post(
`/admin/orders/${tabletOrder.id}/fulfillments/${fulOrder2.fulfillments[0].id}/cancel`,
`/admin/orders/${tabletOrder.id}/fulfillments/${
fulOrder2.fulfillments.find((f) => !f.canceled_at).id
}/cancel`,
{},
adminHeaders
)
@@ -1477,10 +1479,18 @@ medusaIntegrationTestRunner({
expect.objectContaining({
inventory_item_id: inventoryItemDesk.id,
quantity: 2,
inventory_item: expect.objectContaining({
reserved_quantity: 2,
stocked_quantity: 10,
}),
}),
expect.objectContaining({
inventory_item_id: inventoryItemLeg.id,
quantity: 8,
inventory_item: expect.objectContaining({
reserved_quantity: 8,
stocked_quantity: 40,
}),
}),
])
)
@@ -1514,8 +1524,12 @@ medusaIntegrationTestRunner({
expect(fulOrder.items[0].detail.fulfilled_quantity).toEqual(1)
reservations = (await api.get(`/admin/reservations`, adminHeaders)).data
.reservations
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemId}`,
adminHeaders
)
).data.reservations
// 3. reservations need to be reduced by half since we fulfilled 1 item out of 2 in the order
expect(reservations).toEqual(
@@ -1523,10 +1537,18 @@ medusaIntegrationTestRunner({
expect.objectContaining({
inventory_item_id: inventoryItemDesk.id,
quantity: 1,
inventory_item: expect.objectContaining({
reserved_quantity: 1,
stocked_quantity: 9,
}),
}),
expect.objectContaining({
inventory_item_id: inventoryItemLeg.id,
quantity: 4,
inventory_item: expect.objectContaining({
reserved_quantity: 4,
stocked_quantity: 36,
}),
}),
])
)
@@ -1540,8 +1562,12 @@ medusaIntegrationTestRunner({
expect(data.order.fulfillments[0].canceled_at).toBeDefined()
expect(data.order.items[0].detail.fulfilled_quantity).toEqual(0)
reservations = (await api.get(`/admin/reservations`, adminHeaders)).data
.reservations
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemId}`,
adminHeaders
)
).data.reservations
// 4. reservation qunatities are restored after partial fulfillment is canceled
expect(reservations).toEqual(
@@ -1549,10 +1575,18 @@ medusaIntegrationTestRunner({
expect.objectContaining({
inventory_item_id: inventoryItemDesk.id,
quantity: 2,
inventory_item: expect.objectContaining({
reserved_quantity: 2,
stocked_quantity: 10,
}),
}),
expect.objectContaining({
inventory_item_id: inventoryItemLeg.id,
quantity: 8,
inventory_item: expect.objectContaining({
reserved_quantity: 8,
stocked_quantity: 40,
}),
}),
])
)
@@ -1604,8 +1638,12 @@ medusaIntegrationTestRunner({
adminHeaders
)
reservations = (await api.get(`/admin/reservations`, adminHeaders)).data
.reservations
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemId}`,
adminHeaders
)
).data.reservations
// 8. reservation need to be restored to the initiall quantities
expect(reservations).toEqual(
@@ -1613,15 +1651,546 @@ medusaIntegrationTestRunner({
expect.objectContaining({
inventory_item_id: inventoryItemDesk.id,
quantity: 2,
inventory_item: expect.objectContaining({
reserved_quantity: 2,
stocked_quantity: 10,
}),
}),
expect.objectContaining({
inventory_item_id: inventoryItemLeg.id,
quantity: 8,
inventory_item: expect.objectContaining({
reserved_quantity: 8,
stocked_quantity: 40,
}),
}),
])
)
})
it("should correctly manage reservations (with shared inventory item case)", async () => {
const inventoryItemBottle = (
await api.post(
`/admin/inventory-items`,
{ sku: "bottle" },
adminHeaders
)
).data.inventory_item
await api.post(
`/admin/inventory-items/${inventoryItemBottle.id}/location-levels`,
{
location_id: stockLocation.id,
stocked_quantity: 100,
},
adminHeaders
)
const productBottle = (
await api.post(
"/admin/products",
{
title: `Bottle Packs`,
shipping_profile_id: shippingProfile.id,
options: [{ title: "packs", values: ["one", "two", "three"] }],
variants: [
{
title: "One Pack",
sku: "one-pack",
inventory_items: [
{
inventory_item_id: inventoryItemBottle.id,
required_quantity: 1,
},
],
prices: [
{
currency_code: "usd",
amount: 10,
},
],
options: {
packs: "one",
},
},
{
title: "Two Pack",
sku: "two-pack",
inventory_items: [
{
inventory_item_id: inventoryItemBottle.id,
required_quantity: 2,
},
],
prices: [
{
currency_code: "usd",
amount: 20,
},
],
options: {
packs: "two",
},
},
{
title: "Three Pack",
sku: "three-pack",
inventory_items: [
{
inventory_item_id: inventoryItemBottle.id,
required_quantity: 3,
},
],
prices: [
{
currency_code: "usd",
amount: 30,
},
],
options: {
packs: "three",
},
},
],
},
adminHeaders
)
).data.product
const cartBottle = (
await api.post(
`/store/carts`,
{
currency_code: "usd",
email: "tony@stark-industries.com",
region_id: region.id,
shipping_address: {
address_1: "test address 1",
address_2: "test address 2",
city: "ny",
country_code: "us",
province: "ny",
postal_code: "94016",
},
billing_address: {
address_1: "test billing address 1",
address_2: "test billing address 2",
city: "ny",
country_code: "us",
province: "ny",
postal_code: "94016",
},
sales_channel_id: salesChannel.id,
items: [
{ quantity: 2, variant_id: productBottle.variants[0].id },
{ quantity: 2, variant_id: productBottle.variants[1].id },
{ quantity: 2, variant_id: productBottle.variants[2].id },
],
},
storeHeaders
)
).data.cart
await api.post(
`/store/carts/${cartBottle.id}/shipping-methods`,
{ option_id: shippingOption.id },
storeHeaders
)
const paymentCollectionBottle = (
await api.post(
`/store/payment-collections`,
{
cart_id: cartBottle.id,
},
storeHeaders
)
).data.payment_collection
await api.post(
`/store/payment-collections/${paymentCollectionBottle.id}/payment-sessions`,
{ provider_id: "pp_system_default" },
storeHeaders
)
const bottleOrder = (
await api.post(
`/store/carts/${cartBottle.id}/complete`,
{},
storeHeaders
)
).data.order
const lineItemIds = bottleOrder.items.map((i) => i.id).join(",")
const onePackItemId = bottleOrder.items.find(
(i) => i.subtitle === "One Pack"
)!.id
const twoPackItemId = bottleOrder.items.find(
(i) => i.subtitle === "Two Pack"
)!.id
const threePackItemId = bottleOrder.items.find(
(i) => i.subtitle === "Three Pack"
)!.id
let reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
expect.objectContaining({
line_item_id: twoPackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 4,
}),
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 6,
}),
])
)
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 12, // 2 * (onepack + twopack + threepack)
stocked_quantity: 100,
})
)
// create a partial fulfillment only for one "Three Pack"
const fulOrder = (
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments?fields=*fulfillments,*fulfillments.items`,
{
items: [
{
id: threePackItemId,
quantity: 1,
},
],
},
adminHeaders
)
).data.order
expect(
fulOrder.items.find((i) => i.id === threePackItemId)!.detail
.fulfilled_quantity
).toEqual(1)
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
expect.objectContaining({
line_item_id: twoPackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 4,
}),
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 3, // This was partially fulfilled
}),
])
)
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 9,
stocked_quantity: 97,
})
)
// cancel the first partial fulfillment
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments/${fulOrder.fulfillments[0].id}/cancel`,
{},
adminHeaders
)
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
// reservations are restored to the initial quantities
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
expect.objectContaining({
line_item_id: twoPackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 4,
}),
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 6,
}),
])
)
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 12, // 2 * (onepack + twopack + threepack)
stocked_quantity: 100,
})
)
// create a partial fulfillment only for the entier "Two Pack" item
const fulOrder2 = (
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments?fields=*fulfillments,*fulfillments.items`,
{
items: [
{
id: bottleOrder.items.find((i) => i.subtitle === "Two Pack")!
.id,
quantity: 2,
},
],
},
adminHeaders
)
).data.order
expect(
fulOrder2.items.find((i) => i.id === twoPackItemId)!.detail
.fulfilled_quantity
).toEqual(2)
expect(
fulOrder2.items.find((i) => i.id === threePackItemId)!.detail
.fulfilled_quantity
).toEqual(0)
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
expect(reservations.length).toEqual(2)
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
// Two pack was fully fulfilled
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 6,
}),
])
)
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 8,
stocked_quantity: 96,
})
)
const latestFulfillment = fulOrder2.fulfillments.find(
(f) => !f.canceled_at
)!
// cancel the second partial fulfillment of the "Two Pack" item
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments/${latestFulfillment.id}/cancel`,
{},
adminHeaders
)
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
// reservations are restored to the initial quantities
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
expect.objectContaining({
line_item_id: twoPackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 4,
}),
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 6,
}),
])
)
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 12, // 2 * (onepack + twopack + threepack)
stocked_quantity: 100,
})
)
// finally create a full fulfillment for the entire order
const fulOrder3 = (
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments?fields=*fulfillments,*fulfillments.items`,
{
items: [
{
id: onePackItemId,
quantity: 2,
},
{
id: twoPackItemId,
quantity: 2,
},
{
id: threePackItemId,
quantity: 2,
},
],
},
adminHeaders
)
).data.order
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
expect(reservations.length).toEqual(0)
expect(fulOrder3.items).toEqual(
expect.arrayContaining([
expect.objectContaining({
id: onePackItemId,
detail: expect.objectContaining({
fulfilled_quantity: 2,
}),
}),
expect.objectContaining({
id: twoPackItemId,
detail: expect.objectContaining({
fulfilled_quantity: 2,
}),
}),
expect.objectContaining({
id: threePackItemId,
detail: expect.objectContaining({
fulfilled_quantity: 2,
}),
}),
])
)
// cancel the fulfillment for the entire order
await api.post(
`/admin/orders/${bottleOrder.id}/fulfillments/${
fulOrder3.fulfillments.find((f) => !f.canceled_at)!.id
}/cancel`,
{},
adminHeaders
)
reservations = (
await api.get(
`/admin/reservations?line_item_id[]=${lineItemIds}`,
adminHeaders
)
).data.reservations
// reservations are restored to the initial quantities
expect(reservations).toEqual(
expect.arrayContaining([
expect.objectContaining({
line_item_id: onePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 2,
}),
expect.objectContaining({
line_item_id: twoPackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 4,
}),
expect.objectContaining({
line_item_id: threePackItemId,
inventory_item_id: inventoryItemBottle.id,
quantity: 6,
}),
])
)
// inventory is back to the initial quantities
expect(reservations[0].inventory_item).toEqual(
expect.objectContaining({
id: inventoryItemBottle.id,
reserved_quantity: 12,
stocked_quantity: 100,
})
)
const finalOrder = (
await api.get(`/admin/orders/${bottleOrder.id}`, adminHeaders)
).data.order
expect(finalOrder.fulfillments.every((f) => f.canceled_at)).toEqual(
true
)
expect(
finalOrder.items.every((i) => i.detail.fulfilled_quantity === 0)
).toEqual(true)
})
it("should throw an error if the quantity to fulfill exceeds the reserved quantity (inventory kit case)", async () => {
let reservations = (await api.get(`/admin/reservations`, adminHeaders))
.data.reservations