**Why** - We have some workflow-like flows in @medusajs/medusa. These should be moved over to the workflows package. - Inventory Items <> Variant currently assume a 1-1 mapping. There should be support for a many-to-many mapping. **What** - PR introduces a feature flag for supporting many-to-many mappings for inventory and variants. - Deletes legacy transaction handler in @medusajs/medusa. - Adjusts existing createInventoryItems handler to remove dependency on variant data. **Unkowns** ~~1. Couldn't find an existing test for the CreateProduct workflow. It should be tested that this still works as expected.~~ 2. Have removed transaction managers as we should move to handling consistency through orchestration tooling. Are we ready for that?
155 lines
3.3 KiB
TypeScript
155 lines
3.3 KiB
TypeScript
import { ExtendedFindConfig, FindConfig } from "@medusajs/types"
|
|
import {
|
|
And,
|
|
FindManyOptions,
|
|
FindOperator,
|
|
FindOptionsOrder,
|
|
FindOptionsRelations,
|
|
FindOptionsSelect,
|
|
FindOptionsWhere,
|
|
ILike,
|
|
In,
|
|
IsNull,
|
|
LessThan,
|
|
LessThanOrEqual,
|
|
MoreThan,
|
|
MoreThanOrEqual,
|
|
} from "typeorm"
|
|
import { buildOrder, buildRelations, buildSelects } from "@medusajs/utils"
|
|
|
|
const operatorsMap = {
|
|
lt: (value) => LessThan(value),
|
|
gt: (value) => MoreThan(value),
|
|
lte: (value) => LessThanOrEqual(value),
|
|
gte: (value) => MoreThanOrEqual(value),
|
|
contains: (value) => ILike(`%${value}%`),
|
|
starts_with: (value) => ILike(`${value}%`),
|
|
ends_with: (value) => ILike(`%${value}`),
|
|
}
|
|
|
|
/**
|
|
* Used to build TypeORM queries.
|
|
* @param selector The selector
|
|
* @param config The config
|
|
* @return The QueryBuilderConfig
|
|
*/
|
|
export function buildQuery<TWhereKeys extends object, TEntity = unknown>(
|
|
selector: TWhereKeys,
|
|
config: FindConfig<TEntity> = {}
|
|
) {
|
|
const query: ExtendedFindConfig<TEntity> = {
|
|
where: buildWhere<TWhereKeys, TEntity>(selector),
|
|
}
|
|
|
|
if ("deleted_at" in selector || config.withDeleted) {
|
|
query.withDeleted = true
|
|
}
|
|
|
|
if ("skip" in config) {
|
|
;(query as FindManyOptions<TEntity>).skip = config.skip
|
|
}
|
|
|
|
if ("take" in config) {
|
|
;(query as FindManyOptions<TEntity>).take = config.take
|
|
}
|
|
|
|
if (config.relations) {
|
|
query.relations = buildRelations(
|
|
config.relations
|
|
) as FindOptionsRelations<TEntity>
|
|
}
|
|
|
|
if (config.select) {
|
|
query.select = buildSelects(
|
|
config.select as string[]
|
|
) as FindOptionsSelect<TEntity>
|
|
}
|
|
|
|
if (config.order) {
|
|
query.order = buildOrder(config.order) as FindOptionsOrder<TEntity>
|
|
}
|
|
|
|
return query
|
|
}
|
|
|
|
/**
|
|
* @param constraints
|
|
*
|
|
* @example
|
|
* const q = buildWhere(
|
|
* {
|
|
* id: "1234",
|
|
* test1: ["123", "12", "1"],
|
|
* test2: Not("this"),
|
|
* date: { gt: date },
|
|
* amount: { gt: 10 },
|
|
* },
|
|
*)
|
|
*
|
|
* // Output
|
|
* {
|
|
* id: "1234",
|
|
* test1: In(["123", "12", "1"]),
|
|
* test2: Not("this"),
|
|
* date: MoreThan(date),
|
|
* amount: MoreThan(10)
|
|
* }
|
|
*/
|
|
function buildWhere<TWhereKeys extends object, TEntity>(
|
|
constraints: TWhereKeys
|
|
): FindOptionsWhere<TEntity> {
|
|
const where: FindOptionsWhere<TEntity> = {}
|
|
for (const [key, value] of Object.entries(constraints)) {
|
|
if (value === undefined) {
|
|
continue
|
|
}
|
|
|
|
if (value === null) {
|
|
where[key] = IsNull()
|
|
continue
|
|
}
|
|
|
|
if (value instanceof FindOperator) {
|
|
where[key] = value
|
|
continue
|
|
}
|
|
|
|
if (Array.isArray(value)) {
|
|
where[key] = In(value)
|
|
continue
|
|
}
|
|
|
|
if (typeof value === "object") {
|
|
Object.entries(value).forEach(([objectKey, objectValue]) => {
|
|
where[key] = where[key] || []
|
|
if (operatorsMap[objectKey]) {
|
|
where[key].push(operatorsMap[objectKey](objectValue))
|
|
} else {
|
|
if (objectValue != undefined && typeof objectValue === "object") {
|
|
where[key] = buildWhere<any, TEntity>(objectValue)
|
|
return
|
|
}
|
|
where[key] = value
|
|
}
|
|
return
|
|
})
|
|
|
|
if (!Array.isArray(where[key])) {
|
|
continue
|
|
}
|
|
|
|
if (where[key].length === 1) {
|
|
where[key] = where[key][0]
|
|
} else {
|
|
where[key] = And(...where[key])
|
|
}
|
|
|
|
continue
|
|
}
|
|
|
|
where[key] = value
|
|
}
|
|
|
|
return where
|
|
}
|