feat(medusa): added endpoints for rule attribute/operator/values options (#6911)
what: adds endpoints that returns attribute options, operator options and value options for a particular rule.
This commit is contained in:
5
.changeset/fifty-gifts-search.md
Normal file
5
.changeset/fifty-gifts-search.md
Normal file
@@ -0,0 +1,5 @@
|
||||
---
|
||||
"@medusajs/medusa": patch
|
||||
---
|
||||
|
||||
feat(medusa): added endpoints for rule attribute/operator/values options
|
||||
@@ -1,5 +1,11 @@
|
||||
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
|
||||
import { IPromotionModuleService } from "@medusajs/types"
|
||||
import {
|
||||
ICustomerModuleService,
|
||||
IProductModuleService,
|
||||
IPromotionModuleService,
|
||||
IRegionModuleService,
|
||||
ISalesChannelModuleService,
|
||||
} from "@medusajs/types"
|
||||
import { PromotionType } from "@medusajs/utils"
|
||||
import { medusaIntegrationTestRunner } from "medusa-test-utils"
|
||||
import { createAdminUser } from "../../../../helpers/create-admin-user"
|
||||
@@ -16,6 +22,11 @@ medusaIntegrationTestRunner({
|
||||
let appContainer
|
||||
let standardPromotion
|
||||
let promotionModule: IPromotionModuleService
|
||||
let regionService: IRegionModuleService
|
||||
let productService: IProductModuleService
|
||||
let customerService: ICustomerModuleService
|
||||
let salesChannelService: ISalesChannelModuleService
|
||||
|
||||
const promotionRule = {
|
||||
operator: "eq",
|
||||
attribute: "old_attr",
|
||||
@@ -25,6 +36,12 @@ medusaIntegrationTestRunner({
|
||||
beforeAll(async () => {
|
||||
appContainer = getContainer()
|
||||
promotionModule = appContainer.resolve(ModuleRegistrationName.PROMOTION)
|
||||
regionService = appContainer.resolve(ModuleRegistrationName.REGION)
|
||||
productService = appContainer.resolve(ModuleRegistrationName.PRODUCT)
|
||||
customerService = appContainer.resolve(ModuleRegistrationName.CUSTOMER)
|
||||
salesChannelService = appContainer.resolve(
|
||||
ModuleRegistrationName.SALES_CHANNEL
|
||||
)
|
||||
})
|
||||
|
||||
beforeEach(async () => {
|
||||
@@ -636,6 +653,302 @@ medusaIntegrationTestRunner({
|
||||
)
|
||||
})
|
||||
})
|
||||
|
||||
describe("GET /admin/promotions/rule-attribute-options/:ruleType", () => {
|
||||
it("should throw error when ruleType is invalid", async () => {
|
||||
const { response } = await api
|
||||
.get(
|
||||
`/admin/promotions/rule-attribute-options/does-not-exist`,
|
||||
adminHeaders
|
||||
)
|
||||
.catch((e) => e)
|
||||
|
||||
expect(response.status).toEqual(400)
|
||||
expect(response.data).toEqual({
|
||||
type: "invalid_data",
|
||||
message: "Invalid param rule_type (does-not-exist)",
|
||||
})
|
||||
})
|
||||
|
||||
it("return all rule attributes for a valid ruleType", async () => {
|
||||
const response = await api.get(
|
||||
`/admin/promotions/rule-attribute-options/rules`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.attributes).toEqual([
|
||||
{
|
||||
id: "currency",
|
||||
label: "Currency code",
|
||||
required: true,
|
||||
value: "currency_code",
|
||||
},
|
||||
{
|
||||
id: "customer_group",
|
||||
label: "Customer Group",
|
||||
required: false,
|
||||
value: "customer_group.id",
|
||||
},
|
||||
{
|
||||
id: "region",
|
||||
label: "Region",
|
||||
required: false,
|
||||
value: "region.id",
|
||||
},
|
||||
{
|
||||
id: "country",
|
||||
label: "Country",
|
||||
required: false,
|
||||
value: "shipping_address.country_code",
|
||||
},
|
||||
{
|
||||
id: "sales_channel",
|
||||
label: "Sales Channel",
|
||||
required: false,
|
||||
value: "sales_channel.id",
|
||||
},
|
||||
])
|
||||
})
|
||||
})
|
||||
|
||||
describe("GET /admin/promotions/rule-operator-options", () => {
|
||||
it("return all rule operators", async () => {
|
||||
const response = await api.get(
|
||||
`/admin/promotions/rule-operator-options`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.operators).toEqual([
|
||||
{
|
||||
id: "in",
|
||||
label: "In",
|
||||
value: "in",
|
||||
},
|
||||
{
|
||||
id: "eq",
|
||||
label: "Equals",
|
||||
value: "eq",
|
||||
},
|
||||
{
|
||||
id: "ne",
|
||||
label: "Not In",
|
||||
value: "ne",
|
||||
},
|
||||
])
|
||||
})
|
||||
})
|
||||
|
||||
describe("GET /admin/promotions/rule-value-options/:ruleType/:ruleAttributeId", () => {
|
||||
it("should throw error when ruleType is invalid", async () => {
|
||||
const { response } = await api
|
||||
.get(
|
||||
`/admin/promotions/rule-value-options/does-not-exist/region`,
|
||||
adminHeaders
|
||||
)
|
||||
.catch((e) => e)
|
||||
|
||||
expect(response.status).toEqual(400)
|
||||
expect(response.data).toEqual({
|
||||
type: "invalid_data",
|
||||
message: "Invalid param rule_type (does-not-exist)",
|
||||
})
|
||||
})
|
||||
|
||||
it("should throw error when ruleAttributeId is invalid", async () => {
|
||||
const { response } = await api
|
||||
.get(
|
||||
`/admin/promotions/rule-value-options/rules/does-not-exist`,
|
||||
adminHeaders
|
||||
)
|
||||
.catch((e) => e)
|
||||
|
||||
expect(response.status).toEqual(400)
|
||||
expect(response.data).toEqual({
|
||||
type: "invalid_data",
|
||||
message: "Invalid rule attribute - does-not-exist",
|
||||
})
|
||||
})
|
||||
|
||||
it("should return all values based on rule types", async () => {
|
||||
const [region1, region2] = await regionService.create([
|
||||
{ name: "North America", currency_code: "usd" },
|
||||
{ name: "Europe", currency_code: "eur" },
|
||||
])
|
||||
|
||||
let response = await api.get(
|
||||
`/admin/promotions/rule-value-options/rules/region`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values.length).toEqual(2)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{
|
||||
label: "North America",
|
||||
value: region1.id,
|
||||
},
|
||||
{
|
||||
label: "Europe",
|
||||
value: region2.id,
|
||||
},
|
||||
])
|
||||
)
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/rules/currency?limit=2`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values.length).toEqual(2)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{
|
||||
label: "United Arab Emirates Dirham",
|
||||
value: "aed",
|
||||
},
|
||||
{
|
||||
label: "Afghan Afghani",
|
||||
value: "afn",
|
||||
},
|
||||
])
|
||||
)
|
||||
|
||||
const group = await customerService.createCustomerGroup({
|
||||
name: "VIP",
|
||||
})
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/rules/customer_group`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values).toEqual([
|
||||
{
|
||||
label: "VIP",
|
||||
value: group.id,
|
||||
},
|
||||
])
|
||||
|
||||
const salesChannel = await salesChannelService.create({
|
||||
name: "Instagram",
|
||||
})
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/rules/sales_channel`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
// TODO: This is returning a default sales channel, but very flakily
|
||||
// Figure out why this happens and fix
|
||||
// expect(response.data.values.length).toEqual(1)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{ label: "Instagram", value: salesChannel.id },
|
||||
])
|
||||
)
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/rules/country?limit=2`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values.length).toEqual(2)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{ label: "Andorra", value: "ad" },
|
||||
{ label: "United Arab Emirates", value: "ae" },
|
||||
])
|
||||
)
|
||||
|
||||
const [product1, product2] = await productService.create([
|
||||
{ title: "test product 1" },
|
||||
{ title: "test product 2" },
|
||||
])
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/target-rules/product`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values.length).toEqual(2)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{ label: "test product 1", value: product1.id },
|
||||
{ label: "test product 2", value: product2.id },
|
||||
])
|
||||
)
|
||||
|
||||
const category = await productService.createCategory({
|
||||
name: "test category 1",
|
||||
parent_category_id: null,
|
||||
})
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/target-rules/product_category`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values).toEqual([
|
||||
{ label: "test category 1", value: category.id },
|
||||
])
|
||||
|
||||
const collection = await productService.createCollections({
|
||||
title: "test collection 1",
|
||||
})
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/target-rules/product_collection`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values).toEqual([
|
||||
{ label: "test collection 1", value: collection.id },
|
||||
])
|
||||
|
||||
const type = await productService.createTypes({
|
||||
value: "test type",
|
||||
})
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/target-rules/product_type`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values).toEqual([
|
||||
{ label: "test type", value: type.id },
|
||||
])
|
||||
|
||||
const [tag1, tag2] = await productService.createTags([
|
||||
{ value: "test tag 1" },
|
||||
{ value: "test tag 2" },
|
||||
])
|
||||
|
||||
response = await api.get(
|
||||
`/admin/promotions/rule-value-options/target-rules/product_tag`,
|
||||
adminHeaders
|
||||
)
|
||||
|
||||
expect(response.status).toEqual(200)
|
||||
expect(response.data.values.length).toEqual(2)
|
||||
expect(response.data.values).toEqual(
|
||||
expect.arrayContaining([
|
||||
{ label: "test tag 1", value: tag1.id },
|
||||
{ label: "test tag 2", value: tag2.id },
|
||||
])
|
||||
)
|
||||
})
|
||||
})
|
||||
})
|
||||
},
|
||||
})
|
||||
|
||||
@@ -23,10 +23,19 @@ export const joinerConfig: ModuleJoinerConfig = {
|
||||
serviceName: Modules.CUSTOMER,
|
||||
primaryKeys: ["id"],
|
||||
linkableKeys: LinkableKeys,
|
||||
alias: {
|
||||
name: ["customer", "customers"],
|
||||
args: {
|
||||
entity: Customer.name,
|
||||
alias: [
|
||||
{
|
||||
name: ["customer", "customers"],
|
||||
args: {
|
||||
entity: Customer.name,
|
||||
},
|
||||
},
|
||||
},
|
||||
{
|
||||
name: ["customer_group", "customer_groups"],
|
||||
args: {
|
||||
entity: CustomerGroup.name,
|
||||
methodSuffix: "CustomerGroups",
|
||||
},
|
||||
},
|
||||
],
|
||||
}
|
||||
|
||||
@@ -4,6 +4,7 @@ import { transformBody, transformQuery } from "../../../api/middlewares"
|
||||
import {
|
||||
AdminGetPromotionsParams,
|
||||
AdminGetPromotionsPromotionParams,
|
||||
AdminGetPromotionsRuleValueParams,
|
||||
AdminPostPromotionsPromotionReq,
|
||||
AdminPostPromotionsPromotionRulesBatchAddReq,
|
||||
AdminPostPromotionsPromotionRulesBatchRemoveReq,
|
||||
@@ -92,4 +93,15 @@ export const adminPromotionRoutesMiddlewares: MiddlewareRoute[] = [
|
||||
transformBody(AdminPostPromotionsPromotionRulesBatchRemoveReq),
|
||||
],
|
||||
},
|
||||
{
|
||||
method: ["GET"],
|
||||
matcher:
|
||||
"/admin/promotions/rule-value-options/:rule_type/:rule_attribute_id",
|
||||
middlewares: [
|
||||
transformQuery(
|
||||
AdminGetPromotionsRuleValueParams,
|
||||
QueryConfig.listRuleValueTransformQueryConfig
|
||||
),
|
||||
],
|
||||
},
|
||||
]
|
||||
|
||||
@@ -51,3 +51,9 @@ export const listTransformQueryConfig = {
|
||||
...retrieveTransformQueryConfig,
|
||||
isList: true,
|
||||
}
|
||||
|
||||
export const listRuleValueTransformQueryConfig = {
|
||||
defaults: [],
|
||||
allowed: [],
|
||||
isList: true,
|
||||
}
|
||||
|
||||
@@ -0,0 +1,20 @@
|
||||
import {
|
||||
AuthenticatedMedusaRequest,
|
||||
MedusaResponse,
|
||||
} from "../../../../../types/routing"
|
||||
import { ruleAttributesMap, validateRuleType } from "../../utils"
|
||||
|
||||
export const GET = async (
|
||||
req: AuthenticatedMedusaRequest,
|
||||
res: MedusaResponse
|
||||
) => {
|
||||
const { rule_type: ruleType } = req.params
|
||||
|
||||
validateRuleType(ruleType)
|
||||
|
||||
const attributes = ruleAttributesMap[ruleType] || []
|
||||
|
||||
res.json({
|
||||
attributes,
|
||||
})
|
||||
}
|
||||
@@ -0,0 +1,32 @@
|
||||
import { RuleOperator } from "@medusajs/utils"
|
||||
import {
|
||||
AuthenticatedMedusaRequest,
|
||||
MedusaResponse,
|
||||
} from "../../../../types/routing"
|
||||
|
||||
const operators = [
|
||||
{
|
||||
id: RuleOperator.IN,
|
||||
value: RuleOperator.IN,
|
||||
label: "In",
|
||||
},
|
||||
{
|
||||
id: RuleOperator.EQ,
|
||||
value: RuleOperator.EQ,
|
||||
label: "Equals",
|
||||
},
|
||||
{
|
||||
id: RuleOperator.NE,
|
||||
value: RuleOperator.NE,
|
||||
label: "Not In",
|
||||
},
|
||||
]
|
||||
|
||||
export const GET = async (
|
||||
req: AuthenticatedMedusaRequest,
|
||||
res: MedusaResponse
|
||||
) => {
|
||||
res.json({
|
||||
operators,
|
||||
})
|
||||
}
|
||||
@@ -0,0 +1,94 @@
|
||||
import {
|
||||
ContainerRegistrationKeys,
|
||||
remoteQueryObjectFromString,
|
||||
} from "@medusajs/utils"
|
||||
import {
|
||||
AuthenticatedMedusaRequest,
|
||||
MedusaResponse,
|
||||
} from "../../../../../../types/routing"
|
||||
import { validateRuleAttribute, validateRuleType } from "../../../utils"
|
||||
|
||||
const queryConfigurations = {
|
||||
region: {
|
||||
entryPoint: "region",
|
||||
labelAttr: "name",
|
||||
valueAttr: "id",
|
||||
},
|
||||
currency: {
|
||||
entryPoint: "currency",
|
||||
labelAttr: "name",
|
||||
valueAttr: "code",
|
||||
},
|
||||
customer_group: {
|
||||
entryPoint: "customer_group",
|
||||
labelAttr: "name",
|
||||
valueAttr: "id",
|
||||
},
|
||||
sales_channel: {
|
||||
entryPoint: "sales_channel",
|
||||
labelAttr: "name",
|
||||
valueAttr: "id",
|
||||
},
|
||||
country: {
|
||||
entryPoint: "country",
|
||||
labelAttr: "display_name",
|
||||
valueAttr: "iso_2",
|
||||
},
|
||||
product: {
|
||||
entryPoint: "product",
|
||||
labelAttr: "title",
|
||||
valueAttr: "id",
|
||||
},
|
||||
product_category: {
|
||||
entryPoint: "product_category",
|
||||
labelAttr: "name",
|
||||
valueAttr: "id",
|
||||
},
|
||||
product_collection: {
|
||||
entryPoint: "product_collection",
|
||||
labelAttr: "title",
|
||||
valueAttr: "id",
|
||||
},
|
||||
product_type: {
|
||||
entryPoint: "product_type",
|
||||
labelAttr: "value",
|
||||
valueAttr: "id",
|
||||
},
|
||||
product_tag: {
|
||||
entryPoint: "product_tag",
|
||||
labelAttr: "value",
|
||||
valueAttr: "id",
|
||||
},
|
||||
}
|
||||
|
||||
export const GET = async (
|
||||
req: AuthenticatedMedusaRequest,
|
||||
res: MedusaResponse
|
||||
) => {
|
||||
const { rule_type: ruleType, rule_attribute_id: ruleAttributeId } = req.params
|
||||
const queryConfig = queryConfigurations[ruleAttributeId]
|
||||
const remoteQuery = req.scope.resolve(ContainerRegistrationKeys.REMOTE_QUERY)
|
||||
|
||||
validateRuleType(ruleType)
|
||||
validateRuleAttribute(ruleType, ruleAttributeId)
|
||||
|
||||
const { rows } = await remoteQuery(
|
||||
remoteQueryObjectFromString({
|
||||
entryPoint: queryConfig.entryPoint,
|
||||
variables: {
|
||||
filters: req.filterableFields,
|
||||
...req.remoteQueryConfig.pagination,
|
||||
},
|
||||
fields: [queryConfig.labelAttr, queryConfig.valueAttr],
|
||||
})
|
||||
)
|
||||
|
||||
const values = rows.map((r) => ({
|
||||
label: r[queryConfig.labelAttr],
|
||||
value: r[queryConfig.valueAttr],
|
||||
}))
|
||||
|
||||
res.json({
|
||||
values,
|
||||
})
|
||||
}
|
||||
@@ -0,0 +1,3 @@
|
||||
export * from "./rule-attributes-map"
|
||||
export * from "./validate-rule-attribute"
|
||||
export * from "./validate-rule-type"
|
||||
@@ -0,0 +1,91 @@
|
||||
const ruleAttributes = [
|
||||
{
|
||||
id: "currency",
|
||||
value: "currency_code",
|
||||
label: "Currency code",
|
||||
required: true,
|
||||
},
|
||||
{
|
||||
id: "customer_group",
|
||||
value: "customer_group.id",
|
||||
label: "Customer Group",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "region",
|
||||
value: "region.id",
|
||||
label: "Region",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "country",
|
||||
value: "shipping_address.country_code",
|
||||
label: "Country",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "sales_channel",
|
||||
value: "sales_channel.id",
|
||||
label: "Sales Channel",
|
||||
required: false,
|
||||
},
|
||||
]
|
||||
|
||||
const commonAttributes = [
|
||||
{
|
||||
id: "product",
|
||||
value: "items.product.id",
|
||||
label: "Product",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "product_category",
|
||||
value: "items.product.categories.id",
|
||||
label: "Product Category",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "product_collection",
|
||||
value: "items.product.collection_id",
|
||||
label: "Product Collection",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "product_type",
|
||||
value: "items.product.type_id",
|
||||
label: "Product Type",
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
id: "product_tag",
|
||||
value: "items.product.tags.id",
|
||||
label: "Product Tag",
|
||||
required: false,
|
||||
},
|
||||
]
|
||||
|
||||
const buyRuleAttributes = [
|
||||
{
|
||||
id: "buy_rules_min_quantity",
|
||||
value: "buy_rules_min_quantity",
|
||||
label: "Minimum quantity of items",
|
||||
required: true,
|
||||
},
|
||||
...commonAttributes,
|
||||
]
|
||||
|
||||
const targetRuleAttributes = [
|
||||
{
|
||||
id: "apply_to_quantity",
|
||||
value: "apply_to_quantity",
|
||||
label: "Quantity of items promotion will apply to",
|
||||
required: true,
|
||||
},
|
||||
...commonAttributes,
|
||||
]
|
||||
|
||||
export const ruleAttributesMap = {
|
||||
rules: ruleAttributes,
|
||||
"target-rules": targetRuleAttributes,
|
||||
"buy-rules": buyRuleAttributes,
|
||||
}
|
||||
@@ -0,0 +1,17 @@
|
||||
import { MedusaError } from "@medusajs/utils"
|
||||
import { ruleAttributesMap } from "./rule-attributes-map"
|
||||
|
||||
export function validateRuleAttribute(
|
||||
ruleType: string,
|
||||
ruleAttributeId: string
|
||||
) {
|
||||
const ruleAttributes = ruleAttributesMap[ruleType] || []
|
||||
const ruleAttribute = ruleAttributes.find((obj) => obj.id === ruleAttributeId)
|
||||
|
||||
if (!ruleAttribute) {
|
||||
throw new MedusaError(
|
||||
MedusaError.Types.INVALID_DATA,
|
||||
`Invalid rule attribute - ${ruleAttributeId}`
|
||||
)
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
import { MedusaError, RuleType } from "@medusajs/utils"
|
||||
|
||||
const validRuleTypes: string[] = Object.values(RuleType)
|
||||
|
||||
export function validateRuleType(ruleType: string) {
|
||||
const underscorizedRuleType = ruleType.split("-").join("_")
|
||||
|
||||
if (!validRuleTypes.includes(underscorizedRuleType)) {
|
||||
throw new MedusaError(
|
||||
MedusaError.Types.INVALID_DATA,
|
||||
`Invalid param rule_type (${ruleType})`
|
||||
)
|
||||
}
|
||||
}
|
||||
@@ -30,6 +30,18 @@ import { AdminPostCampaignsReq } from "../campaigns/validators"
|
||||
|
||||
export class AdminGetPromotionsPromotionParams extends FindParams {}
|
||||
|
||||
export class AdminGetPromotionsRuleValueParams extends extendedFindParamsMixin({
|
||||
limit: 100,
|
||||
offset: 0,
|
||||
}) {
|
||||
/**
|
||||
* Search terms to search fields.
|
||||
*/
|
||||
@IsString()
|
||||
@IsOptional()
|
||||
q?: string
|
||||
}
|
||||
|
||||
export class AdminGetPromotionsParams extends extendedFindParamsMixin({
|
||||
limit: 100,
|
||||
offset: 0,
|
||||
@@ -71,7 +83,6 @@ export class AdminPostPromotionsReq {
|
||||
@IsOptional()
|
||||
is_automatic?: boolean
|
||||
|
||||
@IsOptional()
|
||||
@IsEnum(PromotionType)
|
||||
type?: PromotionTypeValues
|
||||
|
||||
@@ -86,8 +97,8 @@ export class AdminPostPromotionsReq {
|
||||
|
||||
@IsNotEmpty()
|
||||
@ValidateNested()
|
||||
@Type(() => ApplicationMethodsPostReq)
|
||||
application_method: ApplicationMethodsPostReq
|
||||
@Type(() => AdminPostApplicationMethodsReq)
|
||||
application_method: AdminPostApplicationMethodsReq
|
||||
|
||||
@IsOptional()
|
||||
@IsArray()
|
||||
@@ -113,7 +124,7 @@ export class PromotionRule {
|
||||
values: string[]
|
||||
}
|
||||
|
||||
export class ApplicationMethodsPostReq {
|
||||
export class AdminPostApplicationMethodsReq {
|
||||
@IsOptional()
|
||||
@IsString()
|
||||
description?: string
|
||||
@@ -130,7 +141,6 @@ export class ApplicationMethodsPostReq {
|
||||
@IsEnum(ApplicationMethodType)
|
||||
type?: ApplicationMethodType
|
||||
|
||||
@IsOptional()
|
||||
@IsEnum(ApplicationMethodTargetType)
|
||||
target_type?: ApplicationMethodTargetType
|
||||
|
||||
@@ -161,7 +171,7 @@ export class ApplicationMethodsPostReq {
|
||||
buy_rules_min_quantity?: number
|
||||
}
|
||||
|
||||
export class ApplicationMethodsMethodPostReq {
|
||||
export class AdminPostApplicationMethodsMethodReq {
|
||||
@IsOptional()
|
||||
@IsString()
|
||||
description?: string
|
||||
@@ -233,8 +243,8 @@ export class AdminPostPromotionsPromotionReq {
|
||||
|
||||
@IsOptional()
|
||||
@ValidateNested()
|
||||
@Type(() => ApplicationMethodsMethodPostReq)
|
||||
application_method?: ApplicationMethodsMethodPostReq
|
||||
@Type(() => AdminPostApplicationMethodsMethodReq)
|
||||
application_method?: AdminPostApplicationMethodsMethodReq
|
||||
|
||||
@IsOptional()
|
||||
@IsArray()
|
||||
|
||||
@@ -30,7 +30,7 @@ export const joinerConfig: ModuleJoinerConfig = {
|
||||
},
|
||||
{
|
||||
name: ["country", "countries"],
|
||||
args: { entity: Country.name },
|
||||
args: { entity: Country.name, methodSuffix: "Countries" },
|
||||
},
|
||||
],
|
||||
} as ModuleJoinerConfig
|
||||
|
||||
@@ -8,7 +8,7 @@ import {
|
||||
RegionCountryDTO,
|
||||
RegionDTO,
|
||||
} from "./common"
|
||||
import { CreateRegionDTO, UpsertRegionDTO, UpdateRegionDTO } from "./mutations"
|
||||
import { CreateRegionDTO, UpdateRegionDTO, UpsertRegionDTO } from "./mutations"
|
||||
|
||||
/**
|
||||
* The main service interface for the region module.
|
||||
|
||||
Reference in New Issue
Block a user