Skip to content

Commit

Permalink
feat: endpoint and testing skeleton
Browse files Browse the repository at this point in the history
  • Loading branch information
srindom committed Jan 25, 2024
1 parent 7d46526 commit c393740
Show file tree
Hide file tree
Showing 17 changed files with 668 additions and 4 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { ICustomerModuleService } from "@medusajs/types"
import path from "path"
import { startBootstrapApp } from "../../../../environment-helpers/bootstrap-app"
import { useApi } from "../../../../environment-helpers/use-api"
import { getContainer } from "../../../../environment-helpers/use-container"
import { initDb, useDb } from "../../../../environment-helpers/use-db"
import adminSeeder from "../../../../helpers/admin-seeder"

const env = { MEDUSA_FF_MEDUSA_V2: true }
const adminHeaders = {
headers: { "x-medusa-access-token": "test_token" },
}

describe("GET /admin/customer-groups", () => {
let dbConnection
let appContainer
let shutdownServer
let customerModuleService: ICustomerModuleService

beforeAll(async () => {
const cwd = path.resolve(path.join(__dirname, "..", "..", ".."))
dbConnection = await initDb({ cwd, env } as any)
shutdownServer = await startBootstrapApp({ cwd, env })
appContainer = getContainer()
customerModuleService = appContainer.resolve(
ModuleRegistrationName.CUSTOMER
)
})

afterAll(async () => {
const db = useDb()
await db.shutdown()
await shutdownServer()
})

beforeEach(async () => {
await adminSeeder(dbConnection)
})

afterEach(async () => {
const db = useDb()
await db.teardown()
})

it("should get all customer groups and its count", async () => {
await customerModuleService.createCustomerGroup({
name: "Test",
})

const api = useApi() as any
const response = await api.get(`/admin/customer-groups`, adminHeaders)

expect(response.status).toEqual(200)
expect(response.data.count).toEqual(1)
expect(response.data.groups).toEqual([
expect.objectContaining({
id: expect.any(String),
name: "Test",
}),
])
})
})
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { ICustomerModuleService } from "@medusajs/types"
import path from "path"
import { startBootstrapApp } from "../../../../environment-helpers/bootstrap-app"
import { useApi } from "../../../../environment-helpers/use-api"
import { getContainer } from "../../../../environment-helpers/use-container"
import { initDb, useDb } from "../../../../environment-helpers/use-db"
import adminSeeder from "../../../../helpers/admin-seeder"

const env = { MEDUSA_FF_MEDUSA_V2: true }
const adminHeaders = {
headers: { "x-medusa-access-token": "test_token" },
}

describe("GET /admin/customers", () => {
let dbConnection
let appContainer
let shutdownServer
let customerModuleService: ICustomerModuleService

beforeAll(async () => {
const cwd = path.resolve(path.join(__dirname, "..", "..", ".."))
dbConnection = await initDb({ cwd, env } as any)
shutdownServer = await startBootstrapApp({ cwd, env })
appContainer = getContainer()
customerModuleService = appContainer.resolve(
ModuleRegistrationName.CUSTOMER
)
})

afterAll(async () => {
const db = useDb()
await db.shutdown()
await shutdownServer()
})

beforeEach(async () => {
await adminSeeder(dbConnection)
})

afterEach(async () => {
const db = useDb()
await db.teardown()
})

it("should get all customers and its count", async () => {
await customerModuleService.create([
{
first_name: "Test",
last_name: "Test",
email: "test@me.com",
},
])

const api = useApi() as any
const response = await api.get(`/admin/customers`, adminHeaders)

expect(response.status).toEqual(200)
expect(response.data.count).toEqual(1)
expect(response.data.customers).toEqual([
expect.objectContaining({
id: expect.any(String),
first_name: "Test",
last_name: "Test",
email: "test@me.com",
}),
])
})
})
5 changes: 5 additions & 0 deletions integration-tests/plugins/medusa-config.js
Original file line number Diff line number Diff line change
Expand Up @@ -71,5 +71,10 @@ module.exports = {
resources: "shared",
resolve: "@medusajs/promotion",
},
[Modules.CUSTOMER]: {
scope: "internal",
resources: "shared",
resolve: "@medusajs/customer",
},
},
}
1 change: 1 addition & 0 deletions integration-tests/plugins/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
},
"dependencies": {
"@medusajs/cache-inmemory": "workspace:*",
"@medusajs/customer": "workspace:^",
"@medusajs/event-bus-local": "workspace:*",
"@medusajs/inventory": "workspace:^",
"@medusajs/medusa": "workspace:*",
Expand Down
2 changes: 1 addition & 1 deletion packages/customer/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@
"@models": ["./src/models"],
"@services": ["./src/services"],
"@repositories": ["./src/repositories"],
"@types": ["./src/types"],
"@types": ["./src/types"]
}
},
"include": ["src"],
Expand Down
52 changes: 52 additions & 0 deletions packages/medusa/src/api-v2/admin/customer-groups/middlewares.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
import { MedusaV2Flag } from "@medusajs/utils"

import {
isFeatureFlagEnabled,
transformBody,
transformQuery,
} from "../../../api/middlewares"
import { MiddlewareRoute } from "../../../loaders/helpers/routing/types"
import * as QueryConfig from "./query-config"
import {
AdminGetCustomerGroupsParams,
AdminGetCustomerGroupsGroupParams,
AdminPostCustomerGroupsReq,
AdminPostCustomerGroupsGroupReq,
} from "./validators"

