Skip to content

Commit

Permalink
feat(medusa): Middleware to add default SC on query if no SC already …
Browse files Browse the repository at this point in the history
…exist on it (#3694)
  • Loading branch information
StephixOne authored Apr 26, 2023
1 parent d97a6f3 commit 3a77e8a
Show file tree
Hide file tree
Showing 7 changed files with 107 additions and 35 deletions.
5 changes: 5 additions & 0 deletions .changeset/rotten-pans-rhyme.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@medusajs/medusa": patch
---

feat(medusa): Middleware to add default SC on store products query if no SC provided
10 changes: 2 additions & 8 deletions integration-tests/api/__tests__/admin/publishable-api-key.js
Original file line number Diff line number Diff line change
Expand Up @@ -682,7 +682,7 @@ describe("Publishable API keys", () => {
)
})

it("returns all products if PK is not passed", async () => {
it("returns default product from default sales channel if PK is not passed", async () => {
const api = useApi()

await api.post(
Expand All @@ -703,15 +703,9 @@ describe("Publishable API keys", () => {
},
})

expect(response.data.products.length).toBe(3)
expect(response.data.products.length).toBe(1)
expect(response.data.products).toEqual(
expect.arrayContaining([
expect.objectContaining({
id: product1.id,
}),
expect.objectContaining({
id: product2.id,
}),
expect.objectContaining({
id: product3.id,
}),
Expand Down
11 changes: 9 additions & 2 deletions integration-tests/api/__tests__/store/products.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ const { initDb, useDb } = require("../../../helpers/use-db")

const {
simpleProductFactory,
simpleProductCategoryFactory,
simpleSalesChannelFactory,
} = require("../../factories")

const productSeeder = require("../../helpers/store-product-seeder")
Expand Down Expand Up @@ -37,7 +37,14 @@ describe("/store/products", () => {

describe("GET /store/products", () => {
beforeEach(async () => {
await productSeeder(dbConnection)
const defaultSalesChannel = await simpleSalesChannelFactory(
dbConnection,
{
id: "sales-channel",
is_default: true,
}
)
await productSeeder(dbConnection, defaultSalesChannel)
await adminSeeder(dbConnection)
})

Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
import { simpleSalesChannelFactory } from "../../../factories"

const path = require("path")
const setupServer = require("../../../../helpers/setup-server")
const { useApi } = require("../../../../helpers/use-api")
const { initDb, useDb } = require("../../../../helpers/use-db")

const {
simpleProductCategoryFactory,
} = require("../../../factories")
const { simpleProductCategoryFactory } = require("../../../factories")

const productSeeder = require("../../../helpers/store-product-seeder")
const adminSeeder = require("../../../helpers/admin-seeder")
Expand All @@ -26,7 +26,7 @@ describe("/store/products", () => {
dbConnection = await initDb({ cwd })
medusaProcess = await setupServer({
cwd,
env: { MEDUSA_FF_PRODUCT_CATEGORIES: true }
env: { MEDUSA_FF_PRODUCT_CATEGORIES: true },
})
})

Expand All @@ -51,9 +51,15 @@ describe("/store/products", () => {
const internalCategoryWithProductId = "inactive-category-with-product-id"

beforeEach(async () => {
const manager = dbConnection.manager
const defaultSalesChannel = await simpleSalesChannelFactory(
dbConnection,
{
id: "sales-channel",
is_default: true,
}
)

await productSeeder(dbConnection)
await productSeeder(dbConnection, defaultSalesChannel)
await adminSeeder(dbConnection)

categoryWithProduct = await simpleProductCategoryFactory(dbConnection, {
Expand All @@ -76,25 +82,31 @@ describe("/store/products", () => {
}
)

inactiveCategoryWithProduct = await simpleProductCategoryFactory(dbConnection, {
id: inactiveCategoryWithProductId,
name: "inactive category with Product",
products: [{ id: testProductFilteringId2 }],
parent_category: nestedCategoryWithProduct,
is_active: false,
is_internal: false,
rank: 0,
})
inactiveCategoryWithProduct = await simpleProductCategoryFactory(
dbConnection,
{
id: inactiveCategoryWithProductId,
name: "inactive category with Product",
products: [{ id: testProductFilteringId2 }],
parent_category: nestedCategoryWithProduct,
is_active: false,
is_internal: false,
rank: 0,
}
)

internalCategoryWithProduct = await simpleProductCategoryFactory(dbConnection, {
id: inactiveCategoryWithProductId,
name: "inactive category with Product",
products: [{ id: testProductFilteringId2 }],
parent_category: nestedCategoryWithProduct,
is_active: true,
is_internal: true,
rank: 1,
})
internalCategoryWithProduct = await simpleProductCategoryFactory(
dbConnection,
{
id: inactiveCategoryWithProductId,
name: "inactive category with Product",
products: [{ id: testProductFilteringId2 }],
parent_category: nestedCategoryWithProduct,
is_active: true,
is_internal: true,
rank: 1,
}
)

nested2CategoryWithProduct = await simpleProductCategoryFactory(
dbConnection,
Expand Down
8 changes: 7 additions & 1 deletion integration-tests/api/helpers/store-product-seeder.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ const {
ShippingProfileType,
} = require("@medusajs/medusa")

module.exports = async (dataSource, data = {}) => {
module.exports = async (dataSource, defaultSalesChannel) => {
const manager = dataSource.manager

const yesterday = ((today) => new Date(today.setDate(today.getDate() - 1)))(
Expand Down Expand Up @@ -138,6 +138,7 @@ module.exports = async (dataSource, data = {}) => {
{ id: "tag1", value: "123" },
{ tag: "tag2", value: "456" },
],
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

p.images = [image]
Expand Down Expand Up @@ -270,6 +271,7 @@ module.exports = async (dataSource, data = {}) => {
{ id: "tag1", value: "123" },
{ tag: "tag2", value: "456" },
],
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

await manager.save(p1)
Expand Down Expand Up @@ -326,6 +328,7 @@ module.exports = async (dataSource, data = {}) => {
collection_id: "test-collection1",
status: "published",
tags: [{ id: "tag3", value: "123" }],
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

await manager.save(product1)
Expand All @@ -340,6 +343,7 @@ module.exports = async (dataSource, data = {}) => {
collection_id: "test-collection2",
status: "published",
tags: [{ id: "tag4", value: "1234" }],
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

await manager.save(product2)
Expand All @@ -354,6 +358,7 @@ module.exports = async (dataSource, data = {}) => {
collection_id: "test-collection1",
status: "draft",
tags: [{ id: "tag4", value: "1234" }],
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

await manager.save(product3)
Expand All @@ -367,6 +372,7 @@ module.exports = async (dataSource, data = {}) => {
description: "test-product-description",
type: { id: "test-type", value: "test-type" },
status: "published",
sales_channels: defaultSalesChannel ? [defaultSalesChannel] : [],
})

await manager.save(gift_card)
Expand Down
45 changes: 45 additions & 0 deletions packages/medusa/src/api/middlewares/with-default-sales-channel.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
import { NextFunction, Request, Response } from "express"
import SalesChannelFeatureFlag from "../../loaders/feature-flags/sales-channels"
import { SalesChannelService } from "../../services"
import { FlagRouter } from "../../utils/flag-router"

/**
* Middleware that includes the default sales channel on the request, if no sales channels present
* @param context Object of options
* @param context.attachChannelAsArray Whether to attach the default sales channel as an array or just a string
*/
export function withDefaultSalesChannel({
attachChannelAsArray,
}: {
attachChannelAsArray?: boolean
}): (req: Request, res: Response, next: NextFunction) => Promise<void> {
return async (req: Request, _, next: NextFunction) => {
const featureFlagRouter = req.scope.resolve(
"featureFlagRouter"
) as FlagRouter

if (
!featureFlagRouter.isFeatureEnabled(SalesChannelFeatureFlag.key) ||
req.query.sales_channel_id?.length ||
req.get("x-publishable-api-key")
) {
return next()
}

const salesChannelService: SalesChannelService = req.scope.resolve(
"salesChannelService"
)

try {
const defaultSalesChannel = await salesChannelService.retrieveDefault()
if (defaultSalesChannel?.id) {
req.query.sales_channel_id = attachChannelAsArray
? [defaultSalesChannel.id]
: defaultSalesChannel.id
}
} catch {
} finally {
next()
}
}
}
3 changes: 3 additions & 0 deletions packages/medusa/src/api/routes/store/products/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import { StoreGetProductsProductParams } from "./get-product"
import { extendRequestParams } from "../../../middlewares/publishable-api-key/extend-request-params"
import { validateProductSalesChannelAssociation } from "../../../middlewares/publishable-api-key/validate-product-sales-channel-association"
import { validateSalesChannelParam } from "../../../middlewares/publishable-api-key/validate-sales-channel-param"
import { withDefaultSalesChannel } from "../../../middlewares/with-default-sales-channel"

const route = Router()

Expand All @@ -26,6 +27,7 @@ export default (app, featureFlagRouter: FlagRouter) => {

route.get(
"/",
withDefaultSalesChannel({ attachChannelAsArray: true }),
transformStoreQuery(StoreGetProductsParams, {
defaultRelations: defaultStoreProductsRelations,
defaultFields: defaultStoreProductsFields,
Expand All @@ -38,6 +40,7 @@ export default (app, featureFlagRouter: FlagRouter) => {

route.get(
"/:id",
withDefaultSalesChannel({}),
transformStoreQuery(StoreGetProductsProductParams, {
defaultRelations: defaultStoreProductsRelations,
defaultFields: defaultStoreProductsFields,
Expand Down

0 comments on commit 3a77e8a

Please sign in to comment.