@open-mercato/core 0.4.2-canary-ed15f2e753 → 0.4.2-canary-f075c3eb92
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/generated/entities.ids.generated.js +0 -1
- package/dist/generated/entities.ids.generated.js.map +2 -2
- package/dist/generated/entity-fields-registry.js +0 -2
- package/dist/generated/entity-fields-registry.js.map +2 -2
- package/dist/modules/api_keys/setup.js +11 -0
- package/dist/modules/api_keys/setup.js.map +7 -0
- package/dist/modules/attachments/components/AttachmentLibrary.js +1 -1
- package/dist/modules/attachments/components/AttachmentLibrary.js.map +2 -2
- package/dist/modules/attachments/lib/assignmentDetails.js +31 -17
- package/dist/modules/attachments/lib/assignmentDetails.js.map +2 -2
- package/dist/modules/attachments/lib/partitions.js +3 -3
- package/dist/modules/attachments/lib/partitions.js.map +2 -2
- package/dist/modules/attachments/setup.js +11 -0
- package/dist/modules/attachments/setup.js.map +7 -0
- package/dist/modules/audit_logs/setup.js +12 -0
- package/dist/modules/audit_logs/setup.js.map +7 -0
- package/dist/modules/auth/lib/setup-app.js +29 -159
- package/dist/modules/auth/lib/setup-app.js.map +2 -2
- package/dist/modules/auth/setup.js +11 -0
- package/dist/modules/auth/setup.js.map +7 -0
- package/dist/modules/business_rules/data/validators.js +0 -34
- package/dist/modules/business_rules/data/validators.js.map +2 -2
- package/dist/modules/business_rules/index.js +1 -21
- package/dist/modules/business_rules/index.js.map +2 -2
- package/dist/modules/business_rules/lib/rule-engine.js +1 -182
- package/dist/modules/business_rules/lib/rule-engine.js.map +2 -2
- package/dist/modules/business_rules/setup.js +11 -0
- package/dist/modules/business_rules/setup.js.map +7 -0
- package/dist/modules/catalog/setup.js +22 -0
- package/dist/modules/catalog/setup.js.map +7 -0
- package/dist/modules/configs/lib/upgrade-actions.js +65 -15
- package/dist/modules/configs/lib/upgrade-actions.js.map +2 -2
- package/dist/modules/configs/setup.js +16 -0
- package/dist/modules/configs/setup.js.map +7 -0
- package/dist/modules/currencies/setup.js +16 -0
- package/dist/modules/currencies/setup.js.map +7 -0
- package/dist/modules/customers/setup.js +36 -0
- package/dist/modules/customers/setup.js.map +7 -0
- package/dist/modules/dashboards/setup.js +12 -0
- package/dist/modules/dashboards/setup.js.map +7 -0
- package/dist/modules/dictionaries/setup.js +12 -0
- package/dist/modules/dictionaries/setup.js.map +7 -0
- package/dist/modules/directory/setup.js +12 -0
- package/dist/modules/directory/setup.js.map +7 -0
- package/dist/modules/entities/setup.js +11 -0
- package/dist/modules/entities/setup.js.map +7 -0
- package/dist/modules/feature_toggles/setup.js +11 -0
- package/dist/modules/feature_toggles/setup.js.map +7 -0
- package/dist/modules/perspectives/setup.js +12 -0
- package/dist/modules/perspectives/setup.js.map +7 -0
- package/dist/modules/planner/setup.js +21 -0
- package/dist/modules/planner/setup.js.map +7 -0
- package/dist/modules/query_index/setup.js +11 -0
- package/dist/modules/query_index/setup.js.map +7 -0
- package/dist/modules/resources/setup.js +21 -0
- package/dist/modules/resources/setup.js.map +7 -0
- package/dist/modules/sales/acl.js +0 -1
- package/dist/modules/sales/acl.js.map +2 -2
- package/dist/modules/sales/backend/sales/documents/[id]/page.js +0 -12
- package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
- package/dist/modules/sales/commands/documents.js +0 -62
- package/dist/modules/sales/commands/documents.js.map +2 -2
- package/dist/modules/sales/lib/dictionaries.js +0 -3
- package/dist/modules/sales/lib/dictionaries.js.map +2 -2
- package/dist/modules/sales/setup.js +99 -0
- package/dist/modules/sales/setup.js.map +7 -0
- package/dist/modules/staff/setup.js +27 -0
- package/dist/modules/staff/setup.js.map +7 -0
- package/dist/modules/workflows/acl.js +0 -2
- package/dist/modules/workflows/acl.js.map +2 -2
- package/dist/modules/workflows/api/instances/route.js +6 -18
- package/dist/modules/workflows/api/instances/route.js.map +2 -2
- package/dist/modules/workflows/api/tasks/route.js +1 -6
- package/dist/modules/workflows/api/tasks/route.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/[id]/page.js +1 -9
- package/dist/modules/workflows/backend/definitions/[id]/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/[id]/page.meta.js +1 -1
- package/dist/modules/workflows/backend/definitions/[id]/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/create/page.js +15 -24
- package/dist/modules/workflows/backend/definitions/create/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/create/page.meta.js +1 -1
- package/dist/modules/workflows/backend/definitions/create/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js +132 -150
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/visual-editor/page.meta.js +1 -1
- package/dist/modules/workflows/backend/definitions/visual-editor/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/events/[id]/page.js +1 -1
- package/dist/modules/workflows/backend/events/[id]/page.js.map +2 -2
- package/dist/modules/workflows/backend/events/[id]/page.meta.js +2 -2
- package/dist/modules/workflows/backend/events/[id]/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/instances/[id]/page.meta.js +2 -2
- package/dist/modules/workflows/backend/instances/[id]/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/tasks/[id]/page.js +1 -1
- package/dist/modules/workflows/backend/tasks/[id]/page.js.map +2 -2
- package/dist/modules/workflows/backend/tasks/[id]/page.meta.js +2 -2
- package/dist/modules/workflows/backend/tasks/[id]/page.meta.js.map +2 -2
- package/dist/modules/workflows/backend/tasks/page.js +6 -5
- package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
- package/dist/modules/workflows/cli.js +3 -81
- package/dist/modules/workflows/cli.js.map +3 -3
- package/dist/modules/workflows/data/entities.js +1 -64
- package/dist/modules/workflows/data/entities.js.map +2 -2
- package/dist/modules/workflows/data/validators.js +0 -115
- package/dist/modules/workflows/data/validators.js.map +2 -2
- package/dist/modules/workflows/examples/checkout-demo-definition.json +5 -1
- package/dist/modules/workflows/lib/activity-executor.js +13 -75
- package/dist/modules/workflows/lib/activity-executor.js.map +2 -2
- package/dist/modules/workflows/lib/graph-utils.js +2 -71
- package/dist/modules/workflows/lib/graph-utils.js.map +2 -2
- package/dist/modules/workflows/lib/seeds.js +7 -36
- package/dist/modules/workflows/lib/seeds.js.map +2 -2
- package/dist/modules/workflows/lib/start-validator.js +23 -33
- package/dist/modules/workflows/lib/start-validator.js.map +2 -2
- package/dist/modules/workflows/lib/transition-handler.js +45 -157
- package/dist/modules/workflows/lib/transition-handler.js.map +3 -3
- package/dist/modules/workflows/migrations/Migration20251207131955.js +76 -72
- package/dist/modules/workflows/migrations/Migration20251207131955.js.map +2 -2
- package/dist/modules/workflows/setup.js +16 -0
- package/dist/modules/workflows/setup.js.map +7 -0
- package/generated/entities.ids.generated.ts +0 -1
- package/generated/entity-fields-registry.ts +0 -2
- package/package.json +2 -2
- package/src/__tests__/module-decoupling.test.ts +356 -0
- package/src/modules/api_keys/setup.ts +9 -0
- package/src/modules/attachments/components/AttachmentLibrary.tsx +2 -2
- package/src/modules/attachments/lib/assignmentDetails.ts +32 -16
- package/src/modules/attachments/lib/partitions.ts +3 -3
- package/src/modules/attachments/setup.ts +9 -0
- package/src/modules/audit_logs/setup.ts +10 -0
- package/src/modules/auth/__tests__/cli-setup-acl.test.ts +30 -0
- package/src/modules/auth/lib/setup-app.ts +40 -177
- package/src/modules/auth/setup.ts +9 -0
- package/src/modules/business_rules/data/validators.ts +0 -40
- package/src/modules/business_rules/index.ts +0 -25
- package/src/modules/business_rules/lib/rule-engine.ts +1 -281
- package/src/modules/business_rules/setup.ts +9 -0
- package/src/modules/catalog/setup.ts +22 -0
- package/src/modules/configs/lib/upgrade-actions.ts +78 -17
- package/src/modules/configs/setup.ts +14 -0
- package/src/modules/currencies/setup.ts +15 -0
- package/src/modules/customers/setup.ts +36 -0
- package/src/modules/dashboards/setup.ts +10 -0
- package/src/modules/dictionaries/setup.ts +10 -0
- package/src/modules/directory/setup.ts +10 -0
- package/src/modules/entities/setup.ts +9 -0
- package/src/modules/feature_toggles/setup.ts +9 -0
- package/src/modules/perspectives/setup.ts +10 -0
- package/src/modules/planner/setup.ts +21 -0
- package/src/modules/query_index/setup.ts +9 -0
- package/src/modules/resources/setup.ts +21 -0
- package/src/modules/sales/acl.ts +0 -1
- package/src/modules/sales/backend/sales/documents/[id]/page.tsx +0 -16
- package/src/modules/sales/commands/documents.ts +1 -74
- package/src/modules/sales/lib/dictionaries.ts +0 -3
- package/src/modules/sales/setup.ts +108 -0
- package/src/modules/staff/setup.ts +27 -0
- package/src/modules/workflows/acl.ts +0 -2
- package/src/modules/workflows/api/__tests__/instances.route.test.ts +2 -5
- package/src/modules/workflows/api/instances/route.ts +7 -21
- package/src/modules/workflows/api/tasks/route.ts +1 -7
- package/src/modules/workflows/backend/definitions/[id]/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/[id]/page.tsx +0 -9
- package/src/modules/workflows/backend/definitions/create/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/create/page.tsx +0 -9
- package/src/modules/workflows/backend/definitions/visual-editor/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +3 -21
- package/src/modules/workflows/backend/events/[id]/page.meta.ts +2 -2
- package/src/modules/workflows/backend/events/[id]/page.tsx +1 -1
- package/src/modules/workflows/backend/instances/[id]/page.meta.ts +2 -2
- package/src/modules/workflows/backend/tasks/[id]/page.meta.ts +2 -2
- package/src/modules/workflows/backend/tasks/[id]/page.tsx +1 -1
- package/src/modules/workflows/backend/tasks/page.tsx +6 -5
- package/src/modules/workflows/cli.ts +0 -111
- package/src/modules/workflows/data/entities.ts +0 -124
- package/src/modules/workflows/data/validators.ts +0 -138
- package/src/modules/workflows/examples/checkout-demo-definition.json +5 -1
- package/src/modules/workflows/i18n/en.json +0 -71
- package/src/modules/workflows/lib/__tests__/activity-executor.test.ts +36 -43
- package/src/modules/workflows/lib/__tests__/transition-handler.test.ts +90 -170
- package/src/modules/workflows/lib/activity-executor.ts +16 -129
- package/src/modules/workflows/lib/graph-utils.ts +2 -117
- package/src/modules/workflows/lib/seeds.ts +12 -50
- package/src/modules/workflows/lib/start-validator.ts +28 -38
- package/src/modules/workflows/lib/transition-handler.ts +55 -208
- package/src/modules/workflows/migrations/Migration20251207131955.ts +77 -143
- package/src/modules/workflows/setup.ts +15 -0
- package/dist/generated/entities/workflow_event_trigger/index.js +0 -33
- package/dist/generated/entities/workflow_event_trigger/index.js.map +0 -7
- package/dist/modules/auth/events.js +0 -30
- package/dist/modules/auth/events.js.map +0 -7
- package/dist/modules/business_rules/api/execute/[ruleId]/route.js +0 -145
- package/dist/modules/business_rules/api/execute/[ruleId]/route.js.map +0 -7
- package/dist/modules/catalog/events.js +0 -34
- package/dist/modules/catalog/events.js.map +0 -7
- package/dist/modules/customers/events.js +0 -49
- package/dist/modules/customers/events.js.map +0 -7
- package/dist/modules/directory/events.js +0 -23
- package/dist/modules/directory/events.js.map +0 -7
- package/dist/modules/sales/events.js +0 -63
- package/dist/modules/sales/events.js.map +0 -7
- package/dist/modules/sales/lib/frontend/documentDataEvents.js +0 -25
- package/dist/modules/sales/lib/frontend/documentDataEvents.js.map +0 -7
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js +0 -481
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +0 -7
- package/dist/modules/workflows/components/EventTriggersEditor.js +0 -553
- package/dist/modules/workflows/components/EventTriggersEditor.js.map +0 -7
- package/dist/modules/workflows/events.js +0 -38
- package/dist/modules/workflows/events.js.map +0 -7
- package/dist/modules/workflows/examples/order-approval-definition.json +0 -257
- package/dist/modules/workflows/examples/order-approval-guard-rules.json +0 -32
- package/dist/modules/workflows/lib/event-trigger-service.js +0 -308
- package/dist/modules/workflows/lib/event-trigger-service.js.map +0 -7
- package/dist/modules/workflows/migrations/Migration20260123143500.js +0 -36
- package/dist/modules/workflows/migrations/Migration20260123143500.js.map +0 -7
- package/dist/modules/workflows/subscribers/event-trigger.js +0 -78
- package/dist/modules/workflows/subscribers/event-trigger.js.map +0 -7
- package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js +0 -323
- package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js.map +0 -7
- package/dist/modules/workflows/widgets/injection/order-approval/widget.js +0 -17
- package/dist/modules/workflows/widgets/injection/order-approval/widget.js.map +0 -7
- package/dist/modules/workflows/widgets/injection-table.js +0 -19
- package/dist/modules/workflows/widgets/injection-table.js.map +0 -7
- package/generated/entities/workflow_event_trigger/index.ts +0 -15
- package/src/modules/auth/events.ts +0 -39
- package/src/modules/business_rules/api/execute/[ruleId]/route.ts +0 -163
- package/src/modules/catalog/events.ts +0 -45
- package/src/modules/customers/events.ts +0 -63
- package/src/modules/directory/events.ts +0 -31
- package/src/modules/sales/events.ts +0 -82
- package/src/modules/sales/lib/frontend/documentDataEvents.ts +0 -28
- package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +0 -581
- package/src/modules/workflows/components/EventTriggersEditor.tsx +0 -664
- package/src/modules/workflows/events.ts +0 -49
- package/src/modules/workflows/examples/order-approval-definition.json +0 -257
- package/src/modules/workflows/examples/order-approval-guard-rules.json +0 -32
- package/src/modules/workflows/lib/event-trigger-service.ts +0 -557
- package/src/modules/workflows/migrations/Migration20260123143500.ts +0 -38
- package/src/modules/workflows/subscribers/event-trigger.ts +0 -109
- package/src/modules/workflows/widgets/injection/order-approval/widget.client.tsx +0 -446
- package/src/modules/workflows/widgets/injection/order-approval/widget.ts +0 -16
- package/src/modules/workflows/widgets/injection-table.ts +0 -21
|
@@ -1,323 +0,0 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
import { jsx, jsxs } from "react/jsx-runtime";
|
|
3
|
-
import * as React from "react";
|
|
4
|
-
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
|
5
|
-
import { apiCall } from "@open-mercato/ui/backend/utils/apiCall";
|
|
6
|
-
import { useT } from "@open-mercato/shared/lib/i18n/context";
|
|
7
|
-
import { Button } from "@open-mercato/ui/primitives/button";
|
|
8
|
-
import { Spinner } from "@open-mercato/ui/primitives/spinner";
|
|
9
|
-
import { Textarea } from "@open-mercato/ui/primitives/textarea";
|
|
10
|
-
import { Badge } from "@open-mercato/ui/primitives/badge";
|
|
11
|
-
import { emitSalesDocumentDataRefresh } from "@open-mercato/core/modules/sales/lib/frontend/documentDataEvents";
|
|
12
|
-
const WORKFLOW_ID = "sales_order_approval_v1";
|
|
13
|
-
function OrderApprovalWidget({ data }) {
|
|
14
|
-
const t = useT();
|
|
15
|
-
const queryClient = useQueryClient();
|
|
16
|
-
const orderId = data?.id;
|
|
17
|
-
const [decision, setDecision] = React.useState("");
|
|
18
|
-
const [comments, setComments] = React.useState("");
|
|
19
|
-
const [error, setError] = React.useState(null);
|
|
20
|
-
const [isWaitingForProcessing, setIsWaitingForProcessing] = React.useState(false);
|
|
21
|
-
const { data: orderData } = useQuery({
|
|
22
|
-
queryKey: ["order-status", orderId],
|
|
23
|
-
queryFn: async () => {
|
|
24
|
-
if (!orderId) return null;
|
|
25
|
-
const result = await apiCall(
|
|
26
|
-
`/api/sales/orders?id=${orderId}&pageSize=1`
|
|
27
|
-
);
|
|
28
|
-
return result.ok && result.result?.items?.[0] ? result.result.items[0] : null;
|
|
29
|
-
},
|
|
30
|
-
enabled: Boolean(orderId),
|
|
31
|
-
staleTime: 5e3,
|
|
32
|
-
// Poll when waiting for processing to detect status change
|
|
33
|
-
refetchInterval: isWaitingForProcessing ? 2e3 : false
|
|
34
|
-
});
|
|
35
|
-
const currentOrderStatus = orderData?.status || data?.status;
|
|
36
|
-
const { data: instancesData, isLoading: instancesLoading } = useQuery({
|
|
37
|
-
queryKey: ["workflow-instances", orderId],
|
|
38
|
-
queryFn: async () => {
|
|
39
|
-
if (!orderId) return { data: [] };
|
|
40
|
-
const result = await apiCall(
|
|
41
|
-
`/api/workflows/instances?entityId=${orderId}&status=RUNNING,PAUSED,WAITING_FOR_ACTIVITIES`
|
|
42
|
-
);
|
|
43
|
-
return result.ok ? result.result : { data: [] };
|
|
44
|
-
},
|
|
45
|
-
enabled: Boolean(orderId),
|
|
46
|
-
staleTime: 5e3,
|
|
47
|
-
// Poll every 2 seconds when waiting for processing
|
|
48
|
-
refetchInterval: isWaitingForProcessing ? 2e3 : false
|
|
49
|
-
});
|
|
50
|
-
const activeInstance = instancesData?.data?.find(
|
|
51
|
-
(inst) => inst.workflowId === WORKFLOW_ID && !["COMPLETED", "FAILED", "CANCELLED"].includes(inst.status)
|
|
52
|
-
);
|
|
53
|
-
const { data: tasksData, isLoading: tasksLoading } = useQuery({
|
|
54
|
-
queryKey: ["workflow-tasks", activeInstance?.id],
|
|
55
|
-
queryFn: async () => {
|
|
56
|
-
if (!activeInstance?.id) return { data: [] };
|
|
57
|
-
const result = await apiCall(
|
|
58
|
-
`/api/workflows/tasks?workflowInstanceId=${activeInstance.id}&status=PENDING,IN_PROGRESS`
|
|
59
|
-
);
|
|
60
|
-
return result.ok ? result.result : { data: [] };
|
|
61
|
-
},
|
|
62
|
-
enabled: Boolean(activeInstance?.id),
|
|
63
|
-
staleTime: 5e3,
|
|
64
|
-
// Poll every 2 seconds when waiting for processing
|
|
65
|
-
refetchInterval: isWaitingForProcessing ? 2e3 : false
|
|
66
|
-
});
|
|
67
|
-
const pendingTask = tasksData?.data?.[0];
|
|
68
|
-
const isProcessing = activeInstance && !pendingTask && !["COMPLETED", "FAILED", "CANCELLED"].includes(activeInstance.status);
|
|
69
|
-
const prevStatusRef = React.useRef(data?.status);
|
|
70
|
-
React.useEffect(() => {
|
|
71
|
-
if (isProcessing) {
|
|
72
|
-
setIsWaitingForProcessing(true);
|
|
73
|
-
} else if (!activeInstance || activeInstance.status === "COMPLETED") {
|
|
74
|
-
setIsWaitingForProcessing(false);
|
|
75
|
-
}
|
|
76
|
-
}, [isProcessing, activeInstance]);
|
|
77
|
-
React.useEffect(() => {
|
|
78
|
-
const newStatus = orderData?.status;
|
|
79
|
-
const oldStatus = prevStatusRef.current;
|
|
80
|
-
if (newStatus && oldStatus && newStatus !== oldStatus) {
|
|
81
|
-
if (orderId) {
|
|
82
|
-
emitSalesDocumentDataRefresh({ documentId: orderId, kind: "order" });
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
prevStatusRef.current = newStatus || oldStatus;
|
|
86
|
-
}, [orderData?.status, orderId]);
|
|
87
|
-
const { data: dictionariesData } = useQuery({
|
|
88
|
-
queryKey: ["dictionaries"],
|
|
89
|
-
queryFn: async () => {
|
|
90
|
-
const result = await apiCall(
|
|
91
|
-
"/api/dictionaries"
|
|
92
|
-
);
|
|
93
|
-
return result.ok ? result.result : { items: [] };
|
|
94
|
-
},
|
|
95
|
-
staleTime: 6e4
|
|
96
|
-
});
|
|
97
|
-
const orderStatusDictionaryId = React.useMemo(() => {
|
|
98
|
-
const dict = dictionariesData?.items?.find((d) => d.key === "sales.order_status");
|
|
99
|
-
return dict?.id;
|
|
100
|
-
}, [dictionariesData]);
|
|
101
|
-
const { data: statusEntriesData } = useQuery({
|
|
102
|
-
queryKey: ["dictionary-entries", orderStatusDictionaryId],
|
|
103
|
-
queryFn: async () => {
|
|
104
|
-
if (!orderStatusDictionaryId) return { items: [] };
|
|
105
|
-
const result = await apiCall(
|
|
106
|
-
`/api/dictionaries/${orderStatusDictionaryId}/entries`
|
|
107
|
-
);
|
|
108
|
-
return result.ok ? result.result : { items: [] };
|
|
109
|
-
},
|
|
110
|
-
enabled: Boolean(orderStatusDictionaryId),
|
|
111
|
-
staleTime: 6e4
|
|
112
|
-
});
|
|
113
|
-
const findStatusId = React.useCallback((code) => {
|
|
114
|
-
const entries = statusEntriesData?.items || [];
|
|
115
|
-
const entry = entries.find(
|
|
116
|
-
(e) => e.value === code || e.label?.toLowerCase() === code.toLowerCase()
|
|
117
|
-
);
|
|
118
|
-
return entry?.id;
|
|
119
|
-
}, [statusEntriesData]);
|
|
120
|
-
const startWorkflowMutation = useMutation({
|
|
121
|
-
mutationFn: async () => {
|
|
122
|
-
const pendingApprovalStatusId = findStatusId("pending_approval");
|
|
123
|
-
const approvedStatusId = findStatusId("approved");
|
|
124
|
-
const rejectedStatusId = findStatusId("rejected");
|
|
125
|
-
if (!pendingApprovalStatusId || !approvedStatusId || !rejectedStatusId) {
|
|
126
|
-
throw new Error(t("workflows.orderApproval.missingStatuses", "Missing order status entries. Please ensure pending_approval, approved, and rejected statuses exist in the sales.order_status dictionary."));
|
|
127
|
-
}
|
|
128
|
-
const result = await apiCall("/api/workflows/instances", {
|
|
129
|
-
method: "POST",
|
|
130
|
-
body: JSON.stringify({
|
|
131
|
-
workflowId: WORKFLOW_ID,
|
|
132
|
-
initialContext: {
|
|
133
|
-
orderId,
|
|
134
|
-
pendingApprovalStatusId,
|
|
135
|
-
approvedStatusId,
|
|
136
|
-
rejectedStatusId
|
|
137
|
-
},
|
|
138
|
-
metadata: {
|
|
139
|
-
entityType: "SalesOrder",
|
|
140
|
-
entityId: orderId
|
|
141
|
-
}
|
|
142
|
-
})
|
|
143
|
-
});
|
|
144
|
-
if (!result.ok) {
|
|
145
|
-
const errorResult = result.result;
|
|
146
|
-
throw new Error(errorResult?.error || t("workflows.orderApproval.startError", "Failed to start approval workflow"));
|
|
147
|
-
}
|
|
148
|
-
return result.result;
|
|
149
|
-
},
|
|
150
|
-
onSuccess: () => {
|
|
151
|
-
setError(null);
|
|
152
|
-
queryClient.invalidateQueries({ queryKey: ["workflow-instances", orderId] });
|
|
153
|
-
},
|
|
154
|
-
onError: (err) => {
|
|
155
|
-
setError(err.message);
|
|
156
|
-
}
|
|
157
|
-
});
|
|
158
|
-
const completeTaskMutation = useMutation({
|
|
159
|
-
mutationFn: async ({ taskId, formData }) => {
|
|
160
|
-
const result = await apiCall(`/api/workflows/tasks/${taskId}/complete`, {
|
|
161
|
-
method: "POST",
|
|
162
|
-
body: JSON.stringify({ formData })
|
|
163
|
-
});
|
|
164
|
-
if (!result.ok) {
|
|
165
|
-
const errorResult = result.result;
|
|
166
|
-
throw new Error(errorResult?.error || t("workflows.orderApproval.completeError", "Failed to complete approval task"));
|
|
167
|
-
}
|
|
168
|
-
return result.result;
|
|
169
|
-
},
|
|
170
|
-
onSuccess: () => {
|
|
171
|
-
setError(null);
|
|
172
|
-
setDecision("");
|
|
173
|
-
setComments("");
|
|
174
|
-
setIsWaitingForProcessing(true);
|
|
175
|
-
queryClient.invalidateQueries({ queryKey: ["workflow-instances", orderId] });
|
|
176
|
-
queryClient.invalidateQueries({ queryKey: ["workflow-tasks", activeInstance?.id] });
|
|
177
|
-
queryClient.invalidateQueries({ queryKey: ["sales-order", orderId] });
|
|
178
|
-
},
|
|
179
|
-
onError: (err) => {
|
|
180
|
-
setError(err.message);
|
|
181
|
-
}
|
|
182
|
-
});
|
|
183
|
-
const handleStartWorkflow = () => {
|
|
184
|
-
setIsWaitingForProcessing(true);
|
|
185
|
-
startWorkflowMutation.mutate();
|
|
186
|
-
};
|
|
187
|
-
const handleCompleteTask = () => {
|
|
188
|
-
if (!pendingTask || !decision) return;
|
|
189
|
-
completeTaskMutation.mutate({
|
|
190
|
-
taskId: pendingTask.id,
|
|
191
|
-
formData: { decision, comments }
|
|
192
|
-
});
|
|
193
|
-
};
|
|
194
|
-
const handleKeyDown = (e) => {
|
|
195
|
-
if ((e.metaKey || e.ctrlKey) && e.key === "Enter") {
|
|
196
|
-
e.preventDefault();
|
|
197
|
-
if (pendingTask && decision && !isSubmitting) {
|
|
198
|
-
handleCompleteTask();
|
|
199
|
-
}
|
|
200
|
-
}
|
|
201
|
-
};
|
|
202
|
-
const isLoading = instancesLoading || tasksLoading;
|
|
203
|
-
const isSubmitting = startWorkflowMutation.isPending || completeTaskMutation.isPending;
|
|
204
|
-
if (!orderId) return null;
|
|
205
|
-
const orderStatus = currentOrderStatus?.toLowerCase();
|
|
206
|
-
if (orderStatus !== "pending_approval") {
|
|
207
|
-
return null;
|
|
208
|
-
}
|
|
209
|
-
if (isLoading) {
|
|
210
|
-
return /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-center p-4", children: [
|
|
211
|
-
/* @__PURE__ */ jsx(Spinner, { size: "sm" }),
|
|
212
|
-
/* @__PURE__ */ jsx("span", { className: "ml-2 text-sm text-muted-foreground", children: t("common.loading", "Loading...") })
|
|
213
|
-
] });
|
|
214
|
-
}
|
|
215
|
-
const getStatusBadge = () => {
|
|
216
|
-
if (!activeInstance) return null;
|
|
217
|
-
const statusVariants = {
|
|
218
|
-
RUNNING: "default",
|
|
219
|
-
PAUSED: "secondary",
|
|
220
|
-
WAITING_FOR_ACTIVITIES: "secondary",
|
|
221
|
-
COMPLETED: "default",
|
|
222
|
-
FAILED: "destructive",
|
|
223
|
-
CANCELLED: "outline"
|
|
224
|
-
};
|
|
225
|
-
return /* @__PURE__ */ jsx(Badge, { variant: statusVariants[activeInstance.status] || "outline", children: t(`workflows.instances.statuses.${activeInstance.status}`, activeInstance.status) });
|
|
226
|
-
};
|
|
227
|
-
return /* @__PURE__ */ jsxs("div", { className: "space-y-3 rounded-lg border bg-card p-4 shadow-sm", children: [
|
|
228
|
-
/* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between gap-3", children: [
|
|
229
|
-
/* @__PURE__ */ jsxs("div", { children: [
|
|
230
|
-
/* @__PURE__ */ jsx("div", { className: "text-sm font-semibold text-foreground", children: t("workflows.orderApproval.groupLabel", "Order Approval") }),
|
|
231
|
-
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: t("workflows.orderApproval.groupDescription", "Review and approve or reject this order") })
|
|
232
|
-
] }),
|
|
233
|
-
getStatusBadge()
|
|
234
|
-
] }),
|
|
235
|
-
error && /* @__PURE__ */ jsx("div", { className: "rounded-md border border-red-200 bg-red-50 p-3 text-xs text-red-800", children: error }),
|
|
236
|
-
!activeInstance && /* @__PURE__ */ jsxs("div", { className: "space-y-3", children: [
|
|
237
|
-
/* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground", children: t("workflows.orderApproval.noWorkflowActive", "No approval workflow is active for this order.") }),
|
|
238
|
-
/* @__PURE__ */ jsxs(
|
|
239
|
-
Button,
|
|
240
|
-
{
|
|
241
|
-
onClick: handleStartWorkflow,
|
|
242
|
-
disabled: isSubmitting,
|
|
243
|
-
variant: "default",
|
|
244
|
-
size: "sm",
|
|
245
|
-
children: [
|
|
246
|
-
isSubmitting && /* @__PURE__ */ jsx(Spinner, { size: "sm", className: "mr-2" }),
|
|
247
|
-
t("workflows.orderApproval.requestApproval", "Request Approval")
|
|
248
|
-
]
|
|
249
|
-
}
|
|
250
|
-
)
|
|
251
|
-
] }),
|
|
252
|
-
activeInstance && pendingTask && /* @__PURE__ */ jsxs("div", { className: "space-y-3", onKeyDown: handleKeyDown, children: [
|
|
253
|
-
/* @__PURE__ */ jsxs("div", { className: "rounded-md border border-amber-200 bg-amber-50 p-3", children: [
|
|
254
|
-
/* @__PURE__ */ jsx("p", { className: "text-sm font-medium text-amber-800", children: t("workflows.orderApproval.pendingTitle", "Pending Approval") }),
|
|
255
|
-
/* @__PURE__ */ jsx("p", { className: "text-xs text-amber-700 mt-1", children: t("workflows.orderApproval.pendingDescription", "This order requires approval before processing.") })
|
|
256
|
-
] }),
|
|
257
|
-
/* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
|
|
258
|
-
/* @__PURE__ */ jsx("label", { className: "text-sm font-medium", children: t("workflows.orderApproval.decisionLabel", "Decision") }),
|
|
259
|
-
/* @__PURE__ */ jsxs("div", { className: "flex gap-2", children: [
|
|
260
|
-
/* @__PURE__ */ jsx(
|
|
261
|
-
Button,
|
|
262
|
-
{
|
|
263
|
-
variant: decision === "approve" ? "default" : "outline",
|
|
264
|
-
size: "sm",
|
|
265
|
-
onClick: () => setDecision("approve"),
|
|
266
|
-
className: decision === "approve" ? "bg-emerald-600 hover:bg-emerald-700" : "",
|
|
267
|
-
children: t("workflows.orderApproval.approveButton", "Approve")
|
|
268
|
-
}
|
|
269
|
-
),
|
|
270
|
-
/* @__PURE__ */ jsx(
|
|
271
|
-
Button,
|
|
272
|
-
{
|
|
273
|
-
variant: decision === "reject" ? "destructive" : "outline",
|
|
274
|
-
size: "sm",
|
|
275
|
-
onClick: () => setDecision("reject"),
|
|
276
|
-
children: t("workflows.orderApproval.rejectButton", "Reject")
|
|
277
|
-
}
|
|
278
|
-
)
|
|
279
|
-
] })
|
|
280
|
-
] }),
|
|
281
|
-
/* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
|
|
282
|
-
/* @__PURE__ */ jsxs("label", { className: "text-sm font-medium", children: [
|
|
283
|
-
t("workflows.orderApproval.commentsLabel", "Comments"),
|
|
284
|
-
" ",
|
|
285
|
-
/* @__PURE__ */ jsxs("span", { className: "text-muted-foreground font-normal", children: [
|
|
286
|
-
"(",
|
|
287
|
-
t("common.optional", "optional"),
|
|
288
|
-
")"
|
|
289
|
-
] })
|
|
290
|
-
] }),
|
|
291
|
-
/* @__PURE__ */ jsx(
|
|
292
|
-
Textarea,
|
|
293
|
-
{
|
|
294
|
-
value: comments,
|
|
295
|
-
onChange: (e) => setComments(e.target.value),
|
|
296
|
-
placeholder: t("workflows.orderApproval.commentsPlaceholder", "Add optional comments..."),
|
|
297
|
-
rows: 2
|
|
298
|
-
}
|
|
299
|
-
)
|
|
300
|
-
] }),
|
|
301
|
-
/* @__PURE__ */ jsxs(
|
|
302
|
-
Button,
|
|
303
|
-
{
|
|
304
|
-
onClick: handleCompleteTask,
|
|
305
|
-
disabled: !decision || isSubmitting,
|
|
306
|
-
variant: "default",
|
|
307
|
-
size: "sm",
|
|
308
|
-
className: "w-full",
|
|
309
|
-
children: [
|
|
310
|
-
isSubmitting && /* @__PURE__ */ jsx(Spinner, { size: "sm", className: "mr-2" }),
|
|
311
|
-
t("workflows.orderApproval.submitDecision", "Submit Decision")
|
|
312
|
-
]
|
|
313
|
-
}
|
|
314
|
-
)
|
|
315
|
-
] }),
|
|
316
|
-
activeInstance && !pendingTask && activeInstance.status !== "COMPLETED" && /* @__PURE__ */ jsx("div", { className: "rounded-md border border-blue-200 bg-blue-50 p-3", children: /* @__PURE__ */ jsx("p", { className: "text-sm text-blue-800", children: t("workflows.orderApproval.processing", "Workflow is processing...") }) }),
|
|
317
|
-
activeInstance && activeInstance.status === "COMPLETED" && /* @__PURE__ */ jsx("div", { className: "rounded-md border border-emerald-200 bg-emerald-50 p-3", children: /* @__PURE__ */ jsx("p", { className: "text-sm text-emerald-800", children: t("workflows.orderApproval.completed", "Approval workflow completed.") }) })
|
|
318
|
-
] });
|
|
319
|
-
}
|
|
320
|
-
export {
|
|
321
|
-
OrderApprovalWidget as default
|
|
322
|
-
};
|
|
323
|
-
//# sourceMappingURL=widget.client.js.map
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": 3,
|
|
3
|
-
"sources": ["../../../../../../src/modules/workflows/widgets/injection/order-approval/widget.client.tsx"],
|
|
4
|
-
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { InjectionWidgetComponentProps } from '@open-mercato/shared/modules/widgets/injection'\nimport { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { Textarea } from '@open-mercato/ui/primitives/textarea'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { emitSalesDocumentDataRefresh } from '@open-mercato/core/modules/sales/lib/frontend/documentDataEvents'\n\ntype OrderRecord = {\n id: string\n orderNumber?: string\n status?: string\n statusEntryId?: string\n}\n\ntype WorkflowInstance = {\n id: string\n workflowId: string\n status: 'RUNNING' | 'PAUSED' | 'COMPLETED' | 'FAILED' | 'CANCELLED' | 'WAITING_FOR_ACTIVITIES'\n currentStepId: string\n context: Record<string, any>\n}\n\ntype UserTask = {\n id: string\n taskName: string\n status: 'PENDING' | 'IN_PROGRESS' | 'COMPLETED' | 'CANCELLED'\n workflowInstanceId: string\n claimedBy: string | null\n}\n\ntype DictionaryEntry = {\n id: string\n value: string\n label: string\n color?: string\n icon?: string\n}\n\nconst WORKFLOW_ID = 'sales_order_approval_v1'\n\nexport default function OrderApprovalWidget({ data }: InjectionWidgetComponentProps<unknown, OrderRecord>) {\n const t = useT()\n const queryClient = useQueryClient()\n const orderId = data?.id\n\n const [decision, setDecision] = React.useState<'approve' | 'reject' | ''>('')\n const [comments, setComments] = React.useState('')\n const [error, setError] = React.useState<string | null>(null)\n\n // Track if we're waiting for workflow to process (for polling)\n const [isWaitingForProcessing, setIsWaitingForProcessing] = React.useState(false)\n\n // Fetch current order status directly (to detect when workflow updates it)\n // Use list endpoint with ID filter since single-item endpoint doesn't exist\n const { data: orderData } = useQuery({\n queryKey: ['order-status', orderId],\n queryFn: async () => {\n if (!orderId) return null\n const result = await apiCall<{ items: Array<{ id: string; status?: string | null }> }>(\n `/api/sales/orders?id=${orderId}&pageSize=1`\n )\n return result.ok && result.result?.items?.[0] ? result.result.items[0] : null\n },\n enabled: Boolean(orderId),\n staleTime: 5_000,\n // Poll when waiting for processing to detect status change\n refetchInterval: isWaitingForProcessing ? 2_000 : false,\n })\n\n // Use fresh order status from query, fallback to prop data\n // The API returns status as a string value (e.g., \"approved\", \"pending_approval\")\n const currentOrderStatus = orderData?.status || data?.status\n\n // Fetch active workflow instances for this order\n const { data: instancesData, isLoading: instancesLoading } = useQuery({\n queryKey: ['workflow-instances', orderId],\n queryFn: async () => {\n if (!orderId) return { data: [] }\n const result = await apiCall<{ data: WorkflowInstance[] }>(\n `/api/workflows/instances?entityId=${orderId}&status=RUNNING,PAUSED,WAITING_FOR_ACTIVITIES`\n )\n return result.ok ? result.result : { data: [] }\n },\n enabled: Boolean(orderId),\n staleTime: 5_000,\n // Poll every 2 seconds when waiting for processing\n refetchInterval: isWaitingForProcessing ? 2_000 : false,\n })\n\n const activeInstance = instancesData?.data?.find(\n (inst) => inst.workflowId === WORKFLOW_ID && !['COMPLETED', 'FAILED', 'CANCELLED'].includes(inst.status)\n )\n\n // Fetch pending user tasks for active instance\n const { data: tasksData, isLoading: tasksLoading } = useQuery({\n queryKey: ['workflow-tasks', activeInstance?.id],\n queryFn: async () => {\n if (!activeInstance?.id) return { data: [] }\n const result = await apiCall<{ data: UserTask[] }>(\n `/api/workflows/tasks?workflowInstanceId=${activeInstance.id}&status=PENDING,IN_PROGRESS`\n )\n return result.ok ? result.result : { data: [] }\n },\n enabled: Boolean(activeInstance?.id),\n staleTime: 5_000,\n // Poll every 2 seconds when waiting for processing\n refetchInterval: isWaitingForProcessing ? 2_000 : false,\n })\n\n const pendingTask = tasksData?.data?.[0]\n\n // Auto-detect when workflow is processing (active instance but no pending task)\n const isProcessing = activeInstance && !pendingTask && !['COMPLETED', 'FAILED', 'CANCELLED'].includes(activeInstance.status)\n\n // Track previous status to detect changes\n const prevStatusRef = React.useRef<string | undefined>(data?.status)\n\n // Update polling state when processing state changes\n React.useEffect(() => {\n if (isProcessing) {\n setIsWaitingForProcessing(true)\n } else if (!activeInstance || activeInstance.status === 'COMPLETED') {\n setIsWaitingForProcessing(false)\n }\n }, [isProcessing, activeInstance])\n\n // When order status changes, emit refresh event to update the page\n React.useEffect(() => {\n const newStatus = orderData?.status\n const oldStatus = prevStatusRef.current\n\n if (newStatus && oldStatus && newStatus !== oldStatus) {\n // Status changed - emit document refresh event to reload the page data\n if (orderId) {\n emitSalesDocumentDataRefresh({ documentId: orderId, kind: 'order' })\n }\n }\n\n prevStatusRef.current = newStatus || oldStatus\n }, [orderData?.status, orderId])\n\n // First fetch dictionaries to find the order_status dictionary ID\n const { data: dictionariesData } = useQuery({\n queryKey: ['dictionaries'],\n queryFn: async () => {\n const result = await apiCall<{ items: Array<{ id: string; key: string }> }>(\n '/api/dictionaries'\n )\n return result.ok ? result.result : { items: [] }\n },\n staleTime: 60_000,\n })\n\n const orderStatusDictionaryId = React.useMemo(() => {\n const dict = dictionariesData?.items?.find(d => d.key === 'sales.order_status')\n return dict?.id\n }, [dictionariesData])\n\n // Fetch order status dictionary entries using the dictionary ID\n const { data: statusEntriesData } = useQuery({\n queryKey: ['dictionary-entries', orderStatusDictionaryId],\n queryFn: async () => {\n if (!orderStatusDictionaryId) return { items: [] }\n const result = await apiCall<{ items: DictionaryEntry[] }>(\n `/api/dictionaries/${orderStatusDictionaryId}/entries`\n )\n return result.ok ? result.result : { items: [] }\n },\n enabled: Boolean(orderStatusDictionaryId),\n staleTime: 60_000,\n })\n\n const findStatusId = React.useCallback((code: string) => {\n const entries = statusEntriesData?.items || []\n // The API returns 'value' as the code/key for dictionary entries\n const entry = entries.find(\n (e) => e.value === code || e.label?.toLowerCase() === code.toLowerCase()\n )\n return entry?.id\n }, [statusEntriesData])\n\n // Start workflow mutation\n const startWorkflowMutation = useMutation({\n mutationFn: async () => {\n const pendingApprovalStatusId = findStatusId('pending_approval')\n const approvedStatusId = findStatusId('approved')\n const rejectedStatusId = findStatusId('rejected')\n\n if (!pendingApprovalStatusId || !approvedStatusId || !rejectedStatusId) {\n throw new Error(t('workflows.orderApproval.missingStatuses', 'Missing order status entries. Please ensure pending_approval, approved, and rejected statuses exist in the sales.order_status dictionary.'))\n }\n\n const result = await apiCall<{ data: WorkflowInstance }>('/api/workflows/instances', {\n method: 'POST',\n body: JSON.stringify({\n workflowId: WORKFLOW_ID,\n initialContext: {\n orderId,\n pendingApprovalStatusId,\n approvedStatusId,\n rejectedStatusId,\n },\n metadata: {\n entityType: 'SalesOrder',\n entityId: orderId,\n },\n }),\n })\n\n if (!result.ok) {\n const errorResult = result.result as { error?: string } | null\n throw new Error(errorResult?.error || t('workflows.orderApproval.startError', 'Failed to start approval workflow'))\n }\n\n return result.result\n },\n onSuccess: () => {\n setError(null)\n queryClient.invalidateQueries({ queryKey: ['workflow-instances', orderId] })\n },\n onError: (err: Error) => {\n setError(err.message)\n },\n })\n\n // Complete task mutation\n const completeTaskMutation = useMutation({\n mutationFn: async ({ taskId, formData }: { taskId: string; formData: { decision: string; comments?: string } }) => {\n const result = await apiCall(`/api/workflows/tasks/${taskId}/complete`, {\n method: 'POST',\n body: JSON.stringify({ formData }),\n })\n\n if (!result.ok) {\n const errorResult = result.result as { error?: string } | null\n throw new Error(errorResult?.error || t('workflows.orderApproval.completeError', 'Failed to complete approval task'))\n }\n\n return result.result\n },\n onSuccess: () => {\n setError(null)\n setDecision('')\n setComments('')\n // Start polling immediately after submitting decision\n setIsWaitingForProcessing(true)\n queryClient.invalidateQueries({ queryKey: ['workflow-instances', orderId] })\n queryClient.invalidateQueries({ queryKey: ['workflow-tasks', activeInstance?.id] })\n // Also refresh the order data\n queryClient.invalidateQueries({ queryKey: ['sales-order', orderId] })\n },\n onError: (err: Error) => {\n setError(err.message)\n },\n })\n\n const handleStartWorkflow = () => {\n // Start polling after starting workflow\n setIsWaitingForProcessing(true)\n startWorkflowMutation.mutate()\n }\n\n const handleCompleteTask = () => {\n if (!pendingTask || !decision) return\n completeTaskMutation.mutate({\n taskId: pendingTask.id,\n formData: { decision, comments },\n })\n }\n\n // Handle keyboard shortcuts (Cmd/Ctrl+Enter to submit)\n const handleKeyDown = (e: React.KeyboardEvent) => {\n if ((e.metaKey || e.ctrlKey) && e.key === 'Enter') {\n e.preventDefault()\n if (pendingTask && decision && !isSubmitting) {\n handleCompleteTask()\n }\n }\n }\n\n const isLoading = instancesLoading || tasksLoading\n const isSubmitting = startWorkflowMutation.isPending || completeTaskMutation.isPending\n\n // Don't render if no orderId\n if (!orderId) return null\n\n // Only show widget when order status is pending_approval\n const orderStatus = currentOrderStatus?.toLowerCase()\n if (orderStatus !== 'pending_approval') {\n return null\n }\n\n // Loading state\n if (isLoading) {\n return (\n <div className=\"flex items-center justify-center p-4\">\n <Spinner size=\"sm\" />\n <span className=\"ml-2 text-sm text-muted-foreground\">{t('common.loading', 'Loading...')}</span>\n </div>\n )\n }\n\n // Show workflow status badge\n const getStatusBadge = () => {\n if (!activeInstance) return null\n\n const statusVariants: Record<string, 'default' | 'secondary' | 'destructive' | 'outline'> = {\n RUNNING: 'default',\n PAUSED: 'secondary',\n WAITING_FOR_ACTIVITIES: 'secondary',\n COMPLETED: 'default',\n FAILED: 'destructive',\n CANCELLED: 'outline',\n }\n\n return (\n <Badge variant={statusVariants[activeInstance.status] || 'outline'}>\n {t(`workflows.instances.statuses.${activeInstance.status}`, activeInstance.status)}\n </Badge>\n )\n }\n\n return (\n <div className=\"space-y-3 rounded-lg border bg-card p-4 shadow-sm\">\n <div className=\"flex items-start justify-between gap-3\">\n <div>\n <div className=\"text-sm font-semibold text-foreground\">\n {t('workflows.orderApproval.groupLabel', 'Order Approval')}\n </div>\n <p className=\"text-xs text-muted-foreground\">\n {t('workflows.orderApproval.groupDescription', 'Review and approve or reject this order')}\n </p>\n </div>\n {getStatusBadge()}\n </div>\n\n {error && (\n <div className=\"rounded-md border border-red-200 bg-red-50 p-3 text-xs text-red-800\">\n {error}\n </div>\n )}\n\n {/* No active workflow - show request approval button */}\n {!activeInstance && (\n <div className=\"space-y-3\">\n <p className=\"text-sm text-muted-foreground\">\n {t('workflows.orderApproval.noWorkflowActive', 'No approval workflow is active for this order.')}\n </p>\n <Button\n onClick={handleStartWorkflow}\n disabled={isSubmitting}\n variant=\"default\"\n size=\"sm\"\n >\n {isSubmitting && <Spinner size=\"sm\" className=\"mr-2\" />}\n {t('workflows.orderApproval.requestApproval', 'Request Approval')}\n </Button>\n </div>\n )}\n\n {/* Active workflow with pending task - show approve/reject UI */}\n {activeInstance && pendingTask && (\n <div className=\"space-y-3\" onKeyDown={handleKeyDown}>\n <div className=\"rounded-md border border-amber-200 bg-amber-50 p-3\">\n <p className=\"text-sm font-medium text-amber-800\">\n {t('workflows.orderApproval.pendingTitle', 'Pending Approval')}\n </p>\n <p className=\"text-xs text-amber-700 mt-1\">\n {t('workflows.orderApproval.pendingDescription', 'This order requires approval before processing.')}\n </p>\n </div>\n\n <div className=\"space-y-2\">\n <label className=\"text-sm font-medium\">\n {t('workflows.orderApproval.decisionLabel', 'Decision')}\n </label>\n <div className=\"flex gap-2\">\n <Button\n variant={decision === 'approve' ? 'default' : 'outline'}\n size=\"sm\"\n onClick={() => setDecision('approve')}\n className={decision === 'approve' ? 'bg-emerald-600 hover:bg-emerald-700' : ''}\n >\n {t('workflows.orderApproval.approveButton', 'Approve')}\n </Button>\n <Button\n variant={decision === 'reject' ? 'destructive' : 'outline'}\n size=\"sm\"\n onClick={() => setDecision('reject')}\n >\n {t('workflows.orderApproval.rejectButton', 'Reject')}\n </Button>\n </div>\n </div>\n\n <div className=\"space-y-2\">\n <label className=\"text-sm font-medium\">\n {t('workflows.orderApproval.commentsLabel', 'Comments')} <span className=\"text-muted-foreground font-normal\">({t('common.optional', 'optional')})</span>\n </label>\n <Textarea\n value={comments}\n onChange={(e) => setComments(e.target.value)}\n placeholder={t('workflows.orderApproval.commentsPlaceholder', 'Add optional comments...')}\n rows={2}\n />\n </div>\n\n <Button\n onClick={handleCompleteTask}\n disabled={!decision || isSubmitting}\n variant=\"default\"\n size=\"sm\"\n className=\"w-full\"\n >\n {isSubmitting && <Spinner size=\"sm\" className=\"mr-2\" />}\n {t('workflows.orderApproval.submitDecision', 'Submit Decision')}\n </Button>\n </div>\n )}\n\n {/* Active workflow but no pending task (processing) */}\n {activeInstance && !pendingTask && activeInstance.status !== 'COMPLETED' && (\n <div className=\"rounded-md border border-blue-200 bg-blue-50 p-3\">\n <p className=\"text-sm text-blue-800\">\n {t('workflows.orderApproval.processing', 'Workflow is processing...')}\n </p>\n </div>\n )}\n\n {/* Completed workflow */}\n {activeInstance && activeInstance.status === 'COMPLETED' && (\n <div className=\"rounded-md border border-emerald-200 bg-emerald-50 p-3\">\n <p className=\"text-sm text-emerald-800\">\n {t('workflows.orderApproval.completed', 'Approval workflow completed.')}\n </p>\n </div>\n )}\n </div>\n )\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
6
|
-
"names": []
|
|
7
|
-
}
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
import OrderApprovalWidget from "./widget.client.js";
|
|
2
|
-
const widget = {
|
|
3
|
-
metadata: {
|
|
4
|
-
id: "workflows.injection.order-approval",
|
|
5
|
-
title: "Order Approval",
|
|
6
|
-
description: "Approve or reject orders requiring authorization",
|
|
7
|
-
features: ["sales.orders.approve"],
|
|
8
|
-
priority: 100,
|
|
9
|
-
enabled: true
|
|
10
|
-
},
|
|
11
|
-
Widget: OrderApprovalWidget
|
|
12
|
-
};
|
|
13
|
-
var widget_default = widget;
|
|
14
|
-
export {
|
|
15
|
-
widget_default as default
|
|
16
|
-
};
|
|
17
|
-
//# sourceMappingURL=widget.js.map
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": 3,
|
|
3
|
-
"sources": ["../../../../../../src/modules/workflows/widgets/injection/order-approval/widget.ts"],
|
|
4
|
-
"sourcesContent": ["import type { InjectionWidgetModule } from '@open-mercato/shared/modules/widgets/injection'\nimport OrderApprovalWidget from './widget.client'\n\nconst widget: InjectionWidgetModule<any, any> = {\n metadata: {\n id: 'workflows.injection.order-approval',\n title: 'Order Approval',\n description: 'Approve or reject orders requiring authorization',\n features: ['sales.orders.approve'],\n priority: 100,\n enabled: true,\n },\n Widget: OrderApprovalWidget,\n}\n\nexport default widget\n"],
|
|
5
|
-
"mappings": "AACA,OAAO,yBAAyB;AAEhC,MAAM,SAA0C;AAAA,EAC9C,UAAU;AAAA,IACR,IAAI;AAAA,IACJ,OAAO;AAAA,IACP,aAAa;AAAA,IACb,UAAU,CAAC,sBAAsB;AAAA,IACjC,UAAU;AAAA,IACV,SAAS;AAAA,EACX;AAAA,EACA,QAAQ;AACV;AAEA,IAAO,iBAAQ;",
|
|
6
|
-
"names": []
|
|
7
|
-
}
|
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
const injectionTable = {
|
|
2
|
-
// Inject the order approval widget into the sales order detail page
|
|
3
|
-
"sales.document.detail.order:details": [
|
|
4
|
-
{
|
|
5
|
-
widgetId: "workflows.injection.order-approval",
|
|
6
|
-
kind: "group",
|
|
7
|
-
column: 2,
|
|
8
|
-
groupLabel: "workflows.orderApproval.groupLabel",
|
|
9
|
-
groupDescription: "workflows.orderApproval.groupDescription",
|
|
10
|
-
priority: 200
|
|
11
|
-
}
|
|
12
|
-
]
|
|
13
|
-
};
|
|
14
|
-
var injection_table_default = injectionTable;
|
|
15
|
-
export {
|
|
16
|
-
injection_table_default as default,
|
|
17
|
-
injectionTable
|
|
18
|
-
};
|
|
19
|
-
//# sourceMappingURL=injection-table.js.map
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": 3,
|
|
3
|
-
"sources": ["../../../../src/modules/workflows/widgets/injection-table.ts"],
|
|
4
|
-
"sourcesContent": ["import type { ModuleInjectionTable } from '@open-mercato/shared/modules/widgets/injection'\n\n/**\n * Workflows module injection table\n * Maps injection spot IDs to widget IDs for automatic widget injection\n */\nexport const injectionTable: ModuleInjectionTable = {\n // Inject the order approval widget into the sales order detail page\n 'sales.document.detail.order:details': [\n {\n widgetId: 'workflows.injection.order-approval',\n kind: 'group',\n column: 2,\n groupLabel: 'workflows.orderApproval.groupLabel',\n groupDescription: 'workflows.orderApproval.groupDescription',\n priority: 200,\n },\n ],\n}\n\nexport default injectionTable\n"],
|
|
5
|
-
"mappings": "AAMO,MAAM,iBAAuC;AAAA;AAAA,EAElD,uCAAuC;AAAA,IACrC;AAAA,MACE,UAAU;AAAA,MACV,MAAM;AAAA,MACN,QAAQ;AAAA,MACR,YAAY;AAAA,MACZ,kBAAkB;AAAA,MAClB,UAAU;AAAA,IACZ;AAAA,EACF;AACF;AAEA,IAAO,0BAAQ;",
|
|
6
|
-
"names": []
|
|
7
|
-
}
|
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
export const id = 'id'
|
|
2
|
-
export const name = 'name'
|
|
3
|
-
export const description = 'description'
|
|
4
|
-
export const workflow_definition_id = 'workflow_definition_id'
|
|
5
|
-
export const event_pattern = 'event_pattern'
|
|
6
|
-
export const config = 'config'
|
|
7
|
-
export const enabled = 'enabled'
|
|
8
|
-
export const priority = 'priority'
|
|
9
|
-
export const tenant_id = 'tenant_id'
|
|
10
|
-
export const organization_id = 'organization_id'
|
|
11
|
-
export const created_by = 'created_by'
|
|
12
|
-
export const updated_by = 'updated_by'
|
|
13
|
-
export const created_at = 'created_at'
|
|
14
|
-
export const updated_at = 'updated_at'
|
|
15
|
-
export const deleted_at = 'deleted_at'
|
|
@@ -1,39 +0,0 @@
|
|
|
1
|
-
import { createModuleEvents } from '@open-mercato/shared/modules/events'
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Auth Module Events
|
|
5
|
-
*
|
|
6
|
-
* Declares all events that can be emitted by the auth module.
|
|
7
|
-
*/
|
|
8
|
-
const events = [
|
|
9
|
-
// Users
|
|
10
|
-
{ id: 'auth.users.created', label: 'User Created', entity: 'users', category: 'crud' },
|
|
11
|
-
{ id: 'auth.users.updated', label: 'User Updated', entity: 'users', category: 'crud' },
|
|
12
|
-
{ id: 'auth.users.deleted', label: 'User Deleted', entity: 'users', category: 'crud' },
|
|
13
|
-
|
|
14
|
-
// Roles
|
|
15
|
-
{ id: 'auth.roles.created', label: 'Role Created', entity: 'roles', category: 'crud' },
|
|
16
|
-
{ id: 'auth.roles.updated', label: 'Role Updated', entity: 'roles', category: 'crud' },
|
|
17
|
-
{ id: 'auth.roles.deleted', label: 'Role Deleted', entity: 'roles', category: 'crud' },
|
|
18
|
-
|
|
19
|
-
// Authentication events
|
|
20
|
-
{ id: 'auth.login.success', label: 'Login Successful', category: 'lifecycle' },
|
|
21
|
-
{ id: 'auth.login.failed', label: 'Login Failed', category: 'lifecycle' },
|
|
22
|
-
{ id: 'auth.logout', label: 'User Logged Out', category: 'lifecycle' },
|
|
23
|
-
{ id: 'auth.password.changed', label: 'Password Changed', category: 'lifecycle' },
|
|
24
|
-
{ id: 'auth.password.reset.requested', label: 'Password Reset Requested', category: 'lifecycle' },
|
|
25
|
-
{ id: 'auth.password.reset.completed', label: 'Password Reset Completed', category: 'lifecycle' },
|
|
26
|
-
] as const
|
|
27
|
-
|
|
28
|
-
export const eventsConfig = createModuleEvents({
|
|
29
|
-
moduleId: 'auth',
|
|
30
|
-
events,
|
|
31
|
-
})
|
|
32
|
-
|
|
33
|
-
/** Type-safe event emitter for auth module */
|
|
34
|
-
export const emitAuthEvent = eventsConfig.emit
|
|
35
|
-
|
|
36
|
-
/** Event IDs that can be emitted by the auth module */
|
|
37
|
-
export type AuthEventId = typeof events[number]['id']
|
|
38
|
-
|
|
39
|
-
export default eventsConfig
|
|
@@ -1,163 +0,0 @@
|
|
|
1
|
-
import { NextResponse } from 'next/server'
|
|
2
|
-
import { z } from 'zod'
|
|
3
|
-
import type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'
|
|
4
|
-
import { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'
|
|
5
|
-
import { createRequestContainer } from '@open-mercato/shared/lib/di/container'
|
|
6
|
-
import type { EntityManager } from '@mikro-orm/postgresql'
|
|
7
|
-
import * as ruleEngine from '../../../lib/rule-engine'
|
|
8
|
-
|
|
9
|
-
const executeByIdRequestSchema = z.object({
|
|
10
|
-
data: z.any(),
|
|
11
|
-
dryRun: z.boolean().optional().default(false),
|
|
12
|
-
entityType: z.string().optional(),
|
|
13
|
-
entityId: z.string().optional(),
|
|
14
|
-
eventType: z.string().optional(),
|
|
15
|
-
})
|
|
16
|
-
|
|
17
|
-
const executeByIdResponseSchema = z.object({
|
|
18
|
-
success: z.boolean(),
|
|
19
|
-
ruleId: z.string(),
|
|
20
|
-
ruleName: z.string(),
|
|
21
|
-
conditionResult: z.boolean(),
|
|
22
|
-
actionsExecuted: z.object({
|
|
23
|
-
success: z.boolean(),
|
|
24
|
-
results: z.array(z.object({
|
|
25
|
-
type: z.string(),
|
|
26
|
-
success: z.boolean(),
|
|
27
|
-
error: z.string().optional(),
|
|
28
|
-
})),
|
|
29
|
-
}).nullable(),
|
|
30
|
-
executionTime: z.number(),
|
|
31
|
-
error: z.string().optional(),
|
|
32
|
-
logId: z.string().optional(),
|
|
33
|
-
})
|
|
34
|
-
|
|
35
|
-
const errorResponseSchema = z.object({
|
|
36
|
-
error: z.string(),
|
|
37
|
-
})
|
|
38
|
-
|
|
39
|
-
const routeMetadata = {
|
|
40
|
-
POST: { requireAuth: true, requireFeatures: ['business_rules.execute'] },
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
export const metadata = routeMetadata
|
|
44
|
-
|
|
45
|
-
interface RouteContext {
|
|
46
|
-
params: Promise<{ ruleId: string }>
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
export async function POST(req: Request, context: RouteContext) {
|
|
50
|
-
const auth = await getAuthFromRequest(req)
|
|
51
|
-
if (!auth) {
|
|
52
|
-
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
const params = await context.params
|
|
56
|
-
const ruleId = params.ruleId
|
|
57
|
-
|
|
58
|
-
if (!ruleId || !z.uuid().safeParse(ruleId).success) {
|
|
59
|
-
return NextResponse.json({ error: 'Invalid rule ID' }, { status: 400 })
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
const container = await createRequestContainer()
|
|
63
|
-
const em = container.resolve('em') as EntityManager
|
|
64
|
-
|
|
65
|
-
let body: any
|
|
66
|
-
try {
|
|
67
|
-
body = await req.json()
|
|
68
|
-
} catch {
|
|
69
|
-
return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
const parsed = executeByIdRequestSchema.safeParse(body)
|
|
73
|
-
if (!parsed.success) {
|
|
74
|
-
const errors = parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)
|
|
75
|
-
return NextResponse.json({ error: `Validation failed: ${errors.join(', ')}` }, { status: 400 })
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
const { data, dryRun, entityType, entityId, eventType } = parsed.data
|
|
79
|
-
|
|
80
|
-
const execContext: ruleEngine.DirectRuleExecutionContext = {
|
|
81
|
-
ruleId,
|
|
82
|
-
data,
|
|
83
|
-
user: {
|
|
84
|
-
id: auth.sub,
|
|
85
|
-
email: auth.email,
|
|
86
|
-
role: (auth.role as string) ?? undefined,
|
|
87
|
-
},
|
|
88
|
-
tenantId: auth.tenantId ?? '',
|
|
89
|
-
organizationId: auth.orgId ?? '',
|
|
90
|
-
executedBy: auth.sub ?? auth.email ?? undefined,
|
|
91
|
-
dryRun,
|
|
92
|
-
entityType,
|
|
93
|
-
entityId,
|
|
94
|
-
eventType,
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
try {
|
|
98
|
-
const result = await ruleEngine.executeRuleById(em, execContext)
|
|
99
|
-
|
|
100
|
-
const response = {
|
|
101
|
-
success: result.success,
|
|
102
|
-
ruleId: result.ruleId,
|
|
103
|
-
ruleName: result.ruleName,
|
|
104
|
-
conditionResult: result.conditionResult,
|
|
105
|
-
actionsExecuted: result.actionsExecuted ? {
|
|
106
|
-
success: result.actionsExecuted.success,
|
|
107
|
-
results: result.actionsExecuted.results.map(ar => ({
|
|
108
|
-
type: ar.action.type,
|
|
109
|
-
success: ar.success,
|
|
110
|
-
error: ar.error,
|
|
111
|
-
})),
|
|
112
|
-
} : null,
|
|
113
|
-
executionTime: result.executionTime,
|
|
114
|
-
error: result.error,
|
|
115
|
-
logId: result.logId,
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
// Return appropriate status based on result
|
|
119
|
-
const status = result.success ? 200 : (result.error === 'Rule not found' ? 404 : 200)
|
|
120
|
-
return NextResponse.json(response, { status })
|
|
121
|
-
} catch (error) {
|
|
122
|
-
const errorMessage = error instanceof Error ? error.message : String(error)
|
|
123
|
-
return NextResponse.json(
|
|
124
|
-
{ error: `Rule execution failed: ${errorMessage}` },
|
|
125
|
-
{ status: 500 }
|
|
126
|
-
)
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
export const openApi: OpenApiRouteDoc = {
|
|
131
|
-
tag: 'Business Rules',
|
|
132
|
-
summary: 'Execute a specific business rule by ID',
|
|
133
|
-
methods: {
|
|
134
|
-
POST: {
|
|
135
|
-
summary: 'Execute a specific rule by its database UUID',
|
|
136
|
-
description: 'Directly executes a specific business rule identified by its UUID, bypassing the normal entityType/eventType discovery mechanism. Useful for workflows and targeted rule execution.',
|
|
137
|
-
pathParams: z.object({
|
|
138
|
-
ruleId: z.string().uuid().describe('The database UUID of the business rule to execute'),
|
|
139
|
-
}),
|
|
140
|
-
requestBody: {
|
|
141
|
-
contentType: 'application/json',
|
|
142
|
-
schema: executeByIdRequestSchema,
|
|
143
|
-
},
|
|
144
|
-
responses: [
|
|
145
|
-
{
|
|
146
|
-
status: 200,
|
|
147
|
-
description: 'Rule executed successfully',
|
|
148
|
-
schema: executeByIdResponseSchema,
|
|
149
|
-
},
|
|
150
|
-
{
|
|
151
|
-
status: 404,
|
|
152
|
-
description: 'Rule not found',
|
|
153
|
-
schema: errorResponseSchema,
|
|
154
|
-
},
|
|
155
|
-
],
|
|
156
|
-
errors: [
|
|
157
|
-
{ status: 400, description: 'Invalid request payload or rule ID', schema: errorResponseSchema },
|
|
158
|
-
{ status: 401, description: 'Unauthorized', schema: errorResponseSchema },
|
|
159
|
-
{ status: 500, description: 'Execution error', schema: errorResponseSchema },
|
|
160
|
-
],
|
|
161
|
-
},
|
|
162
|
-
},
|
|
163
|
-
}
|