export const adminCustomerGroupRoutesMiddlewares: MiddlewareRoute[] = [
{
matcher: "/admin/customer-groups*",
middlewares: [isFeatureFlagEnabled(MedusaV2Flag.key)],
},
{
method: ["GET"],
matcher: "/admin/customer-groups",
middlewares: [
transformQuery(
AdminGetCustomerGroupsParams,
QueryConfig.listTransformQueryConfig
),
],
},
{
method: ["GET"],
matcher: "/admin/customer-groups/:id",
middlewares: [
transformQuery(
AdminGetCustomerGroupsGroupParams,
QueryConfig.retrieveTransformQueryConfig
),
],
},
{
method: ["POST"],
matcher: "/admin/customer-groups",
middlewares: [transformBody(AdminPostCustomerGroupsReq)],
},
{
method: ["POST"],
matcher: "/admin/customer-groups/:id",
middlewares: [transformBody(AdminPostCustomerGroupsGroupReq)],
},
]
21 changes: 21 additions & 0 deletions packages/medusa/src/api-v2/admin/customer-groups/query-config.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
export const defaultAdminCustomerGroupRelations = []
export const allowedAdminCustomerGroupRelations = ["customers"]
export const defaultAdminCustomerGroupFields = [
"id",
"name",
"created_at",
"updated_at",
"deleted_at",
]

export const retrieveTransformQueryConfig = {
defaultFields: defaultAdminCustomerGroupFields,
defaultRelations: defaultAdminCustomerGroupRelations,
allowedRelations: allowedAdminCustomerGroupRelations,
isList: false,
}

export const listTransformQueryConfig = {
...retrieveTransformQueryConfig,
isList: true,
}
24 changes: 24 additions & 0 deletions packages/medusa/src/api-v2/admin/customer-groups/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { ModuleRegistrationName } from "@medusajs/modules-sdk"
import { ICustomerModuleService } from "@medusajs/types"
import { MedusaRequest, MedusaResponse } from "../../../types/routing"

export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
const customerModuleService = req.scope.resolve<ICustomerModuleService>(
ModuleRegistrationName.CUSTOMER
)

const [groups, count] =
await customerModuleService.listAndCountCustomerGroups(
req.filterableFields,
req.listConfig
)

const { offset, limit } = req.validatedQuery

res.json({
count,
groups,
offset,
limit,
})
}
114 changes: 114 additions & 0 deletions packages/medusa/src/api-v2/admin/customer-groups/validators.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
import { OperatorMap } from "@medusajs/types"
import { Type } from "class-transformer"
import {
IsNotEmpty,
IsOptional,
IsString,
ValidateNested,
} from "class-validator"
import { FindParams, extendedFindParamsMixin } from "../../../types/common"
import { OperatorMapValidator } from "../../../types/validators/operator-map"

export class AdminGetCustomerGroupsGroupParams extends FindParams {}

class FilterableCustomerPropsValidator {
@IsOptional()
@IsString({ each: true })
id?: string | string[]

@IsOptional()
@ValidateNested({ each: true })
@Type(() => OperatorMapValidator)
email?: string | string[] | OperatorMap<string>

@IsOptional()
@IsString({ each: true })
default_billing_address_id?: string | string[] | null

@IsOptional()
@IsString({ each: true })
default_shipping_address_id?: string | string[] | null

@IsOptional()
@IsString({ each: true })
company_name?: string | string[] | OperatorMap<string> | null

@IsOptional()
@IsString({ each: true })
first_name?: string | string[] | OperatorMap<string> | null

@IsOptional()
@IsString({ each: true })
last_name?: string | string[] | OperatorMap<string> | null

@IsOptional()
@IsString({ each: true })
created_by?: string | string[] | null

@IsOptional()
@ValidateNested()
@Type(() => OperatorMapValidator)
created_at?: OperatorMap<string>

@IsOptional()
@ValidateNested()
@Type(() => OperatorMapValidator)
updated_at?: OperatorMap<string>
}

export class AdminGetCustomerGroupsParams extends extendedFindParamsMixin({
limit: 100,
offset: 0,
}) {
@IsOptional()
@IsString({ each: true })
id?: string | string[]

@IsOptional()
@ValidateNested({ each: true })
@Type(() => OperatorMapValidator)
name?: string | OperatorMap<string>

@IsOptional()
@ValidateNested()
@Type(() => FilterableCustomerPropsValidator)
customers?: FilterableCustomerPropsValidator | string | string[]

@IsOptional()
@IsString({ each: true })
created_by?: string | string[] | null

@IsOptional()
@ValidateNested()
@Type(() => OperatorMapValidator)
created_at?: OperatorMap<string>

@IsOptional()
@ValidateNested()
@Type(() => OperatorMapValidator)
updated_at?: OperatorMap<string>

// Additional filters from BaseFilterable
@IsOptional()
@ValidateNested({ each: true })
@Type(() => AdminGetCustomerGroupsParams)
$and?: AdminGetCustomerGroupsParams[]

@IsOptional()
@ValidateNested({ each: true })
@Type(() => AdminGetCustomerGroupsParams)
$or?: AdminGetCustomerGroupsParams[]
}

export class AdminPostCustomerGroupsReq {
@IsNotEmpty()
@IsString()
name: string
}

export class AdminPostCustomerGroupsGroupReq {
@IsNotEmpty()
@IsString()
@IsOptional()
name?: string
}
Loading

0 comments on commit c393740

Please sign in to comment.