fix(dashboard, medusa): validate provider for top level tax regions (#12450)
* fix(dashboard, medusa): validate provider for top level tax regions * chore: changesets, message * chore: add test case for province * fix: remove comment
This commit is contained in:
6
.changeset/old-carrots-develop.md
Normal file
6
.changeset/old-carrots-develop.md
Normal file
@@ -0,0 +1,6 @@
|
||||
---
|
||||
"@medusajs/dashboard": patch
|
||||
"@medusajs/medusa": patch
|
||||
---
|
||||
|
||||
fix(dashboard, medusa): validate provider exists when creating a top level tax region
|
||||
@@ -20,12 +20,22 @@ medusaIntegrationTestRunner({
|
||||
let taxRegion
|
||||
|
||||
beforeEach(async () => {
|
||||
const parentRegion = await api.post(
|
||||
"/admin/tax-regions",
|
||||
{
|
||||
country_code: "us",
|
||||
provider_id: "tp_system",
|
||||
},
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
taxRegion = (
|
||||
await api.post(
|
||||
"/admin/tax-regions",
|
||||
{
|
||||
country_code: "us",
|
||||
province_code: "tx",
|
||||
parent_id: parentRegion.data.tax_region.id,
|
||||
metadata: { test: "created" },
|
||||
},
|
||||
adminHeaders
|
||||
@@ -83,6 +93,49 @@ medusaIntegrationTestRunner({
|
||||
)
|
||||
})
|
||||
|
||||
it("should create a province tax region without a provider", async () => {
|
||||
const response = await api.post(
|
||||
`/admin/tax-regions`,
|
||||
{
|
||||
country_code: "us",
|
||||
parent_id: taxRegion.id,
|
||||
province_code: "ny",
|
||||
},
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.tax_region).toEqual(
|
||||
expect.objectContaining({
|
||||
id: expect.any(String),
|
||||
country_code: "us",
|
||||
province_code: "ny",
|
||||
provider_id: null,
|
||||
})
|
||||
)
|
||||
})
|
||||
|
||||
it("should fail to create a country tax region without a provider", async () => {
|
||||
const {
|
||||
response: { status, data },
|
||||
} = await api
|
||||
.post(
|
||||
`/admin/tax-regions`,
|
||||
{
|
||||
country_code: "uk",
|
||||
},
|
||||
adminHeaders
|
||||
)
|
||||
.catch((err) => err)
|
||||
|
||||
expect(status).toEqual(400)
|
||||
expect(data).toEqual({
|
||||
message:
|
||||
"Invalid request: Provider is required when creating a non-province tax region.",
|
||||
type: "invalid_data",
|
||||
})
|
||||
})
|
||||
|
||||
it("should throw if tax region does not exist", async () => {
|
||||
const {
|
||||
response: { status, data },
|
||||
|
||||
@@ -6462,14 +6462,14 @@
|
||||
"errors": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"rateIsRequired": {
|
||||
"missingProvider": {
|
||||
"type": "string"
|
||||
},
|
||||
"nameIsRequired": {
|
||||
"missingCountry": {
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
"required": ["rateIsRequired", "nameIsRequired"],
|
||||
"required": ["missingProvider", "missingCountry"],
|
||||
"additionalProperties": false
|
||||
},
|
||||
"successToast": {
|
||||
|
||||
@@ -1718,8 +1718,8 @@
|
||||
"header": "Create Tax Region",
|
||||
"hint": "Create a new tax region to define tax rates for a specific country.",
|
||||
"errors": {
|
||||
"rateIsRequired": "Tax rate is required when creating a default tax rate.",
|
||||
"nameIsRequired": "Name is required when creating a default tax rate."
|
||||
"missingProvider": "Provider is required when creating a tax region.",
|
||||
"missingCountry": "Country is required when creating a tax region."
|
||||
},
|
||||
"successToast": "The tax region was successfully created."
|
||||
},
|
||||
|
||||
@@ -18,21 +18,40 @@ import { useComboboxData } from "../../../../../hooks/use-combobox-data"
|
||||
import { Combobox } from "../../../../../components/inputs/combobox"
|
||||
import { formatProvider } from "../../../../../lib/format-provider"
|
||||
import { sdk } from "../../../../../lib/client"
|
||||
import { i18n } from "../../../../../components/utilities/i18n"
|
||||
|
||||
type TaxRegionCreateFormProps = {
|
||||
parentId?: string
|
||||
}
|
||||
|
||||
const TaxRegionCreateSchema = z.object({
|
||||
name: z.string().optional(),
|
||||
code: z.string().optional(),
|
||||
rate: z.object({
|
||||
float: z.number().optional(),
|
||||
value: z.string().optional(),
|
||||
}),
|
||||
country_code: z.string().min(1),
|
||||
provider_id: z.string(),
|
||||
})
|
||||
const TaxRegionCreateSchema = z
|
||||
.object({
|
||||
name: z.string().optional(),
|
||||
code: z.string().optional(),
|
||||
rate: z.object({
|
||||
float: z.number().optional(),
|
||||
value: z.string().optional(),
|
||||
}),
|
||||
country_code: z.string(),
|
||||
provider_id: z.string(),
|
||||
})
|
||||
.superRefine(({ provider_id, country_code }, ctx) => {
|
||||
if (!provider_id) {
|
||||
ctx.addIssue({
|
||||
code: z.ZodIssueCode.custom,
|
||||
message: i18n.t("taxRegions.create.errors.missingProvider"),
|
||||
path: ["provider_id"],
|
||||
})
|
||||
}
|
||||
|
||||
if (!country_code) {
|
||||
ctx.addIssue({
|
||||
code: z.ZodIssueCode.custom,
|
||||
message: i18n.t("taxRegions.create.errors.missingCountry"),
|
||||
path: ["country_code"],
|
||||
})
|
||||
}
|
||||
})
|
||||
|
||||
export const TaxRegionCreateForm = ({ parentId }: TaxRegionCreateFormProps) => {
|
||||
const { t } = useTranslation()
|
||||
|
||||
@@ -3,6 +3,7 @@ import {
|
||||
createFindParams,
|
||||
createOperatorMap,
|
||||
createSelectParams,
|
||||
WithAdditionalData,
|
||||
} from "../../utils/validators"
|
||||
import { applyAndAndOrOperators } from "../../utils/common-validators"
|
||||
|
||||
@@ -39,8 +40,8 @@ export const AdminGetTaxRegionsParams = createFindParams({
|
||||
.merge(AdminGetTaxRegionsParamsFields)
|
||||
.merge(applyAndAndOrOperators(AdminGetTaxRegionsParamsFields))
|
||||
|
||||
export type AdminCreateTaxRegionType = z.infer<typeof AdminCreateTaxRegion>
|
||||
export const AdminCreateTaxRegion = z.object({
|
||||
export type AdminCreateTaxRegionType = z.infer<typeof CreateTaxRegion>
|
||||
export const CreateTaxRegion = z.object({
|
||||
country_code: z.string(),
|
||||
provider_id: z.string().nullish(),
|
||||
province_code: z.string().nullish(),
|
||||
@@ -57,6 +58,16 @@ export const AdminCreateTaxRegion = z.object({
|
||||
metadata: z.record(z.unknown()).nullish(),
|
||||
})
|
||||
|
||||
export const AdminCreateTaxRegion = WithAdditionalData(
|
||||
CreateTaxRegion,
|
||||
(schema) => {
|
||||
return schema.refine((data) => data.parent_id || data.provider_id, {
|
||||
path: ["provider_id"],
|
||||
message: "Provider is required when creating a non-province tax region.",
|
||||
})
|
||||
}
|
||||
)
|
||||
|
||||
export type AdminUpdateTaxRegionType = z.infer<typeof AdminUpdateTaxRegion>
|
||||
export const AdminUpdateTaxRegion = z.object({
|
||||
province_code: z.string().nullish(),
|
||||
|
||||
Reference in New Issue
Block a user