* chore(medusa): strict zod versions in workspace * feat(dashboard): add campaign create to promotion UI * wip * fix(medusa): Missing middlewares export (#7289) * fix(docblock-generator): fix how type names created from Zod objects are inferred (#7292) * feat(api-ref): show schema of a tag (#7297) * feat: Add support for sendgrid and logger notification providers (#7290) * feat: Add support for sendgrid and logger notification providers * fix: changes based on PR review * chore: add action to automatically label docs (#7284) * chore: add action to automatically label docs * removes the paths param * docs: preparations for preview (#7267) * configured base paths + added development banner * fix typelist site url * added navbar and sidebar badges * configure algolia filters * remove AI assistant * remove unused imports * change navbar text and badge * lint fixes * fix build error * add to api reference rewrites * fix build error * fix build errors in user-guide * fix feedback component * add parent title to pagination * added breadcrumbs component * remove user-guide links * resolve todos * fix details about authentication * change documentation title * lint content * chore: fix bug with form reset * chore: address reviews * chore: fix specs * chore: loads of FE fixes + BE adds * chore: add more polishes + reorg files * chore: fixes to promotions modal * chore: cleanup * chore: cleanup * chore: fix build * chore: fkix cart spec * chore: fix module tests * chore: fix moar tests * wip * chore: templates + fixes + migrate currency * chore: fix build, add validation for max_quantity * chore: allow removing campaigns * chore: fix specs * chore: scope campaigns based on currency * remove console logs * chore: add translations + update keys * chore: move over filesfrom v2 to routes * chore(dashboard): Delete old translation files (#7423) * feat(dashboard,admin-sdk,admin-shared,admin-vite-plugin): Add support for UI extensions (#7383) * intial work * update lock * add routes and fix HMR of configs * cleanup * rm imports * rm debug from plugin * address feedback * address feedback * temp skip specs --------- Co-authored-by: Adrien de Peretti <adrien.deperetti@gmail.com> Co-authored-by: Shahed Nasser <shahednasser@gmail.com> Co-authored-by: Stevche Radevski <sradevski@live.com> Co-authored-by: Oli Juhl <59018053+olivermrbl@users.noreply.github.com> Co-authored-by: Kasper Fabricius Kristensen <45367945+kasperkristensen@users.noreply.github.com>
329 lines
9.9 KiB
TypeScript
329 lines
9.9 KiB
TypeScript
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
|
|
import { IPromotionModuleService } from "@medusajs/types"
|
|
import { PromotionType } from "@medusajs/utils"
|
|
import { medusaIntegrationTestRunner } from "medusa-test-utils"
|
|
import { createAdminUser } from "../../../../helpers/create-admin-user"
|
|
|
|
jest.setTimeout(50000)
|
|
|
|
const env = { MEDUSA_FF_MEDUSA_V2: true }
|
|
const adminHeaders = {
|
|
headers: { "x-medusa-access-token": "test_token" },
|
|
}
|
|
|
|
medusaIntegrationTestRunner({
|
|
env,
|
|
testSuite: ({ dbConnection, getContainer, api }) => {
|
|
describe("POST /admin/promotions", () => {
|
|
let appContainer
|
|
let promotionModuleService: IPromotionModuleService
|
|
|
|
beforeAll(async () => {
|
|
appContainer = getContainer()
|
|
promotionModuleService = appContainer.resolve(
|
|
ModuleRegistrationName.PROMOTION
|
|
)
|
|
})
|
|
|
|
beforeEach(async () => {
|
|
await createAdminUser(dbConnection, adminHeaders, appContainer)
|
|
})
|
|
|
|
it("should throw an error if required params are not passed", async () => {
|
|
const { response } = await api
|
|
.post(
|
|
`/admin/promotions`,
|
|
{
|
|
type: PromotionType.STANDARD,
|
|
},
|
|
adminHeaders
|
|
)
|
|
.catch((e) => e)
|
|
|
|
expect(response.status).toEqual(400)
|
|
// expect(response.data.message).toEqual(
|
|
// "code must be a string, code should not be empty, application_method should not be empty"
|
|
// )
|
|
})
|
|
|
|
it("should create a standard promotion successfully", async () => {
|
|
const response = await api.post(
|
|
`/admin/promotions`,
|
|
{
|
|
code: "TEST",
|
|
type: PromotionType.STANDARD,
|
|
is_automatic: true,
|
|
campaign: {
|
|
name: "test",
|
|
campaign_identifier: "test-1",
|
|
budget: {
|
|
type: "usage",
|
|
limit: 100,
|
|
},
|
|
},
|
|
application_method: {
|
|
target_type: "items",
|
|
type: "fixed",
|
|
allocation: "each",
|
|
currency_code: "USD",
|
|
value: 100,
|
|
max_quantity: 100,
|
|
target_rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
adminHeaders
|
|
)
|
|
|
|
expect(response.status).toEqual(200)
|
|
expect(response.data.promotion).toEqual(
|
|
expect.objectContaining({
|
|
id: expect.any(String),
|
|
code: "TEST",
|
|
type: "standard",
|
|
is_automatic: true,
|
|
campaign: expect.objectContaining({
|
|
name: "test",
|
|
campaign_identifier: "test-1",
|
|
budget: expect.objectContaining({
|
|
currency_code: null,
|
|
type: "usage",
|
|
limit: 100,
|
|
}),
|
|
}),
|
|
application_method: expect.objectContaining({
|
|
value: 100,
|
|
max_quantity: 100,
|
|
type: "fixed",
|
|
target_type: "items",
|
|
allocation: "each",
|
|
target_rules: [
|
|
expect.objectContaining({
|
|
operator: "eq",
|
|
attribute: "test.test",
|
|
values: expect.arrayContaining([
|
|
expect.objectContaining({ value: "test1" }),
|
|
expect.objectContaining({ value: "test2" }),
|
|
]),
|
|
}),
|
|
],
|
|
}),
|
|
rules: [
|
|
expect.objectContaining({
|
|
operator: "eq",
|
|
attribute: "test.test",
|
|
values: expect.arrayContaining([
|
|
expect.objectContaining({ value: "test1" }),
|
|
expect.objectContaining({ value: "test2" }),
|
|
]),
|
|
}),
|
|
],
|
|
})
|
|
)
|
|
})
|
|
|
|
it("should throw an error if buy_rules params are not passed", async () => {
|
|
const { response } = await api
|
|
.post(
|
|
`/admin/promotions`,
|
|
{
|
|
code: "TEST",
|
|
type: PromotionType.BUYGET,
|
|
is_automatic: true,
|
|
application_method: {
|
|
target_type: "items",
|
|
type: "fixed",
|
|
allocation: "each",
|
|
value: 100,
|
|
max_quantity: 100,
|
|
currency_code: "USD",
|
|
target_rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
adminHeaders
|
|
)
|
|
.catch((e) => e)
|
|
|
|
expect(response.status).toEqual(400)
|
|
// expect(response.data.message).toEqual(
|
|
// "Buy rules are required for buyget promotion type"
|
|
// )
|
|
})
|
|
|
|
it("should throw an error if buy_rules params are not passed", async () => {
|
|
const { response } = await api
|
|
.post(
|
|
`/admin/promotions`,
|
|
{
|
|
code: "TEST",
|
|
type: PromotionType.BUYGET,
|
|
is_automatic: true,
|
|
application_method: {
|
|
target_type: "items",
|
|
type: "fixed",
|
|
allocation: "each",
|
|
value: 100,
|
|
max_quantity: 100,
|
|
currency_code: "USD",
|
|
buy_rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
adminHeaders
|
|
)
|
|
.catch((e) => e)
|
|
|
|
expect(response.status).toEqual(400)
|
|
// expect(response.data.message).toEqual(
|
|
// "Target rules are required for buyget promotion type"
|
|
// )
|
|
})
|
|
|
|
it("should create a buyget promotion successfully", async () => {
|
|
const response = await api.post(
|
|
`/admin/promotions`,
|
|
{
|
|
code: "TEST",
|
|
type: PromotionType.BUYGET,
|
|
is_automatic: true,
|
|
campaign: {
|
|
name: "test",
|
|
campaign_identifier: "test-1",
|
|
budget: {
|
|
type: "usage",
|
|
limit: 100,
|
|
},
|
|
},
|
|
application_method: {
|
|
target_type: "items",
|
|
type: "fixed",
|
|
allocation: "each",
|
|
value: 100,
|
|
max_quantity: 100,
|
|
apply_to_quantity: 1,
|
|
buy_rules_min_quantity: 1,
|
|
currency_code: "USD",
|
|
target_rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
buy_rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
rules: [
|
|
{
|
|
attribute: "test.test",
|
|
operator: "eq",
|
|
values: ["test1", "test2"],
|
|
},
|
|
],
|
|
},
|
|
adminHeaders
|
|
)
|
|
|
|
expect(response.status).toEqual(200)
|
|
expect(response.data.promotion).toEqual(
|
|
expect.objectContaining({
|
|
id: expect.any(String),
|
|
code: "TEST",
|
|
type: "buyget",
|
|
is_automatic: true,
|
|
campaign: expect.objectContaining({
|
|
name: "test",
|
|
campaign_identifier: "test-1",
|
|
budget: expect.objectContaining({
|
|
type: "usage",
|
|
limit: 100,
|
|
}),
|
|
}),
|
|
application_method: expect.objectContaining({
|
|
value: 100,
|
|
max_quantity: 100,
|
|
type: "fixed",
|
|
target_type: "items",
|
|
allocation: "each",
|
|
apply_to_quantity: 1,
|
|
buy_rules_min_quantity: 1,
|
|
target_rules: [
|
|
expect.objectContaining({
|
|
operator: "eq",
|
|
attribute: "test.test",
|
|
values: expect.arrayContaining([
|
|
expect.objectContaining({ value: "test1" }),
|
|
expect.objectContaining({ value: "test2" }),
|
|
]),
|
|
}),
|
|
],
|
|
buy_rules: [
|
|
expect.objectContaining({
|
|
operator: "eq",
|
|
attribute: "test.test",
|
|
values: expect.arrayContaining([
|
|
expect.objectContaining({ value: "test1" }),
|
|
expect.objectContaining({ value: "test2" }),
|
|
]),
|
|
}),
|
|
],
|
|
}),
|
|
rules: [
|
|
expect.objectContaining({
|
|
operator: "eq",
|
|
attribute: "test.test",
|
|
values: expect.arrayContaining([
|
|
expect.objectContaining({ value: "test1" }),
|
|
expect.objectContaining({ value: "test2" }),
|
|
]),
|
|
}),
|
|
],
|
|
})
|
|
)
|
|
})
|
|
})
|
|
},
|
|
})
|