feat: Add decimal digits and rounding info to currencies (#6991)

This commit is contained in:
Stevche Radevski
2024-04-07 16:34:22 +02:00
committed by GitHub
parent 667c8609cc
commit 7302d078a9
9 changed files with 89 additions and 49 deletions

View File

@@ -19,10 +19,12 @@ moduleIntegrationTestRunner({
expect.objectContaining({
code: "cad",
name: "Canadian Dollar",
decimal_digits: 2,
}),
expect.objectContaining({
code: "usd",
name: "US Dollar",
decimal_digits: 2,
}),
])
)
@@ -116,7 +118,7 @@ moduleIntegrationTestRunner({
{},
{
take: 1,
select: ["code"],
select: ["code", "rounding"],
}
)
@@ -126,6 +128,8 @@ moduleIntegrationTestRunner({
expect(serialized).toEqual([
{
code: "aed",
rounding: 0,
raw_rounding: expect.any(Object),
},
])
})

View File

@@ -41,6 +41,35 @@
"primary": false,
"nullable": false,
"mappedType": "text"
},
"decimal_digits": {
"name": "decimal_digits",
"type": "int",
"unsigned": false,
"autoincrement": false,
"primary": false,
"nullable": false,
"default": "0",
"mappedType": "integer"
},
"rounding": {
"name": "rounding",
"type": "numeric",
"unsigned": false,
"autoincrement": false,
"primary": false,
"nullable": false,
"default": "0",
"mappedType": "decimal"
},
"raw_rounding": {
"name": "raw_rounding",
"type": "jsonb",
"unsigned": false,
"autoincrement": false,
"primary": false,
"nullable": false,
"mappedType": "json"
}
},
"name": "currency",

View File

@@ -2,8 +2,24 @@ import { Migration } from "@mikro-orm/migrations"
export class InitialSetup20240228133303 extends Migration {
async up(): Promise<void> {
const currencyTables = await this.execute(
"select * from information_schema.tables where table_name = 'currency' and table_schema = 'public'"
)
if (currencyTables.length > 0) {
// This is so we can still run the api tests, remove completely once that is not needed
this.addSql(
`alter table "currency" add column "decimal_digits" int not null default 0;`
)
this.addSql(
`alter table "currency" add column "rounding" numeric not null default 0;`
)
this.addSql(`alter table "currency" add column "raw_rounding" jsonb;`)
}
this.addSql(`create table if not exists "currency"
("code" text not null, "symbol" text not null, "symbol_native" text not null, "name" text not null,
"decimal_digits" int not null default 0, "rounding" numeric not null default 0, "raw_rounding" jsonb not null,
constraint "currency_pkey" primary key ("code"));`)
}
}

View File

@@ -1,3 +1,5 @@
import { BigNumberRawValue } from "@medusajs/types"
import { BigNumber, MikroOrmBigNumberProperty } from "@medusajs/utils"
import { Entity, PrimaryKey, Property } from "@mikro-orm/core"
@Entity({ tableName: "currency" })
@@ -13,6 +15,15 @@ class Currency {
@Property({ columnType: "text" })
name: string
@Property({ columnType: "int", default: 0 })
decimal_digits: number
@MikroOrmBigNumberProperty({ default: 0 })
rounding: BigNumber | number
@Property({ columnType: "jsonb" })
raw_rounding: BigNumberRawValue
}
export default Currency

View File

@@ -1,6 +1,5 @@
import { remoteQueryObjectFromString } from "@medusajs/utils"
import { MedusaRequest, MedusaResponse } from "../../../../types/routing"
import { defaultAdminCurrencyFields } from "../query-config"
export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const remoteQuery = req.scope.resolve("remoteQuery")
@@ -10,7 +9,7 @@ export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const queryObject = remoteQueryObjectFromString({
entryPoint: "currency",
variables,
fields: defaultAdminCurrencyFields,
fields: req.remoteQueryConfig.fields,
})
const [currency] = await remoteQuery(queryObject)

View File

