feat: Cart SalesChannel link + clean-up (#6418)
This commit is contained in:
@@ -1,238 +0,0 @@
|
||||
import {
|
||||
TransactionStepsDefinition,
|
||||
WorkflowManager,
|
||||
} from "@medusajs/orchestration"
|
||||
import { CartWorkflow } from "@medusajs/types"
|
||||
|
||||
import { Workflows } from "../../definitions"
|
||||
import {
|
||||
AddressHandlers,
|
||||
CartHandlers,
|
||||
CommonHandlers,
|
||||
CustomerHandlers,
|
||||
RegionHandlers,
|
||||
SalesChannelHandlers,
|
||||
} from "../../handlers"
|
||||
import { exportWorkflow, pipe } from "@medusajs/workflows-sdk"
|
||||
|
||||
enum CreateCartActions {
|
||||
setContext = "setContext",
|
||||
attachLineItems = "attachLineItems",
|
||||
attachToSalesChannel = "attachToSalesChannel",
|
||||
findRegion = "findRegion",
|
||||
findSalesChannel = "findSalesChannel",
|
||||
createCart = "createCart",
|
||||
findOrCreateAddresses = "findOrCreateAddresses",
|
||||
findOrCreateCustomer = "findOrCreateCustomer",
|
||||
removeCart = "removeCart",
|
||||
removeAddresses = "removeAddresses",
|
||||
}
|
||||
|
||||
const workflowAlias = "cart"
|
||||
const getWorkflowInput = (alias = workflowAlias) => ({
|
||||
inputAlias: workflowAlias,
|
||||
invoke: {
|
||||
from: workflowAlias,
|
||||
alias,
|
||||
},
|
||||
})
|
||||
|
||||
const workflowSteps: TransactionStepsDefinition = {
|
||||
next: [
|
||||
{
|
||||
action: CreateCartActions.findOrCreateCustomer,
|
||||
noCompensation: true,
|
||||
},
|
||||
{
|
||||
action: CreateCartActions.findSalesChannel,
|
||||
noCompensation: true,
|
||||
},
|
||||
{
|
||||
action: CreateCartActions.setContext,
|
||||
noCompensation: true,
|
||||
},
|
||||
{
|
||||
action: CreateCartActions.findRegion,
|
||||
noCompensation: true,
|
||||
next: {
|
||||
action: CreateCartActions.findOrCreateAddresses,
|
||||
noCompensation: true,
|
||||
next: {
|
||||
action: CreateCartActions.createCart,
|
||||
next: [
|
||||
{
|
||||
action: CreateCartActions.attachLineItems,
|
||||
noCompensation: true,
|
||||
},
|
||||
{ action: CreateCartActions.attachToSalesChannel },
|
||||
],
|
||||
},
|
||||
},
|
||||
},
|
||||
],
|
||||
}
|
||||
|
||||
const handlers = new Map([
|
||||
[
|
||||
CreateCartActions.findOrCreateCustomer,
|
||||
{
|
||||
invoke: pipe(
|
||||
getWorkflowInput(
|
||||
CustomerHandlers.findOrCreateCustomer.aliases.Customer
|
||||
),
|
||||
CustomerHandlers.findOrCreateCustomer
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.findSalesChannel,
|
||||
{
|
||||
invoke: pipe(
|
||||
getWorkflowInput(
|
||||
SalesChannelHandlers.findSalesChannel.aliases.SalesChannel
|
||||
),
|
||||
SalesChannelHandlers.findSalesChannel
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.setContext,
|
||||
{
|
||||
invoke: pipe(
|
||||
getWorkflowInput(CommonHandlers.setContext.aliases.Context),
|
||||
CommonHandlers.setContext
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.findRegion,
|
||||
{
|
||||
invoke: pipe(
|
||||
getWorkflowInput(RegionHandlers.findRegion.aliases.Region),
|
||||
RegionHandlers.findRegion
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.findOrCreateAddresses,
|
||||
{
|
||||
invoke: pipe(
|
||||
{
|
||||
invoke: [
|
||||
getWorkflowInput(
|
||||
AddressHandlers.findOrCreateAddresses.aliases.Addresses
|
||||
).invoke,
|
||||
{
|
||||
from: CreateCartActions.findRegion,
|
||||
alias: AddressHandlers.findOrCreateAddresses.aliases.Region,
|
||||
},
|
||||
],
|
||||
},
|
||||
AddressHandlers.findOrCreateAddresses
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.createCart,
|
||||
{
|
||||
invoke: pipe(
|
||||
{
|
||||
invoke: [
|
||||
{
|
||||
from: CreateCartActions.findSalesChannel,
|
||||
alias: CartHandlers.createCart.aliases.SalesChannel,
|
||||
},
|
||||
{
|
||||
from: CreateCartActions.findRegion,
|
||||
alias: CartHandlers.createCart.aliases.Region,
|
||||
},
|
||||
{
|
||||
from: CreateCartActions.setContext,
|
||||
alias: CartHandlers.createCart.aliases.Context,
|
||||
},
|
||||
{
|
||||
from: CreateCartActions.findOrCreateCustomer,
|
||||
alias: CartHandlers.createCart.aliases.Customer,
|
||||
},
|
||||
{
|
||||
from: CreateCartActions.findOrCreateAddresses,
|
||||
alias: CartHandlers.createCart.aliases.Addresses,
|
||||
},
|
||||
],
|
||||
},
|
||||
CartHandlers.createCart
|
||||
),
|
||||
compensate: pipe(
|
||||
{
|
||||
invoke: [
|
||||
{
|
||||
from: CreateCartActions.createCart,
|
||||
alias: CartHandlers.removeCart.aliases.Cart,
|
||||
},
|
||||
],
|
||||
},
|
||||
CartHandlers.removeCart
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.attachLineItems,
|
||||
{
|
||||
invoke: pipe(
|
||||
{
|
||||
invoke: [
|
||||
getWorkflowInput(
|
||||
CartHandlers.attachLineItemsToCart.aliases.LineItems
|
||||
).invoke,
|
||||
{
|
||||
from: CreateCartActions.createCart,
|
||||
alias: CartHandlers.attachLineItemsToCart.aliases.Cart,
|
||||
},
|
||||
],
|
||||
},
|
||||
CartHandlers.attachLineItemsToCart
|
||||
),
|
||||
},
|
||||
],
|
||||
[
|
||||
CreateCartActions.attachToSalesChannel,
|
||||
{
|
||||
invoke: pipe(
|
||||
{
|
||||
invoke: [
|
||||
{
|
||||
from: CreateCartActions.createCart,
|
||||
alias: CartHandlers.attachCartToSalesChannel.aliases.Cart,
|
||||
},
|
||||
{
|
||||
from: CreateCartActions.findSalesChannel,
|
||||
alias: CartHandlers.attachCartToSalesChannel.aliases.SalesChannel,
|
||||
},
|
||||
],
|
||||
},
|
||||
CartHandlers.attachCartToSalesChannel
|
||||
),
|
||||
compensate: pipe(
|
||||
{
|
||||
invoke: [
|
||||
{
|
||||
from: CreateCartActions.findSalesChannel,
|
||||
alias:
|
||||
CartHandlers.detachCartFromSalesChannel.aliases.SalesChannel,
|
||||
},
|
||||
],
|
||||
},
|
||||
CartHandlers.detachCartFromSalesChannel
|
||||
),
|
||||
},
|
||||
],
|
||||
])
|
||||
|
||||
WorkflowManager.register(Workflows.CreateCart, workflowSteps, handlers)
|
||||
|
||||
type CreateCartWorkflowOutput = Record<any, any>
|
||||
|
||||
export const createCart = exportWorkflow<
|
||||
CartWorkflow.CreateCartWorkflowInputDTO,
|
||||
CreateCartWorkflowOutput
|
||||
>(Workflows.CreateCart, CreateCartActions.createCart)
|
||||
@@ -1,4 +1,3 @@
|
||||
export * from "./create-cart"
|
||||
export * from "./steps"
|
||||
export * from "./workflows"
|
||||
|
||||
|
||||
@@ -6,12 +6,12 @@ import {
|
||||
} from "@medusajs/workflows-sdk"
|
||||
import { createCartsStep, findOneOrAnyRegionStep } from "../steps"
|
||||
|
||||
type WorkflowInput = CreateCartWorkflowInputDTO
|
||||
|
||||
export const createCartWorkflowId = "create-cart"
|
||||
export const createCartWorkflow = createWorkflow(
|
||||
createCartWorkflowId,
|
||||
(input: WorkflowData<WorkflowInput>): WorkflowData<CartDTO[]> => {
|
||||
(
|
||||
input: WorkflowData<CreateCartWorkflowInputDTO>
|
||||
): WorkflowData<CartDTO[]> => {
|
||||
const region = findOneOrAnyRegionStep({
|
||||
regionId: input.region_id,
|
||||
})
|
||||
|
||||
@@ -1,43 +0,0 @@
|
||||
import { MedusaV2Flag } from "@medusajs/utils"
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
import { Modules } from "@medusajs/modules-sdk"
|
||||
|
||||
type HandlerInputData = {
|
||||
cart: {
|
||||
id: string
|
||||
}
|
||||
sales_channel: {
|
||||
sales_channel_id: string
|
||||
}
|
||||
}
|
||||
|
||||
enum Aliases {
|
||||
Cart = "cart",
|
||||
SalesChannel = "sales_channel",
|
||||
}
|
||||
|
||||
export async function attachCartToSalesChannel({
|
||||
container,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<void> {
|
||||
const featureFlagRouter = container.resolve("featureFlagRouter")
|
||||
const remoteLink = container.resolve("remoteLink")
|
||||
|
||||
if (!featureFlagRouter.isFeatureEnabled(MedusaV2Flag.key)) {
|
||||
return
|
||||
}
|
||||
|
||||
const cart = data[Aliases.Cart]
|
||||
const salesChannel = data[Aliases.SalesChannel]
|
||||
|
||||
await remoteLink.create({
|
||||
[Modules.CART]: {
|
||||
cart_id: cart.id,
|
||||
},
|
||||
[Modules.SALES_CHANNEL]: {
|
||||
sales_channel_id: salesChannel.sales_channel_id,
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
attachCartToSalesChannel.aliases = Aliases
|
||||
@@ -1,57 +0,0 @@
|
||||
import { CartWorkflow } from "@medusajs/types"
|
||||
import { SalesChannelFeatureFlag } from "@medusajs/utils"
|
||||
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
|
||||
type HandlerInputData = {
|
||||
line_items: {
|
||||
items?: CartWorkflow.CreateLineItemInputDTO[]
|
||||
}
|
||||
cart: {
|
||||
id: string
|
||||
customer_id: string
|
||||
region_id: string
|
||||
}
|
||||
}
|
||||
|
||||
enum Aliases {
|
||||
LineItems = "line_items",
|
||||
Cart = "cart",
|
||||
}
|
||||
|
||||
export async function attachLineItemsToCart({
|
||||
container,
|
||||
context,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<void> {
|
||||
const { manager } = context
|
||||
|
||||
const featureFlagRouter = container.resolve("featureFlagRouter")
|
||||
const lineItemService = container.resolve("lineItemService")
|
||||
const cartService = container.resolve("cartService")
|
||||
|
||||
const lineItemServiceTx = lineItemService.withTransaction(manager)
|
||||
const cartServiceTx = cartService.withTransaction(manager)
|
||||
let lineItems = data[Aliases.LineItems].items
|
||||
const cart = data[Aliases.Cart]
|
||||
|
||||
if (lineItems?.length) {
|
||||
const generateInputData = lineItems.map((item) => ({
|
||||
variantId: item.variant_id,
|
||||
quantity: item.quantity,
|
||||
}))
|
||||
|
||||
lineItems = await lineItemServiceTx.generate(generateInputData, {
|
||||
region_id: cart.region_id,
|
||||
customer_id: cart.customer_id,
|
||||
})
|
||||
|
||||
await cartServiceTx.addOrUpdateLineItems(cart.id, lineItems, {
|
||||
validateSalesChannels: featureFlagRouter.isFeatureEnabled(
|
||||
SalesChannelFeatureFlag.key
|
||||
),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
attachLineItemsToCart.aliases = Aliases
|
||||
@@ -1,59 +0,0 @@
|
||||
import { AddressDTO, CustomerDTO, RegionDTO, legacy_CartDTO } from "@medusajs/types"
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
|
||||
enum Aliases {
|
||||
SalesChannel = "SalesChannel",
|
||||
Addresses = "addresses",
|
||||
Customer = "customer",
|
||||
Region = "region",
|
||||
Context = "context",
|
||||
}
|
||||
|
||||
type HandlerInputData = {
|
||||
sales_channel: {
|
||||
sales_channel_id?: string
|
||||
}
|
||||
addresses: {
|
||||
shipping_address?: AddressDTO
|
||||
shipping_address_id: string
|
||||
billing_address?: AddressDTO
|
||||
billing_address_id: string
|
||||
}
|
||||
customer: {
|
||||
customer?: CustomerDTO
|
||||
customer_id?: string
|
||||
email?: string
|
||||
}
|
||||
region: {
|
||||
region?: RegionDTO
|
||||
region_id: string
|
||||
}
|
||||
context: {
|
||||
context: Record<any, any>
|
||||
}
|
||||
}
|
||||
|
||||
type HandlerOutputData = {
|
||||
cart: legacy_CartDTO
|
||||
}
|
||||
|
||||
export async function createCart({
|
||||
container,
|
||||
context,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<HandlerOutputData> {
|
||||
const { manager } = context
|
||||
|
||||
const cartService = container.resolve("cartService")
|
||||
const cartServiceTx = cartService.withTransaction(manager)
|
||||
|
||||
return await cartServiceTx.create({
|
||||
...data[Aliases.SalesChannel],
|
||||
...data[Aliases.Addresses],
|
||||
...data[Aliases.Customer],
|
||||
...data[Aliases.Region],
|
||||
...data[Aliases.Context],
|
||||
})
|
||||
}
|
||||
|
||||
createCart.aliases = Aliases
|
||||
@@ -1,43 +0,0 @@
|
||||
import { MedusaV2Flag } from "@medusajs/utils"
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
import { Modules } from "@medusajs/modules-sdk"
|
||||
|
||||
type HandlerInputData = {
|
||||
cart: {
|
||||
id: string
|
||||
}
|
||||
sales_channel: {
|
||||
sales_channel_id: string
|
||||
}
|
||||
}
|
||||
|
||||
enum Aliases {
|
||||
Cart = "cart",
|
||||
SalesChannel = "sales_channel",
|
||||
}
|
||||
|
||||
export async function detachCartFromSalesChannel({
|
||||
container,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<void> {
|
||||
const featureFlagRouter = container.resolve("featureFlagRouter")
|
||||
const remoteLink = container.resolve("remoteLink")
|
||||
|
||||
if (!featureFlagRouter.isFeatureEnabled(MedusaV2Flag.key)) {
|
||||
return
|
||||
}
|
||||
|
||||
const cart = data[Aliases.Cart]
|
||||
const salesChannel = data[Aliases.SalesChannel]
|
||||
|
||||
await remoteLink.dismiss({
|
||||
[Modules.CART]: {
|
||||
cart_id: cart.id,
|
||||
},
|
||||
[Modules.SALES_CHANNEL]: {
|
||||
sales_channel_id: salesChannel.sales_channel_id,
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
detachCartFromSalesChannel.aliases = Aliases
|
||||
@@ -1,6 +0,0 @@
|
||||
export * from "./attach-line-items-to-cart"
|
||||
export * from "./create-cart"
|
||||
export * from "./remove-cart"
|
||||
export * from "./retrieve-cart"
|
||||
export * from "./attach-cart-to-sales-channel"
|
||||
export * from "./detach-cart-from-sales-channel"
|
||||
@@ -1,28 +0,0 @@
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
|
||||
enum Aliases {
|
||||
Cart = "cart",
|
||||
}
|
||||
|
||||
type HandlerInputData = {
|
||||
cart: {
|
||||
id: string
|
||||
}
|
||||
}
|
||||
|
||||
export async function removeCart({
|
||||
container,
|
||||
context,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<void> {
|
||||
const { manager } = context
|
||||
|
||||
const cartService = container.resolve("cartService")
|
||||
|
||||
const cartServiceTx = cartService.withTransaction(manager)
|
||||
const cart = data[Aliases.Cart]
|
||||
|
||||
await cartServiceTx.delete(cart.id)
|
||||
}
|
||||
|
||||
removeCart.aliases = Aliases
|
||||
@@ -1,40 +0,0 @@
|
||||
import { legacy_CartDTO } from "@medusajs/types"
|
||||
import { WorkflowArguments } from "@medusajs/workflows-sdk"
|
||||
|
||||
type HandlerInputData = {
|
||||
cart: {
|
||||
id: string
|
||||
}
|
||||
config: {
|
||||
retrieveConfig: {
|
||||
select: string[]
|
||||
relations: string[]
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
enum Aliases {
|
||||
Cart = "cart",
|
||||
Config = "config",
|
||||
}
|
||||
|
||||
export async function retrieveCart({
|
||||
container,
|
||||
context,
|
||||
data,
|
||||
}: WorkflowArguments<HandlerInputData>): Promise<legacy_CartDTO> {
|
||||
const { manager } = context
|
||||
|
||||
const cartService = container.resolve("cartService")
|
||||
|
||||
const cartServiceTx = cartService.withTransaction(manager)
|
||||
|
||||
const retrieved = await cartServiceTx.retrieve(
|
||||
data[Aliases.Cart].id,
|
||||
data[Aliases.Config].retrieveConfig
|
||||
)
|
||||
|
||||
return retrieved
|
||||
}
|
||||
|
||||
retrieveCart.aliases = Aliases
|
||||
@@ -1,5 +1,4 @@
|
||||
export * as AddressHandlers from "./address"
|
||||
export * as CartHandlers from "./cart"
|
||||
export * as CommonHandlers from "./common"
|
||||
export * as CustomerHandlers from "./customer"
|
||||
export * as InventoryHandlers from "./inventory"
|
||||
|
||||
@@ -1,60 +1,26 @@
|
||||
import { Modules } from "@medusajs/modules-sdk"
|
||||
import { ModuleJoinerConfig } from "@medusajs/types"
|
||||
import { LINKS } from "../links"
|
||||
|
||||
export const CartSalesChannel: ModuleJoinerConfig = {
|
||||
serviceName: LINKS.CartSalesChannel,
|
||||
isLink: true,
|
||||
databaseConfig: {
|
||||
tableName: "cart_sales_channel",
|
||||
idPrefix: "cartsc",
|
||||
},
|
||||
alias: [
|
||||
{
|
||||
name: "cart_sales_channel",
|
||||
},
|
||||
{
|
||||
name: "cart_sales_channels",
|
||||
},
|
||||
],
|
||||
primaryKeys: ["id", "cart_id", "sales_channel_id"],
|
||||
relationships: [
|
||||
{
|
||||
serviceName: Modules.CART,
|
||||
primaryKey: "id",
|
||||
foreignKey: "cart_id",
|
||||
alias: "cart",
|
||||
},
|
||||
{
|
||||
serviceName: Modules.SALES_CHANNEL,
|
||||
primaryKey: "id",
|
||||
foreignKey: "sales_channel_id",
|
||||
alias: "sales_channel",
|
||||
},
|
||||
],
|
||||
isReadOnlyLink: true,
|
||||
extends: [
|
||||
{
|
||||
serviceName: Modules.CART,
|
||||
fieldAlias: {
|
||||
sales_channel: "sales_channel_link.sales_channel",
|
||||
},
|
||||
relationship: {
|
||||
serviceName: LINKS.CartSalesChannel,
|
||||
primaryKey: "cart_id",
|
||||
foreignKey: "id",
|
||||
alias: "sales_channel_link",
|
||||
serviceName: Modules.SALES_CHANNEL,
|
||||
primaryKey: "id",
|
||||
foreignKey: "sales_channel_id",
|
||||
alias: "sales_channel",
|
||||
},
|
||||
},
|
||||
{
|
||||
serviceName: Modules.SALES_CHANNEL,
|
||||
fieldAlias: {
|
||||
carts: "cart_link.cart",
|
||||
},
|
||||
relationship: {
|
||||
serviceName: LINKS.CartSalesChannel,
|
||||
serviceName: Modules.CART,
|
||||
primaryKey: "sales_channel_id",
|
||||
foreignKey: "id",
|
||||
alias: "cart_link",
|
||||
alias: "carts",
|
||||
isList: true,
|
||||
},
|
||||
},
|
||||
|
||||
@@ -28,12 +28,6 @@ export const LINKS = {
|
||||
Modules.SALES_CHANNEL,
|
||||
"sales_channel_id"
|
||||
),
|
||||
CartSalesChannel: composeLinkName(
|
||||
Modules.CART,
|
||||
"cart_id",
|
||||
Modules.SALES_CHANNEL,
|
||||
"sales_channel_id"
|
||||
),
|
||||
OrderSalesChannel: composeLinkName(
|
||||
"orderService",
|
||||
"order_id",
|
||||
|
||||
@@ -2,18 +2,18 @@ import { updateCartsWorkflow } from "@medusajs/core-flows"
|
||||
import { UpdateCartDataDTO } from "@medusajs/types"
|
||||
import { MedusaRequest, MedusaResponse } from "../../../../types/routing"
|
||||
|
||||
import { defaultStoreCartRemoteQueryObject } from "../query-config"
|
||||
import { remoteQueryObjectFromString } from "@medusajs/utils"
|
||||
import { defaultStoreCartFields } from "../query-config"
|
||||
|
||||
export const GET = async (req: MedusaRequest, res: MedusaResponse) => {
|
||||
const remoteQuery = req.scope.resolve("remoteQuery")
|
||||
|
||||
const variables = { id: req.params.id }
|
||||
|
||||
const query = {
|
||||
cart: {
|
||||
...defaultStoreCartRemoteQueryObject,
|
||||
},
|
||||
}
|
||||
const query = remoteQueryObjectFromString({
|
||||
entryPoint: "cart",
|
||||
fields: defaultStoreCartFields,
|
||||
})
|
||||
|
||||
const [cart] = await remoteQuery(query, { cart: variables })
|
||||
|
||||
|
||||
@@ -4,7 +4,36 @@ export const defaultStoreCartFields = [
|
||||
"email",
|
||||
"created_at",
|
||||
"updated_at",
|
||||
"deleted_at",
|
||||
"items.id",
|
||||
"items.created_at",
|
||||
"items.updated_at",
|
||||
"items.title",
|
||||
"items.quantity",
|
||||
"items.unit_price",
|
||||
"shipping_address.id",
|
||||
"shipping_address.first_name",
|
||||
"shipping_address.last_name",
|
||||
"shipping_address.address_1",
|
||||
"shipping_address.address_2",
|
||||
"shipping_address.city",
|
||||
"shipping_address.postal_code",
|
||||
"shipping_address.country_code",
|
||||
"shipping_address.region_code",
|
||||
"shipping_address.phone",
|
||||
"billing_address.id",
|
||||
"billing_address.first_name",
|
||||
"billing_address.last_name",
|
||||
"billing_address.address_1",
|
||||
"billing_address.address_2",
|
||||
"billing_address.city",
|
||||
"billing_address.postal_code",
|
||||
"billing_address.country_code",
|
||||
"billing_address.region_code",
|
||||
"billing_address.phone",
|
||||
"region.id",
|
||||
"region.name",
|
||||
"region.currency_code",
|
||||
"sales_channel_id",
|
||||
]
|
||||
|
||||
export const defaultStoreCartRelations = [
|
||||
@@ -15,54 +44,18 @@ export const defaultStoreCartRelations = [
|
||||
"shipping_methods",
|
||||
]
|
||||
|
||||
export const allowedRelations = [
|
||||
"items",
|
||||
"region",
|
||||
"shipping_address",
|
||||
"billing_address",
|
||||
"shipping_methods",
|
||||
"sales_channel",
|
||||
]
|
||||
|
||||
export const retrieveTransformQueryConfig = {
|
||||
defaultFields: defaultStoreCartFields,
|
||||
defaultRelations: defaultStoreCartRelations,
|
||||
allowedRelations: defaultStoreCartRelations,
|
||||
isList: false,
|
||||
}
|
||||
|
||||
export const defaultStoreCartRemoteQueryObject = {
|
||||
fields: defaultStoreCartFields,
|
||||
items: {
|
||||
fields: [
|
||||
"id",
|
||||
"created_at",
|
||||
"updated_at",
|
||||
"deleted_at",
|
||||
"title",
|
||||
"quantity",
|
||||
"unit_price",
|
||||
],
|
||||
},
|
||||
shipping_address: {
|
||||
fields: [
|
||||
"id",
|
||||
"first_name",
|
||||
"last_name",
|
||||
"address_1",
|
||||
"address_2",
|
||||
"city",
|
||||
"postal_code",
|
||||
"country_code",
|
||||
"region_code",
|
||||
"phone",
|
||||
],
|
||||
},
|
||||
billing_address: {
|
||||
fields: [
|
||||
"id",
|
||||
"first_name",
|
||||
"last_name",
|
||||
"address_1",
|
||||
"address_2",
|
||||
"city",
|
||||
"postal_code",
|
||||
"country_code",
|
||||
"region_code",
|
||||
"phone",
|
||||
],
|
||||
},
|
||||
region: {
|
||||
fields: ["id", "name", "currency_code"],
|
||||
},
|
||||
}
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
import { createCartWorkflow } from "@medusajs/core-flows"
|
||||
import { CreateCartDTO } from "@medusajs/types"
|
||||
import { remoteQueryObjectFromString } from "@medusajs/utils"
|
||||
import { MedusaRequest, MedusaResponse } from "../../../types/routing"
|
||||
import { defaultStoreCartRemoteQueryObject } from "../carts/query-config"
|
||||
import { defaultStoreCartFields } from "../carts/query-config"
|
||||
|
||||
export const POST = async (req: MedusaRequest, res: MedusaResponse) => {
|
||||
const workflow = createCartWorkflow(req.scope)
|
||||
@@ -19,11 +20,10 @@ export const POST = async (req: MedusaRequest, res: MedusaResponse) => {
|
||||
|
||||
const variables = { id: result[0].id }
|
||||
|
||||
const query = {
|
||||
cart: {
|
||||
...defaultStoreCartRemoteQueryObject,
|
||||
},
|
||||
}
|
||||
const query = remoteQueryObjectFromString({
|
||||
entryPoint: "cart",
|
||||
fields: defaultStoreCartFields,
|
||||
})
|
||||
|
||||
const [cart] = await remoteQuery(query, { cart: variables })
|
||||
|
||||
|
||||
@@ -205,9 +205,8 @@ export default async (req, res) => {
|
||||
|
||||
return createdCart
|
||||
})
|
||||
// }
|
||||
|
||||
cart = await cartService.retrieveWithTotals(cart!.id, {
|
||||
cart = await cartService.retrieveWithTotals(cart.id, {
|
||||
select: defaultStoreCartFields,
|
||||
relations: defaultStoreCartRelations,
|
||||
})
|
||||
|
||||
@@ -1,47 +0,0 @@
|
||||
import { MigrationInterface, QueryRunner } from "typeorm"
|
||||
import { MedusaV2Flag } from "@medusajs/utils"
|
||||
|
||||
import SalesChannelFeatureFlag from "../loaders/feature-flags/sales-channels"
|
||||
|
||||
export const featureFlag = [SalesChannelFeatureFlag.key, MedusaV2Flag.key]
|
||||
|
||||
export class CartSalesChannelsLink1698160215000 implements MigrationInterface {
|
||||
public async up(queryRunner: QueryRunner): Promise<void> {
|
||||
await queryRunner.query(`
|
||||
CREATE TABLE IF NOT EXISTS "cart_sales_channel"
|
||||
(
|
||||
"id" character varying NOT NULL,
|
||||
"cart_id" character varying NOT NULL,
|
||||
"sales_channel_id" character varying NOT NULL,
|
||||
"created_at" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now(),
|
||||
"updated_at" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now(),
|
||||
"deleted_at" TIMESTAMP WITH TIME ZONE,
|
||||
CONSTRAINT "cart_sales_channel_pk" PRIMARY KEY ("cart_id", "sales_channel_id"),
|
||||
CONSTRAINT "cart_sales_channel_cart_id_unique" UNIQUE ("cart_id")
|
||||
);
|
||||
|
||||
CREATE INDEX IF NOT EXISTS "IDX_id_cart_sales_channel" ON "cart_sales_channel" ("id");
|
||||
|
||||
insert into "cart_sales_channel" (id, cart_id, sales_channel_id)
|
||||
(select 'cartsc_' || substr(md5(random()::text), 0, 27), id, sales_channel_id from "cart" WHERE sales_channel_id IS NOT NULL);
|
||||
|
||||
ALTER TABLE IF EXISTS "cart" DROP CONSTRAINT IF EXISTS "FK_a2bd3c26f42e754b9249ba78fd6";
|
||||
|
||||
ALTER TABLE IF EXISTS "store" DROP CONSTRAINT IF EXISTS "FK_61b0f48cccbb5f41c750bac7286";
|
||||
`)
|
||||
}
|
||||
|
||||
public async down(queryRunner: QueryRunner): Promise<void> {
|
||||
await queryRunner.query(`
|
||||
UPDATE "cart" SET "sales_channel_id" = "cart_sales_channel"."sales_channel_id"
|
||||
FROM "cart_sales_channel"
|
||||
WHERE "cart"."id" = "cart_sales_channel"."cart_id";
|
||||
|
||||
DROP TABLE IF EXISTS "cart_sales_channel";
|
||||
|
||||
ALTER TABLE IF EXISTS "cart" ADD CONSTRAINT "FK_a2bd3c26f42e754b9249ba78fd6" FOREIGN KEY ("sales_channel_id") REFERENCES "sales_channel"("id") ON DELETE NO ACTION ON UPDATE NO ACTION;
|
||||
|
||||
ALTER TABLE IF EXISTS "store" ADD CONSTRAINT "FK_61b0f48cccbb5f41c750bac7286" FOREIGN KEY ("default_sales_channel_id") REFERENCES "sales_channel"("id") ON DELETE NO ACTION ON UPDATE NO ACTION;
|
||||
`)
|
||||
}
|
||||
}
|
||||
@@ -1,29 +0,0 @@
|
||||
import { BeforeInsert, Column, Index, PrimaryColumn } from "typeorm"
|
||||
import { MedusaV2Flag, SalesChannelFeatureFlag } from "@medusajs/utils"
|
||||
|
||||
import { generateEntityId } from "../utils"
|
||||
import { SoftDeletableEntity } from "../interfaces"
|
||||
import { FeatureFlagEntity } from "../utils/feature-flag-decorators"
|
||||
|
||||
@FeatureFlagEntity([MedusaV2Flag.key, SalesChannelFeatureFlag.key])
|
||||
export class CartSalesChannel extends SoftDeletableEntity {
|
||||
@Column()
|
||||
id: string
|
||||
|
||||
@Index("cart_sales_channel_cart_id_unique", {
|
||||
unique: true,
|
||||
})
|
||||
@PrimaryColumn()
|
||||
cart_id: string
|
||||
|
||||
@PrimaryColumn()
|
||||
sales_channel_id: string
|
||||
|
||||
/**
|
||||
* @apiIgnore
|
||||
*/
|
||||
@BeforeInsert()
|
||||
private beforeInsert(): void {
|
||||
this.id = generateEntityId(this.id, "cartsc")
|
||||
}
|
||||
}
|
||||
@@ -1,3 +1,4 @@
|
||||
import { RemoteQueryFunction } from "@medusajs/types"
|
||||
import {
|
||||
FlagRouter,
|
||||
isDefined,
|
||||
@@ -45,6 +46,11 @@ import {
|
||||
SalesChannel,
|
||||
ShippingMethod,
|
||||
} from "../models"
|
||||
import { AddressRepository } from "../repositories/address"
|
||||
import { CartRepository } from "../repositories/cart"
|
||||
import { LineItemRepository } from "../repositories/line-item"
|
||||
import { PaymentSessionRepository } from "../repositories/payment-session"
|
||||
import { ShippingMethodRepository } from "../repositories/shipping-method"
|
||||
import {
|
||||
CartCreateProps,
|
||||
CartUpdateProps,
|
||||
@@ -59,16 +65,8 @@ import {
|
||||
TotalField,
|
||||
WithRequiredProperty,
|
||||
} from "../types/common"
|
||||
import { buildQuery, isString, setMetadata } from "../utils"
|
||||
|
||||
import { Modules, RemoteLink } from "@medusajs/modules-sdk"
|
||||
import { RemoteQueryFunction } from "@medusajs/types"
|
||||
import { AddressRepository } from "../repositories/address"
|
||||
import { CartRepository } from "../repositories/cart"
|
||||
import { LineItemRepository } from "../repositories/line-item"
|
||||
import { PaymentSessionRepository } from "../repositories/payment-session"
|
||||
import { ShippingMethodRepository } from "../repositories/shipping-method"
|
||||
import { PaymentSessionInput } from "../types/payment"
|
||||
import { buildQuery, isString, setMetadata } from "../utils"
|
||||
import { validateEmail } from "../utils/is-email"
|
||||
|
||||
type InjectedDependencies = {
|
||||
@@ -101,7 +99,6 @@ type InjectedDependencies = {
|
||||
productVariantInventoryService: ProductVariantInventoryService
|
||||
pricingService: PricingService
|
||||
remoteQuery: RemoteQueryFunction
|
||||
remoteLink: RemoteLink
|
||||
}
|
||||
|
||||
type TotalsConfig = {
|
||||
@@ -144,7 +141,6 @@ class CartService extends TransactionBaseService {
|
||||
protected readonly lineItemAdjustmentService_: LineItemAdjustmentService
|
||||
protected readonly featureFlagRouter_: FlagRouter
|
||||
protected remoteQuery_: RemoteQueryFunction
|
||||
protected remoteLink_: RemoteLink
|
||||
// eslint-disable-next-line max-len
|
||||
protected readonly productVariantInventoryService_: ProductVariantInventoryService
|
||||
protected readonly pricingService_: PricingService
|
||||
@@ -176,7 +172,6 @@ class CartService extends TransactionBaseService {
|
||||
featureFlagRouter,
|
||||
storeService,
|
||||
remoteQuery,
|
||||
remoteLink,
|
||||
productVariantInventoryService,
|
||||
pricingService,
|
||||
}: InjectedDependencies) {
|
||||
@@ -211,7 +206,6 @@ class CartService extends TransactionBaseService {
|
||||
this.productVariantInventoryService_ = productVariantInventoryService
|
||||
this.pricingService_ = pricingService
|
||||
this.remoteQuery_ = remoteQuery
|
||||
this.remoteLink_ = remoteLink
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -500,14 +494,7 @@ class CartService extends TransactionBaseService {
|
||||
data.sales_channel_id
|
||||
)
|
||||
|
||||
await this.remoteLink_.create({
|
||||
[Modules.CART]: {
|
||||
cart_id: cart.id,
|
||||
},
|
||||
[Modules.SALES_CHANNEL]: {
|
||||
sales_channel_id: salesChannel.id,
|
||||
},
|
||||
})
|
||||
cart.sales_channel_id = salesChannel.id
|
||||
}
|
||||
|
||||
await this.eventBus_
|
||||
@@ -1287,33 +1274,7 @@ class CartService extends TransactionBaseService {
|
||||
|
||||
await this.onSalesChannelChange(cart, data.sales_channel_id)
|
||||
|
||||
/**
|
||||
* TODO: remove this once update cart workflow is build
|
||||
* since this will be handled in a handler by the workflow
|
||||
*/
|
||||
if (this.featureFlagRouter_.isFeatureEnabled(MedusaV2Flag.key)) {
|
||||
if (cart.sales_channel_id) {
|
||||
await this.remoteLink_.dismiss({
|
||||
[Modules.CART]: {
|
||||
cart_id: cart.id,
|
||||
},
|
||||
[Modules.SALES_CHANNEL]: {
|
||||
sales_channel_id: cart.sales_channel_id,
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
await this.remoteLink_.create({
|
||||
[Modules.CART]: {
|
||||
cart_id: cart.id,
|
||||
},
|
||||
[Modules.SALES_CHANNEL]: {
|
||||
sales_channel_id: salesChannel.id,
|
||||
},
|
||||
})
|
||||
} else {
|
||||
cart.sales_channel_id = salesChannel.id
|
||||
}
|
||||
cart.sales_channel_id = salesChannel.id
|
||||
}
|
||||
|
||||
if (isDefined(data.discounts) && data.discounts.length) {
|
||||
|
||||
@@ -7,6 +7,7 @@ import {
|
||||
Filter,
|
||||
Index,
|
||||
ManyToOne,
|
||||
OnInit,
|
||||
OneToMany,
|
||||
OptionalProps,
|
||||
PrimaryKey,
|
||||
@@ -71,7 +72,7 @@ export default class Region {
|
||||
this.id = generateEntityId(this.id, "reg")
|
||||
}
|
||||
|
||||
@BeforeCreate()
|
||||
@OnInit()
|
||||
onInit() {
|
||||
this.id = generateEntityId(this.id, "reg")
|
||||
}
|
||||
|
||||
@@ -1,15 +1,16 @@
|
||||
import { DALUtils, generateEntityId } from "@medusajs/utils"
|
||||
|
||||
import { DAL } from "@medusajs/types"
|
||||
import {
|
||||
BeforeCreate,
|
||||
Entity,
|
||||
Filter,
|
||||
Index,
|
||||
OnInit,
|
||||
OptionalProps,
|
||||
PrimaryKey,
|
||||
Property,
|
||||
} from "@mikro-orm/core"
|
||||
import { DAL } from "@medusajs/types"
|
||||
|
||||
type SalesChannelOptionalProps = "is_disabled" | DAL.EntityDateColumns
|
||||
|
||||
@@ -54,7 +55,7 @@ export default class SalesChannel {
|
||||
this.id = generateEntityId(this.id, "sc")
|
||||
}
|
||||
|
||||
@BeforeCreate()
|
||||
@OnInit()
|
||||
onInit() {
|
||||
this.id = generateEntityId(this.id, "sc")
|
||||
}
|
||||
|
||||
@@ -9,11 +9,7 @@ import {
|
||||
SalesChannelDTO,
|
||||
UpdateSalesChannelDTO,
|
||||
} from "@medusajs/types"
|
||||
import {
|
||||
InjectTransactionManager,
|
||||
MedusaContext,
|
||||
ModulesSdkUtils,
|
||||
} from "@medusajs/utils"
|
||||
import { MedusaContext, ModulesSdkUtils } from "@medusajs/utils"
|
||||
|
||||
import { SalesChannel } from "@models"
|
||||
|
||||
@@ -61,7 +57,6 @@ export default class SalesChannelModuleService<
|
||||
sharedContext?: Context
|
||||
): Promise<SalesChannelDTO>
|
||||
|
||||
@InjectTransactionManager("baseRepository_")
|
||||
async create(
|
||||
data: CreateSalesChannelDTO | CreateSalesChannelDTO[],
|
||||
@MedusaContext() sharedContext: Context = {}
|
||||
@@ -88,7 +83,6 @@ export default class SalesChannelModuleService<
|
||||
sharedContext?: Context
|
||||
): Promise<SalesChannelDTO>
|
||||
|
||||
@InjectTransactionManager("baseRepository_")
|
||||
async update(
|
||||
data: UpdateSalesChannelDTO | UpdateSalesChannelDTO[],
|
||||
@MedusaContext() sharedContext: Context = {}
|
||||
|
||||
Reference in New Issue
Block a user