feat(medusa): Update shipping option price type (#5895)
**What** - allow updates to shipping option price type - Clean up the use of strings in the api validation Fixes #5824
This commit is contained in:
@@ -48,7 +48,7 @@ describe("/admin/shipping-options", () => {
|
||||
await db.teardown()
|
||||
})
|
||||
|
||||
it("updates a shipping option with no existing requirements", async () => {
|
||||
it("should update a shipping option with no existing requirements", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -66,11 +66,11 @@ describe("/admin/shipping-options", () => {
|
||||
],
|
||||
}
|
||||
|
||||
const res = await api.post(`/admin/shipping-options/test-out`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.post(
|
||||
`/admin/shipping-options/test-out`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
const requirements = res.data.shipping_option.requirements
|
||||
|
||||
@@ -92,7 +92,27 @@ describe("/admin/shipping-options", () => {
|
||||
)
|
||||
})
|
||||
|
||||
it("fails to add a a requirement with an id if it does not exists", async () => {
|
||||
it("should update a shipping option with price_type", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
price_type: "calculated",
|
||||
requirements: [],
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-out`, payload, adminReqConfig)
|
||||
.catch(console.log)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
expect(res.data.shipping_option).toEqual(
|
||||
expect.objectContaining({
|
||||
price_type: "calculated",
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should fail to add a a requirement with an id if it does not exists", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -113,11 +133,7 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-out`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
.post(`/admin/shipping-options/test-out`, payload, adminReqConfig)
|
||||
.catch((err) => {
|
||||
return err.response
|
||||
})
|
||||
@@ -128,7 +144,7 @@ describe("/admin/shipping-options", () => {
|
||||
)
|
||||
})
|
||||
|
||||
it("it successfully updates a set of existing requirements", async () => {
|
||||
it("should successfully updates a set of existing requirements", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -148,11 +164,11 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-option-req`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
.post(
|
||||
`/admin/shipping-options/test-option-req`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
.catch((err) => {
|
||||
console.log(err.response.data.message)
|
||||
})
|
||||
@@ -160,7 +176,7 @@ describe("/admin/shipping-options", () => {
|
||||
expect(res.status).toEqual(200)
|
||||
})
|
||||
|
||||
it("it successfully updates a set of existing requirements by updating one and deleting the other", async () => {
|
||||
it("should successfully updates a set of existing requirements by updating one and deleting the other", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -174,11 +190,11 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-option-req`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
.post(
|
||||
`/admin/shipping-options/test-option-req`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
.catch((err) => {
|
||||
console.log(err.response.data.message)
|
||||
})
|
||||
@@ -186,7 +202,7 @@ describe("/admin/shipping-options", () => {
|
||||
expect(res.status).toEqual(200)
|
||||
})
|
||||
|
||||
it("successfully updates a set of requirements because max. subtotal >= min. subtotal", async () => {
|
||||
it("should successfully updates a set of requirements because max. subtotal >= min. subtotal", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -205,11 +221,11 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-option-req`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
.post(
|
||||
`/admin/shipping-options/test-option-req`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
.catch((err) => {
|
||||
console.log(err.response.data.message)
|
||||
})
|
||||
@@ -219,7 +235,7 @@ describe("/admin/shipping-options", () => {
|
||||
expect(res.data.shipping_option.requirements[1].amount).toEqual(200)
|
||||
})
|
||||
|
||||
it("fails to updates a set of requirements because max. subtotal <= min. subtotal", async () => {
|
||||
it("should fail to updates a set of requirements because max. subtotal <= min. subtotal", async () => {
|
||||
const api = useApi()
|
||||
|
||||
const payload = {
|
||||
@@ -238,11 +254,11 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
|
||||
const res = await api
|
||||
.post(`/admin/shipping-options/test-option-req`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
.post(
|
||||
`/admin/shipping-options/test-option-req`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
.catch((err) => {
|
||||
return err.response
|
||||
})
|
||||
@@ -267,11 +283,7 @@ describe("/admin/shipping-options", () => {
|
||||
{
|
||||
fulfillment_providers: ["test-ful"],
|
||||
},
|
||||
{
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
}
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
const manager = dbConnection.manager
|
||||
@@ -297,7 +309,7 @@ describe("/admin/shipping-options", () => {
|
||||
await db.teardown()
|
||||
})
|
||||
|
||||
it("creates a shipping option with requirements", async () => {
|
||||
it("should create a shipping option with requirements", async () => {
|
||||
const api = useApi()
|
||||
payload.requirements = [
|
||||
{
|
||||
@@ -310,29 +322,29 @@ describe("/admin/shipping-options", () => {
|
||||
},
|
||||
]
|
||||
|
||||
const res = await api.post(`/admin/shipping-options`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.post(
|
||||
`/admin/shipping-options`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
expect(res.data.shipping_option.requirements.length).toEqual(2)
|
||||
})
|
||||
|
||||
it("creates a shipping option with no requirements", async () => {
|
||||
it("should create a shipping option with no requirements", async () => {
|
||||
const api = useApi()
|
||||
const res = await api.post(`/admin/shipping-options`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.post(
|
||||
`/admin/shipping-options`,
|
||||
payload,
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
expect(res.data.shipping_option.requirements.length).toEqual(0)
|
||||
})
|
||||
|
||||
it("fails on same requirement types", async () => {
|
||||
it("should fail on same requirement types", async () => {
|
||||
const api = useApi()
|
||||
payload.requirements = [
|
||||
{
|
||||
@@ -346,11 +358,7 @@ describe("/admin/shipping-options", () => {
|
||||
]
|
||||
|
||||
try {
|
||||
await api.post(`/admin/shipping-options`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
await api.post(`/admin/shipping-options`, payload, adminReqConfig)
|
||||
} catch (error) {
|
||||
expect(error.response.data.message).toEqual(
|
||||
"Only one requirement of each type is allowed"
|
||||
@@ -358,7 +366,7 @@ describe("/admin/shipping-options", () => {
|
||||
}
|
||||
})
|
||||
|
||||
it("fails when min_subtotal > max_subtotal", async () => {
|
||||
it("should fail when min_subtotal > max_subtotal", async () => {
|
||||
const api = useApi()
|
||||
payload.requirements = [
|
||||
{
|
||||
@@ -372,11 +380,7 @@ describe("/admin/shipping-options", () => {
|
||||
]
|
||||
|
||||
try {
|
||||
await api.post(`/admin/shipping-options`, payload, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
await api.post(`/admin/shipping-options`, payload, adminReqConfig)
|
||||
} catch (error) {
|
||||
expect(error.response.data.message).toEqual(
|
||||
"Max. subtotal must be greater than Min. subtotal"
|
||||
@@ -395,24 +399,19 @@ describe("/admin/shipping-options", () => {
|
||||
await db.teardown()
|
||||
})
|
||||
|
||||
it("lists shipping options", async () => {
|
||||
it("should list shipping options", async () => {
|
||||
const api = useApi()
|
||||
const res = await api.get(`/admin/shipping-options`, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.get(`/admin/shipping-options`, adminReqConfig)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
})
|
||||
|
||||
it("lists admin only shipping options", async () => {
|
||||
it("should list admin only shipping options", async () => {
|
||||
const api = useApi()
|
||||
const res = await api.get(`/admin/shipping-options?admin_only=true`, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.get(
|
||||
`/admin/shipping-options?admin_only=true`,
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
expect(res.data.shipping_options).toEqual(
|
||||
@@ -425,13 +424,12 @@ describe("/admin/shipping-options", () => {
|
||||
)
|
||||
})
|
||||
|
||||
it("lists return shipping options", async () => {
|
||||
it("should list return shipping options", async () => {
|
||||
const api = useApi()
|
||||
const res = await api.get(`/admin/shipping-options?is_return=true`, {
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
})
|
||||
const res = await api.get(
|
||||
`/admin/shipping-options?is_return=true`,
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
expect(res.data.shipping_options).toEqual(
|
||||
@@ -444,15 +442,11 @@ describe("/admin/shipping-options", () => {
|
||||
)
|
||||
})
|
||||
|
||||
it("lists shipping options without return and admin options", async () => {
|
||||
it("should list shipping options without return and admin options", async () => {
|
||||
const api = useApi()
|
||||
const res = await api.get(
|
||||
`/admin/shipping-options?is_return=false&admin_only=true`,
|
||||
{
|
||||
headers: {
|
||||
"x-medusa-access-token": "test_token",
|
||||
},
|
||||
}
|
||||
adminReqConfig
|
||||
)
|
||||
|
||||
expect(res.status).toEqual(200)
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
import { IdMap } from "medusa-test-utils"
|
||||
import { request } from "../../../../../helpers/test-request"
|
||||
import { ShippingOptionServiceMock } from "../../../../../services/__mocks__/shipping-option"
|
||||
import { request } from "../../../../../helpers/test-request"
|
||||
|
||||
describe("POST /admin/shipping-options", () => {
|
||||
describe("successful creation", () => {
|
||||
|
||||
@@ -1,19 +1,23 @@
|
||||
import {
|
||||
IsArray,
|
||||
IsBoolean,
|
||||
IsEnum,
|
||||
IsNumber,
|
||||
IsObject,
|
||||
IsOptional,
|
||||
IsString,
|
||||
ValidateNested,
|
||||
} from "class-validator"
|
||||
import { RequirementType, ShippingOptionPriceType } from "../../../../models"
|
||||
import { defaultFields, defaultRelations } from "."
|
||||
|
||||
import { Type } from "class-transformer"
|
||||
import { EntityManager } from "typeorm"
|
||||
import TaxInclusivePricingFeatureFlag from "../../../../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import { FeatureFlagDecorators } from "../../../../utils/feature-flag-decorators"
|
||||
import { ShippingOptionService } from "../../../../services"
|
||||
import TaxInclusivePricingFeatureFlag from "../../../../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import { Type } from "class-transformer"
|
||||
import { validator } from "../../../../utils/validator"
|
||||
import { CreateShippingOptionInput } from "../../../../types/shipping-options"
|
||||
|
||||
/**
|
||||
* @oas [post] /admin/shipping-options
|
||||
@@ -88,7 +92,9 @@ import { validator } from "../../../../utils/validator"
|
||||
export default async (req, res) => {
|
||||
const validated = await validator(AdminPostShippingOptionsReq, req.body)
|
||||
|
||||
const optionService = req.scope.resolve("shippingOptionService")
|
||||
const optionService: ShippingOptionService = req.scope.resolve(
|
||||
"shippingOptionService"
|
||||
)
|
||||
const shippingProfileService = req.scope.resolve("shippingProfileService")
|
||||
|
||||
// Add to default shipping profile
|
||||
@@ -101,7 +107,7 @@ export default async (req, res) => {
|
||||
const result = await manager.transaction(async (transactionManager) => {
|
||||
return await optionService
|
||||
.withTransaction(transactionManager)
|
||||
.create(validated)
|
||||
.create(validated as CreateShippingOptionInput)
|
||||
})
|
||||
|
||||
const data = await optionService.retrieve(result.id, {
|
||||
@@ -113,8 +119,11 @@ export default async (req, res) => {
|
||||
}
|
||||
|
||||
class OptionRequirement {
|
||||
@IsString()
|
||||
type: string
|
||||
@IsEnum(RequirementType, {
|
||||
message: `Invalid option type, must be one of "min_subtotal" or "max_subtotal"`,
|
||||
})
|
||||
type: RequirementType
|
||||
|
||||
@IsNumber()
|
||||
amount: number
|
||||
}
|
||||
@@ -203,15 +212,17 @@ export class AdminPostShippingOptionsReq {
|
||||
@IsString()
|
||||
provider_id: string
|
||||
|
||||
@IsOptional()
|
||||
@IsString()
|
||||
@IsOptional()
|
||||
profile_id?: string
|
||||
|
||||
@IsObject()
|
||||
data: object
|
||||
data: Record<string, unknown>
|
||||
|
||||
@IsString()
|
||||
price_type: string
|
||||
@IsEnum(ShippingOptionPriceType, {
|
||||
message: `Invalid price type, must be one of "flat_rate" or "calculated"`,
|
||||
})
|
||||
price_type: ShippingOptionPriceType
|
||||
|
||||
@IsOptional()
|
||||
@IsNumber()
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
import {
|
||||
IsArray,
|
||||
IsBoolean,
|
||||
IsEnum,
|
||||
IsNumber,
|
||||
IsObject,
|
||||
IsOptional,
|
||||
@@ -9,13 +10,14 @@ import {
|
||||
} from "class-validator"
|
||||
import { defaultFields, defaultRelations } from "."
|
||||
|
||||
import { Type } from "class-transformer"
|
||||
import { EntityManager } from "typeorm"
|
||||
import TaxInclusivePricingFeatureFlag from "../../../../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import { FeatureFlagDecorators } from "../../../../utils/feature-flag-decorators"
|
||||
import { validator } from "../../../../utils/validator"
|
||||
import { ShippingOptionPriceType } from "../../../../models"
|
||||
import { ShippingOptionService } from "../../../../services"
|
||||
import TaxInclusivePricingFeatureFlag from "../../../../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import { Type } from "class-transformer"
|
||||
import { UpdateShippingOptionInput } from "../../../../types/shipping-options"
|
||||
import { validator } from "../../../../utils/validator"
|
||||
|
||||
/**
|
||||
* @oas [post] /admin/shipping-options/{id}
|
||||
@@ -123,8 +125,10 @@ class OptionRequirement {
|
||||
@IsString()
|
||||
@IsOptional()
|
||||
id: string
|
||||
|
||||
@IsString()
|
||||
type: string
|
||||
|
||||
@IsNumber()
|
||||
amount: number
|
||||
}
|
||||
@@ -187,6 +191,12 @@ export class AdminPostShippingOptionsOptionReq {
|
||||
@IsOptional()
|
||||
amount?: number
|
||||
|
||||
@IsEnum(ShippingOptionPriceType, {
|
||||
message: `Invalid price type, must be one of "flat_rate" or "calculated"`,
|
||||
})
|
||||
@IsOptional()
|
||||
price_type?: ShippingOptionPriceType
|
||||
|
||||
@IsArray()
|
||||
@ValidateNested({ each: true })
|
||||
@Type(() => OptionRequirement)
|
||||
|
||||
@@ -15,7 +15,7 @@ import { generateEntityId } from "../utils/generate-entity-id"
|
||||
|
||||
/**
|
||||
* @enum
|
||||
*
|
||||
*
|
||||
* The type of shipping option requirement.
|
||||
*/
|
||||
export enum RequirementType {
|
||||
|
||||
@@ -1,8 +1,3 @@
|
||||
import { FlagRouter, promiseAll } from "@medusajs/utils"
|
||||
import { isDefined, MedusaError } from "medusa-core-utils"
|
||||
import { EntityManager } from "typeorm"
|
||||
import { TransactionBaseService } from "../interfaces"
|
||||
import TaxInclusivePricingFeatureFlag from "../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import {
|
||||
Cart,
|
||||
Order,
|
||||
@@ -11,20 +6,27 @@ import {
|
||||
ShippingOptionPriceType,
|
||||
ShippingOptionRequirement,
|
||||
} from "../models"
|
||||
import { ShippingMethodRepository } from "../repositories/shipping-method"
|
||||
import { ShippingOptionRepository } from "../repositories/shipping-option"
|
||||
import { ShippingOptionRequirementRepository } from "../repositories/shipping-option-requirement"
|
||||
import { FindConfig, Selector } from "../types/common"
|
||||
import {
|
||||
CreateShippingMethodDto,
|
||||
CreateShippingOptionInput,
|
||||
ShippingMethodUpdate,
|
||||
UpdateShippingOptionInput,
|
||||
ValidatePriceTypeAndAmountInput,
|
||||
ValidateRequirementTypeInput,
|
||||
} from "../types/shipping-options"
|
||||
import { FindConfig, Selector } from "../types/common"
|
||||
import { FlagRouter, promiseAll } from "@medusajs/utils"
|
||||
import { MedusaError, isDefined } from "medusa-core-utils"
|
||||
import { buildQuery, isString, setMetadata } from "../utils"
|
||||
|
||||
import { EntityManager } from "typeorm"
|
||||
import FulfillmentProviderService from "./fulfillment-provider"
|
||||
import RegionService from "./region"
|
||||
import { ShippingMethodRepository } from "../repositories/shipping-method"
|
||||
import { ShippingOptionRepository } from "../repositories/shipping-option"
|
||||
import { ShippingOptionRequirementRepository } from "../repositories/shipping-option-requirement"
|
||||
import TaxInclusivePricingFeatureFlag from "../loaders/feature-flags/tax-inclusive-pricing"
|
||||
import { TransactionBaseService } from "../interfaces"
|
||||
|
||||
type InjectedDependencies = {
|
||||
manager: EntityManager
|
||||
@@ -75,7 +77,7 @@ class ShippingOptionService extends TransactionBaseService {
|
||||
* @return {ShippingOptionRequirement} a validated shipping requirement
|
||||
*/
|
||||
async validateRequirement_(
|
||||
requirement: ShippingOptionRequirement,
|
||||
requirement: ValidateRequirementTypeInput,
|
||||
optionId: string | undefined = undefined
|
||||
): Promise<ShippingOptionRequirement> {
|
||||
return await this.atomicPhase_(async (manager) => {
|
||||
|
||||
@@ -1,10 +1,11 @@
|
||||
import { Cart, Order } from ".."
|
||||
import { ShippingOptionPriceType } from "../models/shipping-option"
|
||||
import {
|
||||
RequirementType,
|
||||
ShippingOptionRequirement,
|
||||
} from "../models/shipping-option-requirement"
|
||||
|
||||
import { ShippingOptionPriceType } from "../models/shipping-option"
|
||||
|
||||
export type ShippingRequirement = {
|
||||
type: RequirementType
|
||||
amount: number
|
||||
@@ -50,7 +51,15 @@ export type CreateShippingOptionInput = {
|
||||
is_return?: boolean
|
||||
admin_only?: boolean
|
||||
metadata?: Record<string, unknown>
|
||||
requirements?: ShippingOptionRequirement[]
|
||||
requirements?: RequirementInput[]
|
||||
}
|
||||
|
||||
export type RequirementInput = { type: RequirementType; amount: number }
|
||||
|
||||
export type ValidateRequirementTypeInput = {
|
||||
id?: string
|
||||
type: RequirementType
|
||||
amount: number
|
||||
}
|
||||
|
||||
export type CreateCustomShippingOptionInput = {
|
||||
@@ -67,7 +76,7 @@ export type UpdateShippingOptionInput = {
|
||||
name?: string
|
||||
admin_only?: boolean
|
||||
is_return?: boolean
|
||||
requirements?: ShippingOptionRequirement[]
|
||||
requirements?: RequirementInput[]
|
||||
region_id?: string
|
||||
provider_id?: string
|
||||
profile_id?: string
|
||||
|
||||
Reference in New Issue
Block a user