@@ -1,11 +1,8 @@
import { transformQuery } from "../../../api/middlewares"
import { MiddlewareRoute } from "../../../loaders/helpers/routing/types"
import { authenticate } from "../../../utils/authenticate-middleware"
import { validateAndTransformQuery } from "../../utils/validate-query"
import * as QueryConfig from "./query-config"
import {
AdminGetCurrenciesCurrencyParams,
AdminGetCurrenciesParams,
} from "./validators"
import { AdminGetCurrencyParams, AdminGetCurrenciesParams } from "./validators"
export const adminCurrencyRoutesMiddlewares: MiddlewareRoute[] = [
{
@@ -17,7 +14,7 @@ export const adminCurrencyRoutesMiddlewares: MiddlewareRoute[] = [
method: ["GET"],
matcher: "/admin/currencies",
middlewares: [
transformQuery(
validateAndTransformQuery(
AdminGetCurrenciesParams,
QueryConfig.listTransformQueryConfig
),
@@ -27,8 +24,8 @@ export const adminCurrencyRoutesMiddlewares: MiddlewareRoute[] = [
method: ["GET"],
matcher: "/admin/currencies/:code",
middlewares: [
transformQuery(
AdminGetCurrenciesCurrencyParams,
validateAndTransformQuery(
AdminGetCurrencyParams,
QueryConfig.retrieveTransformQueryConfig
),
],

View File

@@ -1,20 +1,19 @@
export const defaultAdminCurrencyRelations = []
export const allowedAdminCurrencyRelations = []
export const defaultAdminCurrencyFields = [
"code",
"name",
"symbol",
"symbol_native",
"decimal_digits",
"rounding",
]
export const retrieveTransformQueryConfig = {
defaultFields: defaultAdminCurrencyFields,
defaultRelations: defaultAdminCurrencyRelations,
allowedRelations: allowedAdminCurrencyRelations,
defaults: defaultAdminCurrencyFields,
isList: false,
}
export const listTransformQueryConfig = {
...retrieveTransformQueryConfig,
defaultLimit: 50,
isList: true,
}

View File

@@ -1,6 +1,5 @@
import { remoteQueryObjectFromString } from "@medusajs/utils"
import { MedusaRequest, MedusaResponse } from "../../../types/routing"
import { defaultAdminCurrencyFields } from "./query-config"
export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const remoteQuery = req.scope.resolve("remoteQuery")
@@ -9,11 +8,9 @@ export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
entryPoint: "currency",
variables: {
filters: req.filterableFields,
order: req.listConfig.order,
skip: req.listConfig.skip,
take: req.listConfig.take,
...req.remoteQueryConfig.pagination,
},
fields: defaultAdminCurrencyFields,
fields: req.remoteQueryConfig.fields,
})
const { rows: currencies, metadata } = await remoteQuery(queryObject)

View File

@@ -1,30 +1,18 @@
import { Type } from "class-transformer"
import { IsOptional, IsString, ValidateNested } from "class-validator"
import { FindParams, extendedFindParamsMixin } from "../../../types/common"
import { createFindParams, createSelectParams } from "../../utils/validators"
import { z } from "zod"
export class AdminGetCurrenciesCurrencyParams extends FindParams {}
/**
* Parameters used to filter and configure the pagination of the retrieved currencies.
*/
export class AdminGetCurrenciesParams extends extendedFindParamsMixin({
limit: 50,
export const AdminGetCurrencyParams = createSelectParams()
export type AdminGetCurrenciesParamsType = z.infer<
typeof AdminGetCurrenciesParams
>
export const AdminGetCurrenciesParams = createFindParams({
offset: 0,
}) {
/**
* Search parameter for currencies.
*/
@IsString({ each: true })
@IsOptional()
code?: string | string[]
// Additional filters from BaseFilterable
@IsOptional()
@ValidateNested({ each: true })
@Type(() => AdminGetCurrenciesParams)
$and?: AdminGetCurrenciesParams[]
@IsOptional()
@ValidateNested({ each: true })
@Type(() => AdminGetCurrenciesParams)
$or?: AdminGetCurrenciesParams[]
}
limit: 50,
}).merge(
z.object({
code: z.union([z.string(), z.array(z.string())]).optional(),
$and: z.lazy(() => AdminGetCurrenciesParams.array()).optional(),
$or: z.lazy(() => AdminGetCurrenciesParams.array()).optional(),
})
)