** What
- Allow auto-loaded Medusa files to export a config object.
- Currently supports isDisabled to control loading.
- new instance `FeatureFlag` exported by `@medusajs/framework/utils`
- `feature-flags` is now a supported folder for medusa projects, modules, providers and plugins. They will be loaded and added to `FeatureFlag`
** Why
- Enables conditional loading of routes, migrations, jobs, subscribers, workflows, and other files based on feature flags.
```ts
// /src/feature-flags
import { FlagSettings } from "@medusajs/framework/feature-flags"
const CustomFeatureFlag: FlagSettings = {
key: "custom_feature",
default_val: false,
env_key: "FF_MY_CUSTOM_FEATURE",
description: "Enable xyz",
}
export default CustomFeatureFlag
```
```ts
// /src/modules/my-custom-module/migration/Migration20250822135845.ts
import { FeatureFlag } from "@medusajs/framework/utils"
export class Migration20250822135845 extends Migration {
override async up(){ }
override async down(){ }
}
defineFileConfig({
isDisabled: () => !FeatureFlag.isFeatureEnabled("custom_feature")
})
```
81 lines
2.2 KiB
TypeScript
81 lines
2.2 KiB
TypeScript
import { medusaIntegrationTestRunner } from "@medusajs/test-utils"
|
|
import {
|
|
IFulfillmentModuleService,
|
|
IStockLocationService,
|
|
} from "@medusajs/types"
|
|
import {
|
|
ContainerRegistrationKeys,
|
|
Modules,
|
|
remoteQueryObjectFromString,
|
|
} from "@medusajs/utils"
|
|
|
|
jest.setTimeout(50000)
|
|
|
|
const env = {}
|
|
|
|
medusaIntegrationTestRunner({
|
|
env,
|
|
testSuite: ({ getContainer }) => {
|
|
describe("FulfillmentSet and Location", () => {
|
|
let appContainer
|
|
let fulfillmentModule: IFulfillmentModuleService
|
|
let locationModule: IStockLocationService
|
|
let remoteQuery
|
|
let remoteLink
|
|
|
|
beforeAll(async () => {
|
|
appContainer = getContainer()
|
|
fulfillmentModule = appContainer.resolve(Modules.FULFILLMENT)
|
|
locationModule = appContainer.resolve(Modules.STOCK_LOCATION)
|
|
remoteQuery = appContainer.resolve(
|
|
ContainerRegistrationKeys.REMOTE_QUERY
|
|
)
|
|
remoteLink = appContainer.resolve(ContainerRegistrationKeys.REMOTE_LINK)
|
|
})
|
|
|
|
it("should query fulfillment set and location link with remote query", async () => {
|
|
const fulfillmentSet = await fulfillmentModule.createFulfillmentSets({
|
|
name: "Test fulfillment set",
|
|
type: "delivery",
|
|
})
|
|
|
|
const euWarehouse = await locationModule.createStockLocations({
|
|
name: "EU Warehouse",
|
|
})
|
|
|
|
await remoteLink.create([
|
|
{
|
|
[Modules.STOCK_LOCATION]: {
|
|
stock_location_id: euWarehouse.id,
|
|
},
|
|
[Modules.FULFILLMENT]: {
|
|
fulfillment_set_id: fulfillmentSet.id,
|
|
},
|
|
},
|
|
])
|
|
|
|
const linkQuery = remoteQueryObjectFromString({
|
|
entryPoint: "stock_locations",
|
|
fields: ["id", "fulfillment_sets.id"],
|
|
})
|
|
|
|
const link = await remoteQuery(linkQuery)
|
|
|
|
expect(link).toHaveLength(1)
|
|
expect(link).toEqual(
|
|
expect.arrayContaining([
|
|
expect.objectContaining({
|
|
id: euWarehouse.id,
|
|
fulfillment_sets: expect.arrayContaining([
|
|
expect.objectContaining({
|
|
id: fulfillmentSet.id,
|
|
}),
|
|
]),
|
|
}),
|
|
])
|
|
)
|
|
})
|
|
})
|
|
},
|
|
})
|