fix(dashboard, product): product attributes update with a relation update (#13019)
* fix(dashboard, product): product attributes update with a relation update * fix: rm log * chore: refactor
This commit is contained in:
6
.changeset/polite-falcons-glow.md
Normal file
6
.changeset/polite-falcons-glow.md
Normal file
@@ -0,0 +1,6 @@
|
||||
---
|
||||
"@medusajs/dashboard": patch
|
||||
"@medusajs/product": patch
|
||||
---
|
||||
|
||||
fix(dashboard, product): update product attributes
|
||||
@@ -1813,6 +1813,110 @@ medusaIntegrationTestRunner({
|
||||
)
|
||||
})
|
||||
|
||||
it("updates products relations and attributes", async () => {
|
||||
const shortsCategory = (
|
||||
await api.post(
|
||||
"/admin/product-categories",
|
||||
{ name: "Shorts", is_internal: false, is_active: true },
|
||||
adminHeaders
|
||||
)
|
||||
).data.product_category
|
||||
|
||||
const pantsCategory = (
|
||||
await api.post(
|
||||
"/admin/product-categories",
|
||||
{ name: "Pants", is_internal: false, is_active: true },
|
||||
adminHeaders
|
||||
)
|
||||
).data.product_category
|
||||
|
||||
const payload = {
|
||||
title: "Test an update",
|
||||
weight: 100,
|
||||
length: 100,
|
||||
width: 100,
|
||||
height: 100,
|
||||
options: [{ title: "size", values: ["large", "small"] }],
|
||||
variants: [
|
||||
{
|
||||
options: { size: "large" },
|
||||
title: "New variant",
|
||||
prices: [
|
||||
{
|
||||
currency_code: "usd",
|
||||
amount: 200,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
}
|
||||
|
||||
const createdProduct = (
|
||||
await api.post("/admin/products", payload, adminHeaders)
|
||||
).data.product
|
||||
|
||||
let updatedProduct = (
|
||||
await api.post(
|
||||
`/admin/products/${createdProduct.id}`,
|
||||
{ weight: 20, length: null },
|
||||
adminHeaders
|
||||
)
|
||||
).data.product
|
||||
|
||||
expect(updatedProduct).toEqual(
|
||||
expect.objectContaining({
|
||||
weight: "20",
|
||||
length: null,
|
||||
width: "100",
|
||||
height: "100",
|
||||
})
|
||||
)
|
||||
|
||||
updatedProduct = (
|
||||
await api.post(
|
||||
`/admin/products/${createdProduct.id}?fields=+categories.id`,
|
||||
{ categories: [{ id: pantsCategory.id }] },
|
||||
adminHeaders
|
||||
)
|
||||
).data.product
|
||||
|
||||
expect(updatedProduct).toEqual(
|
||||
expect.objectContaining({
|
||||
weight: "20",
|
||||
length: null,
|
||||
width: "100",
|
||||
height: "100",
|
||||
categories: expect.arrayContaining([
|
||||
expect.objectContaining({
|
||||
id: pantsCategory.id,
|
||||
}),
|
||||
]),
|
||||
})
|
||||
)
|
||||
|
||||
updatedProduct = (
|
||||
await api.post(
|
||||
`/admin/products/${createdProduct.id}?fields=+categories.id`,
|
||||
{ weight: null, length: 20, width: 50 },
|
||||
adminHeaders
|
||||
)
|
||||
).data.product
|
||||
|
||||
expect(updatedProduct).toEqual(
|
||||
expect.objectContaining({
|
||||
weight: null,
|
||||
length: "20",
|
||||
width: "50",
|
||||
height: "100",
|
||||
categories: expect.arrayContaining([
|
||||
expect.objectContaining({
|
||||
id: pantsCategory.id,
|
||||
}),
|
||||
]),
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("updates a product (update prices, tags, update status, delete collection, delete type, replaces images)", async () => {
|
||||
const payload = {
|
||||
collection_id: null,
|
||||
|
||||
@@ -68,10 +68,10 @@ export const ProductAttributesForm = ({
|
||||
const handleSubmit = form.handleSubmit(async (data) => {
|
||||
await mutateAsync(
|
||||
{
|
||||
weight: data.weight ? data.weight : undefined,
|
||||
length: data.length ? data.length : undefined,
|
||||
width: data.width ? data.width : undefined,
|
||||
height: data.height ? data.height : undefined,
|
||||
weight: data.weight ? data.weight : null,
|
||||
length: data.length ? data.length : null,
|
||||
width: data.width ? data.width : null,
|
||||
height: data.height ? data.height : null,
|
||||
mid_code: data.mid_code,
|
||||
hs_code: data.hs_code,
|
||||
origin_country: data.origin_country,
|
||||
|
||||
@@ -32,14 +32,14 @@ export type CancelReturnValidateOrderInput = {
|
||||
* This step validates that a return can be canceled.
|
||||
* If the return is canceled, its fulfillment aren't canceled,
|
||||
* or it has received items, the step will throw an error.
|
||||
*
|
||||
*
|
||||
* :::note
|
||||
*
|
||||
*
|
||||
* You can retrieve a return details using [Query](https://docs.medusajs.com/learn/fundamentals/module-links/query),
|
||||
* or [useQueryGraphStep](https://docs.medusajs.com/resources/references/medusa-workflows/steps/useQueryGraphStep).
|
||||
*
|
||||
*
|
||||
* :::
|
||||
*
|
||||
*
|
||||
* @example
|
||||
* const data = cancelReturnValidateOrder({
|
||||
* orderReturn: {
|
||||
@@ -53,9 +53,7 @@ export type CancelReturnValidateOrderInput = {
|
||||
*/
|
||||
export const cancelReturnValidateOrder = createStep(
|
||||
"validate-return",
|
||||
({
|
||||
orderReturn,
|
||||
}: CancelReturnValidateOrderInput) => {
|
||||
({ orderReturn }: CancelReturnValidateOrderInput) => {
|
||||
const orderReturn_ = orderReturn as ReturnDTO & {
|
||||
payment_collections: PaymentCollectionDTO[]
|
||||
fulfillments: FulfillmentDTO[]
|
||||
@@ -92,12 +90,12 @@ export const cancelReturnValidateOrder = createStep(
|
||||
|
||||
export const cancelReturnWorkflowId = "cancel-return"
|
||||
/**
|
||||
* This workflow cancels a return. It's used by the
|
||||
* This workflow cancels a return. It's used by the
|
||||
* [Cancel Return Admin API Route](https://docs.medusajs.com/api/admin#returns_postreturnsidcancel).
|
||||
*
|
||||
*
|
||||
* You can use this workflow within your customizations or your own custom workflows, allowing you
|
||||
* to cancel a return in your custom flow.
|
||||
*
|
||||
*
|
||||
* @example
|
||||
* const { result } = await cancelReturnWorkflow(container)
|
||||
* .run({
|
||||
@@ -105,9 +103,9 @@ export const cancelReturnWorkflowId = "cancel-return"
|
||||
* return_id: "return_123",
|
||||
* }
|
||||
* })
|
||||
*
|
||||
*
|
||||
* @summary
|
||||
*
|
||||
*
|
||||
* Cancel a return.
|
||||
*/
|
||||
export const cancelReturnWorkflow = createWorkflow(
|
||||
|
||||
@@ -11,9 +11,9 @@ import { StepResponse, createStep } from "@medusajs/framework/workflows-sdk"
|
||||
*/
|
||||
export type UpdateProductsStepInput =
|
||||
| {
|
||||
/**
|
||||
* The filters to select the products to update.
|
||||
*/
|
||||
/**
|
||||
* The filters to select the products to update.
|
||||
*/
|
||||
selector: ProductTypes.FilterableProductProps
|
||||
/**
|
||||
* The data to update the products with.
|
||||
@@ -21,19 +21,19 @@ export type UpdateProductsStepInput =
|
||||
update: ProductTypes.UpdateProductDTO
|
||||
}
|
||||
| {
|
||||
/**
|
||||
* The data to create or update products.
|
||||
*/
|
||||
/**
|
||||
* The data to create or update products.
|
||||
*/
|
||||
products: ProductTypes.UpsertProductDTO[]
|
||||
}
|
||||
|
||||
export const updateProductsStepId = "update-products"
|
||||
/**
|
||||
* This step updates one or more products.
|
||||
*
|
||||
*
|
||||
* @example
|
||||
* To update products by their ID:
|
||||
*
|
||||
*
|
||||
* ```ts
|
||||
* const data = updateProductsStep({
|
||||
* products: [
|
||||
@@ -44,9 +44,9 @@ export const updateProductsStepId = "update-products"
|
||||
* ]
|
||||
* })
|
||||
* ```
|
||||
*
|
||||
*
|
||||
* To update products matching a filter:
|
||||
*
|
||||
*
|
||||
* ```ts
|
||||
* const data = updateProductsStep({
|
||||
* selector: {
|
||||
|
||||
@@ -343,12 +343,12 @@ export const updateProductsWorkflowId = "update-products"
|
||||
* allows you to update custom data models linked to the products.
|
||||
*
|
||||
* You can also use this workflow within your customizations or your own custom workflows, allowing you to wrap custom logic around product update.
|
||||
*
|
||||
*
|
||||
* :::note
|
||||
*
|
||||
* Learn more about adding rules to the product variant's prices in the Pricing Module's
|
||||
*
|
||||
* Learn more about adding rules to the product variant's prices in the Pricing Module's
|
||||
* [Price Rules](https://docs.medusajs.com/resources/commerce-modules/pricing/price-rules) documentation.
|
||||
*
|
||||
*
|
||||
* :::
|
||||
*
|
||||
* @example
|
||||
|
||||
@@ -8,6 +8,7 @@ import {
|
||||
MedusaError,
|
||||
isPresent,
|
||||
mergeMetadata,
|
||||
isDefined,
|
||||
} from "@medusajs/framework/utils"
|
||||
import { SqlEntityManager, wrap } from "@mikro-orm/postgresql"
|
||||
|
||||
@@ -57,10 +58,18 @@ export class ProductRepository extends DALUtils.mikroOrmBaseRepositoryFactory(
|
||||
height?: string | number
|
||||
width?: string | number
|
||||
}) {
|
||||
productToUpdate.weight = productToUpdate.weight?.toString()
|
||||
productToUpdate.length = productToUpdate.length?.toString()
|
||||
productToUpdate.height = productToUpdate.height?.toString()
|
||||
productToUpdate.width = productToUpdate.width?.toString()
|
||||
if (isDefined(productToUpdate.weight)) {
|
||||
productToUpdate.weight = productToUpdate.weight?.toString()
|
||||
}
|
||||
if (isDefined(productToUpdate.length)) {
|
||||
productToUpdate.length = productToUpdate.length?.toString()
|
||||
}
|
||||
if (isDefined(productToUpdate.height)) {
|
||||
productToUpdate.height = productToUpdate.height?.toString()
|
||||
}
|
||||
if (isDefined(productToUpdate.width)) {
|
||||
productToUpdate.width = productToUpdate.width?.toString()
|
||||
}
|
||||
}
|
||||
|
||||
async deepUpdate(
|
||||
@@ -72,6 +81,7 @@ export class ProductRepository extends DALUtils.mikroOrmBaseRepositoryFactory(
|
||||
context: Context = {}
|
||||
): Promise<InferEntityType<typeof Product>[]> {
|
||||
const productIdsToUpdate: string[] = []
|
||||
|
||||
productsToUpdate.forEach((productToUpdate) => {
|
||||
ProductRepository.#correctUpdateDTOTypes(productToUpdate)
|
||||
productIdsToUpdate.push(productToUpdate.id)
|
||||
@@ -151,7 +161,10 @@ export class ProductRepository extends DALUtils.mikroOrmBaseRepositoryFactory(
|
||||
}
|
||||
|
||||
if (isPresent(productToUpdate.metadata)) {
|
||||
productToUpdate.metadata = mergeMetadata(product.metadata ?? {}, productToUpdate.metadata)
|
||||
productToUpdate.metadata = mergeMetadata(
|
||||
product.metadata ?? {},
|
||||
productToUpdate.metadata
|
||||
)
|
||||
}
|
||||
|
||||
wrappedProduct.assign(productToUpdate)
|
||||
|
||||
Reference in New Issue
Block a user