@open-mercato/core 0.4.2-canary-51881f6bf3 → 0.4.2-canary-5f415b8a44
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/workflow_event_trigger/index.js +33 -0
- package/dist/generated/entities/workflow_event_trigger/index.js.map +7 -0
- package/dist/generated/entities.ids.generated.js +59 -58
- package/dist/generated/entities.ids.generated.js.map +2 -2
- package/dist/generated/entity-fields-registry.js +2 -0
- package/dist/generated/entity-fields-registry.js.map +2 -2
- package/dist/modules/auth/events.js +30 -0
- package/dist/modules/auth/events.js.map +7 -0
- package/dist/modules/business_rules/api/execute/[ruleId]/route.js +145 -0
- package/dist/modules/business_rules/api/execute/[ruleId]/route.js.map +7 -0
- package/dist/modules/business_rules/data/validators.js +34 -0
- package/dist/modules/business_rules/data/validators.js.map +2 -2
- package/dist/modules/business_rules/index.js +21 -1
- package/dist/modules/business_rules/index.js.map +2 -2
- package/dist/modules/business_rules/lib/rule-engine.js +182 -1
- package/dist/modules/business_rules/lib/rule-engine.js.map +2 -2
- package/dist/modules/catalog/events.js +34 -0
- package/dist/modules/catalog/events.js.map +7 -0
- package/dist/modules/customers/events.js +49 -0
- package/dist/modules/customers/events.js.map +7 -0
- package/dist/modules/directory/events.js +23 -0
- package/dist/modules/directory/events.js.map +7 -0
- package/dist/modules/sales/acl.js +1 -0
- package/dist/modules/sales/acl.js.map +2 -2
- package/dist/modules/sales/backend/sales/documents/[id]/page.js +12 -0
- package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
- package/dist/modules/sales/commands/documents.js +62 -0
- package/dist/modules/sales/commands/documents.js.map +2 -2
- package/dist/modules/sales/events.js +63 -0
- package/dist/modules/sales/events.js.map +7 -0
- package/dist/modules/sales/lib/dictionaries.js +3 -0
- package/dist/modules/sales/lib/dictionaries.js.map +2 -2
- package/dist/modules/sales/lib/frontend/documentDataEvents.js +25 -0
- package/dist/modules/sales/lib/frontend/documentDataEvents.js.map +7 -0
- package/dist/modules/workflows/acl.js +2 -0
- package/dist/modules/workflows/acl.js.map +2 -2
- package/dist/modules/workflows/api/instances/route.js +18 -6
- package/dist/modules/workflows/api/instances/route.js.map +2 -2
- package/dist/modules/workflows/api/tasks/route.js +6 -1
- package/dist/modules/workflows/api/tasks/route.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/[id]/page.js +9 -1
- 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 +24 -15
- 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 +150 -132
- 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 +5 -6
- package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
- package/dist/modules/workflows/cli.js +81 -3
- package/dist/modules/workflows/cli.js.map +3 -3
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js +481 -0
- package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +7 -0
- package/dist/modules/workflows/components/EventTriggersEditor.js +553 -0
- package/dist/modules/workflows/components/EventTriggersEditor.js.map +7 -0
- package/dist/modules/workflows/data/entities.js +64 -1
- package/dist/modules/workflows/data/entities.js.map +2 -2
- package/dist/modules/workflows/data/validators.js +115 -0
- package/dist/modules/workflows/data/validators.js.map +2 -2
- package/dist/modules/workflows/events.js +38 -0
- package/dist/modules/workflows/events.js.map +7 -0
- package/dist/modules/workflows/examples/checkout-demo-definition.json +1 -5
- package/dist/modules/workflows/examples/order-approval-definition.json +257 -0
- package/dist/modules/workflows/examples/order-approval-guard-rules.json +32 -0
- package/dist/modules/workflows/lib/activity-executor.js +75 -13
- package/dist/modules/workflows/lib/activity-executor.js.map +2 -2
- package/dist/modules/workflows/lib/event-trigger-service.js +308 -0
- package/dist/modules/workflows/lib/event-trigger-service.js.map +7 -0
- package/dist/modules/workflows/lib/graph-utils.js +71 -2
- package/dist/modules/workflows/lib/graph-utils.js.map +2 -2
- package/dist/modules/workflows/lib/seeds.js +22 -5
- package/dist/modules/workflows/lib/seeds.js.map +2 -2
- package/dist/modules/workflows/lib/start-validator.js +33 -23
- package/dist/modules/workflows/lib/start-validator.js.map +2 -2
- package/dist/modules/workflows/lib/transition-handler.js +157 -45
- package/dist/modules/workflows/lib/transition-handler.js.map +3 -3
- package/dist/modules/workflows/migrations/Migration20260123143500.js +36 -0
- package/dist/modules/workflows/migrations/Migration20260123143500.js.map +7 -0
- package/dist/modules/workflows/subscribers/event-trigger.js +78 -0
- package/dist/modules/workflows/subscribers/event-trigger.js.map +7 -0
- package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js +323 -0
- package/dist/modules/workflows/widgets/injection/order-approval/widget.client.js.map +7 -0
- package/dist/modules/workflows/widgets/injection/order-approval/widget.js +17 -0
- package/dist/modules/workflows/widgets/injection/order-approval/widget.js.map +7 -0
- package/dist/modules/workflows/widgets/injection-table.js +19 -0
- package/dist/modules/workflows/widgets/injection-table.js.map +7 -0
- package/generated/entities/workflow_event_trigger/index.ts +15 -0
- package/generated/entities.ids.generated.ts +59 -58
- package/generated/entity-fields-registry.ts +2 -0
- package/package.json +3 -5
- package/src/modules/auth/events.ts +39 -0
- package/src/modules/business_rules/api/execute/[ruleId]/route.ts +163 -0
- package/src/modules/business_rules/data/validators.ts +40 -0
- package/src/modules/business_rules/index.ts +25 -0
- package/src/modules/business_rules/lib/rule-engine.ts +281 -1
- package/src/modules/catalog/events.ts +45 -0
- package/src/modules/customers/events.ts +63 -0
- package/src/modules/directory/events.ts +31 -0
- package/src/modules/sales/acl.ts +1 -0
- package/src/modules/sales/backend/sales/documents/[id]/page.tsx +16 -0
- package/src/modules/sales/commands/documents.ts +74 -1
- package/src/modules/sales/events.ts +82 -0
- package/src/modules/sales/lib/dictionaries.ts +3 -0
- package/src/modules/sales/lib/frontend/documentDataEvents.ts +28 -0
- package/src/modules/workflows/acl.ts +2 -0
- package/src/modules/workflows/api/instances/route.ts +21 -7
- package/src/modules/workflows/api/tasks/route.ts +7 -1
- package/src/modules/workflows/backend/definitions/[id]/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/[id]/page.tsx +9 -0
- package/src/modules/workflows/backend/definitions/create/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/create/page.tsx +9 -0
- package/src/modules/workflows/backend/definitions/visual-editor/page.meta.ts +1 -1
- package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +21 -3
- 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 +5 -6
- package/src/modules/workflows/cli.ts +111 -0
- package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +581 -0
- package/src/modules/workflows/components/EventTriggersEditor.tsx +664 -0
- package/src/modules/workflows/data/entities.ts +124 -0
- package/src/modules/workflows/data/validators.ts +138 -0
- package/src/modules/workflows/events.ts +49 -0
- package/src/modules/workflows/examples/checkout-demo-definition.json +1 -5
- package/src/modules/workflows/examples/order-approval-definition.json +257 -0
- package/src/modules/workflows/examples/order-approval-guard-rules.json +32 -0
- package/src/modules/workflows/i18n/en.json +71 -0
- package/src/modules/workflows/lib/activity-executor.ts +129 -16
- package/src/modules/workflows/lib/event-trigger-service.ts +557 -0
- package/src/modules/workflows/lib/graph-utils.ts +117 -2
- package/src/modules/workflows/lib/seeds.ts +34 -8
- package/src/modules/workflows/lib/start-validator.ts +38 -28
- package/src/modules/workflows/lib/transition-handler.ts +208 -55
- package/src/modules/workflows/migrations/Migration20260123143500.ts +38 -0
- package/src/modules/workflows/subscribers/event-trigger.ts +109 -0
- package/src/modules/workflows/widgets/injection/order-approval/widget.client.tsx +446 -0
- package/src/modules/workflows/widgets/injection/order-approval/widget.ts +16 -0
- package/src/modules/workflows/widgets/injection-table.ts +21 -0
|
@@ -0,0 +1,553 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
3
|
+
import { useState, useCallback } from "react";
|
|
4
|
+
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
|
5
|
+
import { Button } from "@open-mercato/ui/primitives/button";
|
|
6
|
+
import { Input } from "@open-mercato/ui/primitives/input";
|
|
7
|
+
import { Textarea } from "@open-mercato/ui/primitives/textarea";
|
|
8
|
+
import { Label } from "@open-mercato/ui/primitives/label";
|
|
9
|
+
import { Switch } from "@open-mercato/ui/primitives/switch";
|
|
10
|
+
import { Spinner } from "@open-mercato/ui/primitives/spinner";
|
|
11
|
+
import { Badge } from "@open-mercato/ui/primitives/badge";
|
|
12
|
+
import {
|
|
13
|
+
Dialog,
|
|
14
|
+
DialogContent,
|
|
15
|
+
DialogDescription,
|
|
16
|
+
DialogFooter,
|
|
17
|
+
DialogHeader,
|
|
18
|
+
DialogTitle
|
|
19
|
+
} from "@open-mercato/ui/primitives/dialog";
|
|
20
|
+
import { Alert, AlertDescription, AlertTitle } from "@open-mercato/ui/primitives/alert";
|
|
21
|
+
import { apiCall } from "@open-mercato/ui/backend/utils/apiCall";
|
|
22
|
+
import { flash } from "@open-mercato/ui/backend/FlashMessages";
|
|
23
|
+
import { EventSelect } from "@open-mercato/ui/backend/inputs/EventSelect";
|
|
24
|
+
import { useT } from "@open-mercato/shared/lib/i18n/context";
|
|
25
|
+
import { Plus, Trash2, Edit2, Zap, AlertCircle, X } from "lucide-react";
|
|
26
|
+
const FILTER_OPERATORS = [
|
|
27
|
+
{ value: "eq", label: "Equals" },
|
|
28
|
+
{ value: "neq", label: "Not Equals" },
|
|
29
|
+
{ value: "gt", label: "Greater Than" },
|
|
30
|
+
{ value: "gte", label: "Greater Than or Equal" },
|
|
31
|
+
{ value: "lt", label: "Less Than" },
|
|
32
|
+
{ value: "lte", label: "Less Than or Equal" },
|
|
33
|
+
{ value: "contains", label: "Contains" },
|
|
34
|
+
{ value: "startsWith", label: "Starts With" },
|
|
35
|
+
{ value: "endsWith", label: "Ends With" },
|
|
36
|
+
{ value: "in", label: "In (array)" },
|
|
37
|
+
{ value: "notIn", label: "Not In (array)" },
|
|
38
|
+
{ value: "exists", label: "Exists" },
|
|
39
|
+
{ value: "notExists", label: "Not Exists" },
|
|
40
|
+
{ value: "regex", label: "Regex Match" }
|
|
41
|
+
];
|
|
42
|
+
const defaultFormValues = {
|
|
43
|
+
name: "",
|
|
44
|
+
description: "",
|
|
45
|
+
eventPattern: "",
|
|
46
|
+
enabled: true,
|
|
47
|
+
priority: 0,
|
|
48
|
+
filterConditions: [],
|
|
49
|
+
contextMappings: [],
|
|
50
|
+
debounceMs: "",
|
|
51
|
+
maxConcurrentInstances: ""
|
|
52
|
+
};
|
|
53
|
+
function EventTriggersEditor({
|
|
54
|
+
workflowDefinitionId,
|
|
55
|
+
workflowId,
|
|
56
|
+
className
|
|
57
|
+
}) {
|
|
58
|
+
const t = useT();
|
|
59
|
+
const queryClient = useQueryClient();
|
|
60
|
+
const [showDialog, setShowDialog] = useState(false);
|
|
61
|
+
const [editingTrigger, setEditingTrigger] = useState(null);
|
|
62
|
+
const [formValues, setFormValues] = useState(defaultFormValues);
|
|
63
|
+
const [deleteConfirmId, setDeleteConfirmId] = useState(null);
|
|
64
|
+
const { data: triggersData, isLoading } = useQuery({
|
|
65
|
+
queryKey: ["workflow-triggers", workflowDefinitionId],
|
|
66
|
+
queryFn: async () => {
|
|
67
|
+
const result = await apiCall(
|
|
68
|
+
`/api/workflows/triggers?workflowDefinitionId=${workflowDefinitionId}&limit=100`
|
|
69
|
+
);
|
|
70
|
+
if (!result.ok) {
|
|
71
|
+
throw new Error("Failed to load triggers");
|
|
72
|
+
}
|
|
73
|
+
return result.result?.data || [];
|
|
74
|
+
},
|
|
75
|
+
enabled: !!workflowDefinitionId
|
|
76
|
+
});
|
|
77
|
+
const triggers = triggersData || [];
|
|
78
|
+
const createMutation = useMutation({
|
|
79
|
+
mutationFn: async (values) => {
|
|
80
|
+
const payload = buildTriggerPayload(values);
|
|
81
|
+
const result = await apiCall(
|
|
82
|
+
"/api/workflows/triggers",
|
|
83
|
+
{
|
|
84
|
+
method: "POST",
|
|
85
|
+
headers: { "Content-Type": "application/json" },
|
|
86
|
+
body: JSON.stringify({
|
|
87
|
+
...payload,
|
|
88
|
+
workflowDefinitionId
|
|
89
|
+
})
|
|
90
|
+
}
|
|
91
|
+
);
|
|
92
|
+
if (!result.ok) {
|
|
93
|
+
throw new Error(result.result?.error || "Failed to create trigger");
|
|
94
|
+
}
|
|
95
|
+
return result.result?.data;
|
|
96
|
+
},
|
|
97
|
+
onSuccess: () => {
|
|
98
|
+
queryClient.invalidateQueries({ queryKey: ["workflow-triggers", workflowDefinitionId] });
|
|
99
|
+
flash("Event trigger created successfully", "success");
|
|
100
|
+
handleCloseDialog();
|
|
101
|
+
},
|
|
102
|
+
onError: (error) => {
|
|
103
|
+
flash(error.message, "error");
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
const updateMutation = useMutation({
|
|
107
|
+
mutationFn: async ({ id, values }) => {
|
|
108
|
+
const payload = buildTriggerPayload(values);
|
|
109
|
+
const result = await apiCall(
|
|
110
|
+
`/api/workflows/triggers/${id}`,
|
|
111
|
+
{
|
|
112
|
+
method: "PUT",
|
|
113
|
+
headers: { "Content-Type": "application/json" },
|
|
114
|
+
body: JSON.stringify(payload)
|
|
115
|
+
}
|
|
116
|
+
);
|
|
117
|
+
if (!result.ok) {
|
|
118
|
+
throw new Error(result.result?.error || "Failed to update trigger");
|
|
119
|
+
}
|
|
120
|
+
return result.result?.data;
|
|
121
|
+
},
|
|
122
|
+
onSuccess: () => {
|
|
123
|
+
queryClient.invalidateQueries({ queryKey: ["workflow-triggers", workflowDefinitionId] });
|
|
124
|
+
flash("Event trigger updated successfully", "success");
|
|
125
|
+
handleCloseDialog();
|
|
126
|
+
},
|
|
127
|
+
onError: (error) => {
|
|
128
|
+
flash(error.message, "error");
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
const deleteMutation = useMutation({
|
|
132
|
+
mutationFn: async (id) => {
|
|
133
|
+
const result = await apiCall(
|
|
134
|
+
`/api/workflows/triggers/${id}`,
|
|
135
|
+
{ method: "DELETE" }
|
|
136
|
+
);
|
|
137
|
+
if (!result.ok) {
|
|
138
|
+
throw new Error(result.result?.error || "Failed to delete trigger");
|
|
139
|
+
}
|
|
140
|
+
},
|
|
141
|
+
onSuccess: () => {
|
|
142
|
+
queryClient.invalidateQueries({ queryKey: ["workflow-triggers", workflowDefinitionId] });
|
|
143
|
+
flash("Event trigger deleted successfully", "success");
|
|
144
|
+
setDeleteConfirmId(null);
|
|
145
|
+
},
|
|
146
|
+
onError: (error) => {
|
|
147
|
+
flash(error.message, "error");
|
|
148
|
+
}
|
|
149
|
+
});
|
|
150
|
+
const buildTriggerPayload = useCallback((values) => {
|
|
151
|
+
const config = {};
|
|
152
|
+
if (values.filterConditions.length > 0) {
|
|
153
|
+
config.filterConditions = values.filterConditions.map((fc) => ({
|
|
154
|
+
field: fc.field,
|
|
155
|
+
operator: fc.operator,
|
|
156
|
+
value: parseConditionValue(fc.value)
|
|
157
|
+
}));
|
|
158
|
+
}
|
|
159
|
+
if (values.contextMappings.length > 0) {
|
|
160
|
+
config.contextMapping = values.contextMappings.map((cm) => ({
|
|
161
|
+
targetKey: cm.targetKey,
|
|
162
|
+
sourceExpression: cm.sourceExpression,
|
|
163
|
+
defaultValue: cm.defaultValue ? parseConditionValue(cm.defaultValue) : void 0
|
|
164
|
+
}));
|
|
165
|
+
}
|
|
166
|
+
if (values.debounceMs) {
|
|
167
|
+
config.debounceMs = parseInt(values.debounceMs, 10);
|
|
168
|
+
}
|
|
169
|
+
if (values.maxConcurrentInstances) {
|
|
170
|
+
config.maxConcurrentInstances = parseInt(values.maxConcurrentInstances, 10);
|
|
171
|
+
}
|
|
172
|
+
return {
|
|
173
|
+
name: values.name,
|
|
174
|
+
description: values.description || null,
|
|
175
|
+
eventPattern: values.eventPattern,
|
|
176
|
+
enabled: values.enabled,
|
|
177
|
+
priority: values.priority,
|
|
178
|
+
config: Object.keys(config).length > 0 ? config : null
|
|
179
|
+
};
|
|
180
|
+
}, []);
|
|
181
|
+
const parseConditionValue = (value) => {
|
|
182
|
+
try {
|
|
183
|
+
return JSON.parse(value);
|
|
184
|
+
} catch {
|
|
185
|
+
return value;
|
|
186
|
+
}
|
|
187
|
+
};
|
|
188
|
+
const handleCreateNew = useCallback(() => {
|
|
189
|
+
setEditingTrigger(null);
|
|
190
|
+
setFormValues(defaultFormValues);
|
|
191
|
+
setShowDialog(true);
|
|
192
|
+
}, []);
|
|
193
|
+
const handleEdit = useCallback((trigger) => {
|
|
194
|
+
setEditingTrigger(trigger);
|
|
195
|
+
setFormValues({
|
|
196
|
+
name: trigger.name,
|
|
197
|
+
description: trigger.description || "",
|
|
198
|
+
eventPattern: trigger.eventPattern,
|
|
199
|
+
enabled: trigger.enabled,
|
|
200
|
+
priority: trigger.priority,
|
|
201
|
+
filterConditions: trigger.config?.filterConditions?.map((fc) => ({
|
|
202
|
+
field: fc.field,
|
|
203
|
+
operator: fc.operator,
|
|
204
|
+
value: typeof fc.value === "string" ? fc.value : JSON.stringify(fc.value)
|
|
205
|
+
})) || [],
|
|
206
|
+
contextMappings: trigger.config?.contextMapping?.map((cm) => ({
|
|
207
|
+
targetKey: cm.targetKey,
|
|
208
|
+
sourceExpression: cm.sourceExpression,
|
|
209
|
+
defaultValue: cm.defaultValue !== void 0 ? typeof cm.defaultValue === "string" ? cm.defaultValue : JSON.stringify(cm.defaultValue) : ""
|
|
210
|
+
})) || [],
|
|
211
|
+
debounceMs: trigger.config?.debounceMs?.toString() || "",
|
|
212
|
+
maxConcurrentInstances: trigger.config?.maxConcurrentInstances?.toString() || ""
|
|
213
|
+
});
|
|
214
|
+
setShowDialog(true);
|
|
215
|
+
}, []);
|
|
216
|
+
const handleCloseDialog = useCallback(() => {
|
|
217
|
+
setShowDialog(false);
|
|
218
|
+
setEditingTrigger(null);
|
|
219
|
+
setFormValues(defaultFormValues);
|
|
220
|
+
}, []);
|
|
221
|
+
const handleSubmit = useCallback(() => {
|
|
222
|
+
if (!formValues.name.trim()) {
|
|
223
|
+
flash("Name is required", "error");
|
|
224
|
+
return;
|
|
225
|
+
}
|
|
226
|
+
if (!formValues.eventPattern.trim()) {
|
|
227
|
+
flash("Event pattern is required", "error");
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
if (editingTrigger) {
|
|
231
|
+
updateMutation.mutate({ id: editingTrigger.id, values: formValues });
|
|
232
|
+
} else {
|
|
233
|
+
createMutation.mutate(formValues);
|
|
234
|
+
}
|
|
235
|
+
}, [formValues, editingTrigger, createMutation, updateMutation]);
|
|
236
|
+
const addFilterCondition = useCallback(() => {
|
|
237
|
+
setFormValues((prev) => ({
|
|
238
|
+
...prev,
|
|
239
|
+
filterConditions: [...prev.filterConditions, { field: "", operator: "eq", value: "" }]
|
|
240
|
+
}));
|
|
241
|
+
}, []);
|
|
242
|
+
const removeFilterCondition = useCallback((index) => {
|
|
243
|
+
setFormValues((prev) => ({
|
|
244
|
+
...prev,
|
|
245
|
+
filterConditions: prev.filterConditions.filter((_, i) => i !== index)
|
|
246
|
+
}));
|
|
247
|
+
}, []);
|
|
248
|
+
const updateFilterCondition = useCallback((index, field, value) => {
|
|
249
|
+
setFormValues((prev) => ({
|
|
250
|
+
...prev,
|
|
251
|
+
filterConditions: prev.filterConditions.map(
|
|
252
|
+
(fc, i) => i === index ? { ...fc, [field]: value } : fc
|
|
253
|
+
)
|
|
254
|
+
}));
|
|
255
|
+
}, []);
|
|
256
|
+
const addContextMapping = useCallback(() => {
|
|
257
|
+
setFormValues((prev) => ({
|
|
258
|
+
...prev,
|
|
259
|
+
contextMappings: [...prev.contextMappings, { targetKey: "", sourceExpression: "", defaultValue: "" }]
|
|
260
|
+
}));
|
|
261
|
+
}, []);
|
|
262
|
+
const removeContextMapping = useCallback((index) => {
|
|
263
|
+
setFormValues((prev) => ({
|
|
264
|
+
...prev,
|
|
265
|
+
contextMappings: prev.contextMappings.filter((_, i) => i !== index)
|
|
266
|
+
}));
|
|
267
|
+
}, []);
|
|
268
|
+
const updateContextMapping = useCallback((index, field, value) => {
|
|
269
|
+
setFormValues((prev) => ({
|
|
270
|
+
...prev,
|
|
271
|
+
contextMappings: prev.contextMappings.map(
|
|
272
|
+
(cm, i) => i === index ? { ...cm, [field]: value } : cm
|
|
273
|
+
)
|
|
274
|
+
}));
|
|
275
|
+
}, []);
|
|
276
|
+
const isSaving = createMutation.isPending || updateMutation.isPending;
|
|
277
|
+
return /* @__PURE__ */ jsxs("div", { className, children: [
|
|
278
|
+
/* @__PURE__ */ jsxs("div", { className: "rounded-lg border bg-card p-4", children: [
|
|
279
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between mb-4", children: [
|
|
280
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
281
|
+
/* @__PURE__ */ jsx(Zap, { className: "w-5 h-5 text-amber-500" }),
|
|
282
|
+
/* @__PURE__ */ jsx("h3", { className: "text-sm font-semibold uppercase text-muted-foreground", children: "Event Triggers" })
|
|
283
|
+
] }),
|
|
284
|
+
/* @__PURE__ */ jsxs(Button, { size: "sm", variant: "outline", onClick: handleCreateNew, children: [
|
|
285
|
+
/* @__PURE__ */ jsx(Plus, { className: "w-4 h-4 mr-1" }),
|
|
286
|
+
"Add Trigger"
|
|
287
|
+
] })
|
|
288
|
+
] }),
|
|
289
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground mb-4", children: "Configure events that automatically start this workflow. When a matching event occurs in the system, a new workflow instance will be created with the mapped context." }),
|
|
290
|
+
isLoading ? /* @__PURE__ */ jsx("div", { className: "flex items-center justify-center py-8", children: /* @__PURE__ */ jsx(Spinner, { className: "w-6 h-6" }) }) : triggers.length === 0 ? /* @__PURE__ */ jsxs(Alert, { variant: "info", children: [
|
|
291
|
+
/* @__PURE__ */ jsx(AlertCircle, { className: "w-4 h-4" }),
|
|
292
|
+
/* @__PURE__ */ jsx(AlertTitle, { children: "No triggers configured" }),
|
|
293
|
+
/* @__PURE__ */ jsx(AlertDescription, { children: 'Click "Add Trigger" to create an event trigger that automatically starts this workflow.' })
|
|
294
|
+
] }) : /* @__PURE__ */ jsx("div", { className: "space-y-2", children: triggers.map((trigger) => /* @__PURE__ */ jsxs(
|
|
295
|
+
"div",
|
|
296
|
+
{
|
|
297
|
+
className: "flex items-center justify-between p-3 rounded-lg border bg-background hover:bg-accent/50 transition-colors",
|
|
298
|
+
children: [
|
|
299
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-3", children: [
|
|
300
|
+
/* @__PURE__ */ jsx(Badge, { variant: trigger.enabled ? "default" : "secondary", children: trigger.enabled ? "Active" : "Disabled" }),
|
|
301
|
+
/* @__PURE__ */ jsxs("div", { children: [
|
|
302
|
+
/* @__PURE__ */ jsx("div", { className: "font-medium text-sm", children: trigger.name }),
|
|
303
|
+
/* @__PURE__ */ jsx("code", { className: "text-xs text-muted-foreground", children: trigger.eventPattern })
|
|
304
|
+
] })
|
|
305
|
+
] }),
|
|
306
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
307
|
+
/* @__PURE__ */ jsx(Button, { size: "sm", variant: "ghost", onClick: () => handleEdit(trigger), children: /* @__PURE__ */ jsx(Edit2, { className: "w-4 h-4" }) }),
|
|
308
|
+
/* @__PURE__ */ jsx(
|
|
309
|
+
Button,
|
|
310
|
+
{
|
|
311
|
+
size: "sm",
|
|
312
|
+
variant: "ghost",
|
|
313
|
+
className: "text-destructive hover:text-destructive",
|
|
314
|
+
onClick: () => setDeleteConfirmId(trigger.id),
|
|
315
|
+
children: /* @__PURE__ */ jsx(Trash2, { className: "w-4 h-4" })
|
|
316
|
+
}
|
|
317
|
+
)
|
|
318
|
+
] })
|
|
319
|
+
]
|
|
320
|
+
},
|
|
321
|
+
trigger.id
|
|
322
|
+
)) })
|
|
323
|
+
] }),
|
|
324
|
+
/* @__PURE__ */ jsx(Dialog, { open: showDialog, onOpenChange: setShowDialog, children: /* @__PURE__ */ jsxs(DialogContent, { className: "max-w-2xl max-h-[90vh] overflow-y-auto", children: [
|
|
325
|
+
/* @__PURE__ */ jsxs(DialogHeader, { children: [
|
|
326
|
+
/* @__PURE__ */ jsx(DialogTitle, { children: editingTrigger ? "Edit Event Trigger" : "Create Event Trigger" }),
|
|
327
|
+
/* @__PURE__ */ jsx(DialogDescription, { children: "Configure when this workflow should be automatically started based on system events." })
|
|
328
|
+
] }),
|
|
329
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-4 py-4", children: [
|
|
330
|
+
/* @__PURE__ */ jsxs("div", { className: "grid grid-cols-2 gap-4", children: [
|
|
331
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
332
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-name", children: "Name *" }),
|
|
333
|
+
/* @__PURE__ */ jsx(
|
|
334
|
+
Input,
|
|
335
|
+
{
|
|
336
|
+
id: "trigger-name",
|
|
337
|
+
value: formValues.name,
|
|
338
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, name: e.target.value })),
|
|
339
|
+
placeholder: "Order Approval Trigger"
|
|
340
|
+
}
|
|
341
|
+
)
|
|
342
|
+
] }),
|
|
343
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
344
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-priority", children: "Priority" }),
|
|
345
|
+
/* @__PURE__ */ jsx(
|
|
346
|
+
Input,
|
|
347
|
+
{
|
|
348
|
+
id: "trigger-priority",
|
|
349
|
+
type: "number",
|
|
350
|
+
value: formValues.priority,
|
|
351
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, priority: parseInt(e.target.value) || 0 })),
|
|
352
|
+
placeholder: "0"
|
|
353
|
+
}
|
|
354
|
+
),
|
|
355
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: "Higher priority triggers execute first" })
|
|
356
|
+
] })
|
|
357
|
+
] }),
|
|
358
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
359
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-description", children: "Description" }),
|
|
360
|
+
/* @__PURE__ */ jsx(
|
|
361
|
+
Textarea,
|
|
362
|
+
{
|
|
363
|
+
id: "trigger-description",
|
|
364
|
+
value: formValues.description,
|
|
365
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, description: e.target.value })),
|
|
366
|
+
placeholder: "Describe when this trigger should fire...",
|
|
367
|
+
rows: 2
|
|
368
|
+
}
|
|
369
|
+
)
|
|
370
|
+
] }),
|
|
371
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
372
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-pattern", children: "Event Pattern *" }),
|
|
373
|
+
/* @__PURE__ */ jsxs("div", { className: "flex gap-2", children: [
|
|
374
|
+
/* @__PURE__ */ jsx(
|
|
375
|
+
Input,
|
|
376
|
+
{
|
|
377
|
+
id: "trigger-pattern",
|
|
378
|
+
value: formValues.eventPattern,
|
|
379
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, eventPattern: e.target.value })),
|
|
380
|
+
placeholder: "sales.orders.updated",
|
|
381
|
+
className: "flex-1"
|
|
382
|
+
}
|
|
383
|
+
),
|
|
384
|
+
/* @__PURE__ */ jsx(
|
|
385
|
+
EventSelect,
|
|
386
|
+
{
|
|
387
|
+
value: "",
|
|
388
|
+
onChange: (eventId) => setFormValues((prev) => ({ ...prev, eventPattern: eventId })),
|
|
389
|
+
placeholder: "Quick select...",
|
|
390
|
+
className: "w-[200px]"
|
|
391
|
+
}
|
|
392
|
+
)
|
|
393
|
+
] }),
|
|
394
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: 'Use * as wildcard: "sales.orders.*" matches any order event' })
|
|
395
|
+
] }),
|
|
396
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
397
|
+
/* @__PURE__ */ jsx(
|
|
398
|
+
Switch,
|
|
399
|
+
{
|
|
400
|
+
id: "trigger-enabled",
|
|
401
|
+
checked: formValues.enabled,
|
|
402
|
+
onCheckedChange: (checked) => setFormValues((prev) => ({ ...prev, enabled: checked }))
|
|
403
|
+
}
|
|
404
|
+
),
|
|
405
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-enabled", children: "Enabled" })
|
|
406
|
+
] }),
|
|
407
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
|
|
408
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between", children: [
|
|
409
|
+
/* @__PURE__ */ jsx(Label, { children: "Filter Conditions" }),
|
|
410
|
+
/* @__PURE__ */ jsxs(Button, { size: "sm", variant: "ghost", onClick: addFilterCondition, children: [
|
|
411
|
+
/* @__PURE__ */ jsx(Plus, { className: "w-4 h-4 mr-1" }),
|
|
412
|
+
"Add Condition"
|
|
413
|
+
] })
|
|
414
|
+
] }),
|
|
415
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: "Only trigger when the event payload matches these conditions (all must match)" }),
|
|
416
|
+
formValues.filterConditions.map((fc, index) => /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
417
|
+
/* @__PURE__ */ jsx(
|
|
418
|
+
Input,
|
|
419
|
+
{
|
|
420
|
+
value: fc.field,
|
|
421
|
+
onChange: (e) => updateFilterCondition(index, "field", e.target.value),
|
|
422
|
+
placeholder: "status",
|
|
423
|
+
className: "w-1/3"
|
|
424
|
+
}
|
|
425
|
+
),
|
|
426
|
+
/* @__PURE__ */ jsx(
|
|
427
|
+
"select",
|
|
428
|
+
{
|
|
429
|
+
value: fc.operator,
|
|
430
|
+
onChange: (e) => updateFilterCondition(index, "operator", e.target.value),
|
|
431
|
+
className: "h-10 w-[140px] rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
|
|
432
|
+
children: FILTER_OPERATORS.map((op) => /* @__PURE__ */ jsx("option", { value: op.value, children: op.label }, op.value))
|
|
433
|
+
}
|
|
434
|
+
),
|
|
435
|
+
/* @__PURE__ */ jsx(
|
|
436
|
+
Input,
|
|
437
|
+
{
|
|
438
|
+
value: fc.value,
|
|
439
|
+
onChange: (e) => updateFilterCondition(index, "value", e.target.value),
|
|
440
|
+
placeholder: "submitted",
|
|
441
|
+
className: "flex-1"
|
|
442
|
+
}
|
|
443
|
+
),
|
|
444
|
+
/* @__PURE__ */ jsx(Button, { size: "icon", variant: "ghost", onClick: () => removeFilterCondition(index), children: /* @__PURE__ */ jsx(X, { className: "w-4 h-4" }) })
|
|
445
|
+
] }, index))
|
|
446
|
+
] }),
|
|
447
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
|
|
448
|
+
/* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between", children: [
|
|
449
|
+
/* @__PURE__ */ jsx(Label, { children: "Context Mapping" }),
|
|
450
|
+
/* @__PURE__ */ jsxs(Button, { size: "sm", variant: "ghost", onClick: addContextMapping, children: [
|
|
451
|
+
/* @__PURE__ */ jsx(Plus, { className: "w-4 h-4 mr-1" }),
|
|
452
|
+
"Add Mapping"
|
|
453
|
+
] })
|
|
454
|
+
] }),
|
|
455
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: "Map values from the event payload to the workflow's initial context" }),
|
|
456
|
+
formValues.contextMappings.map((cm, index) => /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
|
|
457
|
+
/* @__PURE__ */ jsx(
|
|
458
|
+
Input,
|
|
459
|
+
{
|
|
460
|
+
value: cm.targetKey,
|
|
461
|
+
onChange: (e) => updateContextMapping(index, "targetKey", e.target.value),
|
|
462
|
+
placeholder: "orderId",
|
|
463
|
+
className: "w-1/3"
|
|
464
|
+
}
|
|
465
|
+
),
|
|
466
|
+
/* @__PURE__ */ jsx("span", { className: "text-muted-foreground", children: "=" }),
|
|
467
|
+
/* @__PURE__ */ jsx(
|
|
468
|
+
Input,
|
|
469
|
+
{
|
|
470
|
+
value: cm.sourceExpression,
|
|
471
|
+
onChange: (e) => updateContextMapping(index, "sourceExpression", e.target.value),
|
|
472
|
+
placeholder: "id",
|
|
473
|
+
className: "flex-1"
|
|
474
|
+
}
|
|
475
|
+
),
|
|
476
|
+
/* @__PURE__ */ jsx(
|
|
477
|
+
Input,
|
|
478
|
+
{
|
|
479
|
+
value: cm.defaultValue,
|
|
480
|
+
onChange: (e) => updateContextMapping(index, "defaultValue", e.target.value),
|
|
481
|
+
placeholder: "default",
|
|
482
|
+
className: "w-24"
|
|
483
|
+
}
|
|
484
|
+
),
|
|
485
|
+
/* @__PURE__ */ jsx(Button, { size: "icon", variant: "ghost", onClick: () => removeContextMapping(index), children: /* @__PURE__ */ jsx(X, { className: "w-4 h-4" }) })
|
|
486
|
+
] }, index))
|
|
487
|
+
] }),
|
|
488
|
+
/* @__PURE__ */ jsxs("div", { className: "grid grid-cols-2 gap-4", children: [
|
|
489
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
490
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-debounce", children: "Debounce (ms)" }),
|
|
491
|
+
/* @__PURE__ */ jsx(
|
|
492
|
+
Input,
|
|
493
|
+
{
|
|
494
|
+
id: "trigger-debounce",
|
|
495
|
+
type: "number",
|
|
496
|
+
value: formValues.debounceMs,
|
|
497
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, debounceMs: e.target.value })),
|
|
498
|
+
placeholder: "0"
|
|
499
|
+
}
|
|
500
|
+
),
|
|
501
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: "Delay to prevent rapid re-triggers" })
|
|
502
|
+
] }),
|
|
503
|
+
/* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
|
|
504
|
+
/* @__PURE__ */ jsx(Label, { htmlFor: "trigger-max-concurrent", children: "Max Concurrent Instances" }),
|
|
505
|
+
/* @__PURE__ */ jsx(
|
|
506
|
+
Input,
|
|
507
|
+
{
|
|
508
|
+
id: "trigger-max-concurrent",
|
|
509
|
+
type: "number",
|
|
510
|
+
value: formValues.maxConcurrentInstances,
|
|
511
|
+
onChange: (e) => setFormValues((prev) => ({ ...prev, maxConcurrentInstances: e.target.value })),
|
|
512
|
+
placeholder: "Unlimited"
|
|
513
|
+
}
|
|
514
|
+
),
|
|
515
|
+
/* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: "Limit simultaneous workflow instances" })
|
|
516
|
+
] })
|
|
517
|
+
] })
|
|
518
|
+
] }),
|
|
519
|
+
/* @__PURE__ */ jsxs(DialogFooter, { children: [
|
|
520
|
+
/* @__PURE__ */ jsx(Button, { variant: "outline", onClick: handleCloseDialog, disabled: isSaving, children: "Cancel" }),
|
|
521
|
+
/* @__PURE__ */ jsxs(Button, { onClick: handleSubmit, disabled: isSaving, children: [
|
|
522
|
+
isSaving ? /* @__PURE__ */ jsx(Spinner, { className: "w-4 h-4 mr-2" }) : null,
|
|
523
|
+
editingTrigger ? "Update" : "Create"
|
|
524
|
+
] })
|
|
525
|
+
] })
|
|
526
|
+
] }) }),
|
|
527
|
+
/* @__PURE__ */ jsx(Dialog, { open: !!deleteConfirmId, onOpenChange: () => setDeleteConfirmId(null), children: /* @__PURE__ */ jsxs(DialogContent, { children: [
|
|
528
|
+
/* @__PURE__ */ jsxs(DialogHeader, { children: [
|
|
529
|
+
/* @__PURE__ */ jsx(DialogTitle, { children: "Delete Event Trigger?" }),
|
|
530
|
+
/* @__PURE__ */ jsx(DialogDescription, { children: "This will permanently delete the event trigger. Workflows will no longer be automatically started by this trigger." })
|
|
531
|
+
] }),
|
|
532
|
+
/* @__PURE__ */ jsxs(DialogFooter, { children: [
|
|
533
|
+
/* @__PURE__ */ jsx(Button, { variant: "outline", onClick: () => setDeleteConfirmId(null), children: "Cancel" }),
|
|
534
|
+
/* @__PURE__ */ jsxs(
|
|
535
|
+
Button,
|
|
536
|
+
{
|
|
537
|
+
variant: "destructive",
|
|
538
|
+
onClick: () => deleteConfirmId && deleteMutation.mutate(deleteConfirmId),
|
|
539
|
+
disabled: deleteMutation.isPending,
|
|
540
|
+
children: [
|
|
541
|
+
deleteMutation.isPending ? /* @__PURE__ */ jsx(Spinner, { className: "w-4 h-4 mr-2" }) : null,
|
|
542
|
+
"Delete"
|
|
543
|
+
]
|
|
544
|
+
}
|
|
545
|
+
)
|
|
546
|
+
] })
|
|
547
|
+
] }) })
|
|
548
|
+
] });
|
|
549
|
+
}
|
|
550
|
+
export {
|
|
551
|
+
EventTriggersEditor
|
|
552
|
+
};
|
|
553
|
+
//# sourceMappingURL=EventTriggersEditor.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../src/modules/workflows/components/EventTriggersEditor.tsx"],
|
|
4
|
+
"sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport { useState, useCallback } from 'react'\nimport { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { Textarea } from '@open-mercato/ui/primitives/textarea'\nimport { Label } from '@open-mercato/ui/primitives/label'\nimport { Switch } from '@open-mercato/ui/primitives/switch'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport {\n Dialog,\n DialogContent,\n DialogDescription,\n DialogFooter,\n DialogHeader,\n DialogTitle,\n} from '@open-mercato/ui/primitives/dialog'\nimport { Alert, AlertDescription, AlertTitle } from '@open-mercato/ui/primitives/alert'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { EventSelect } from '@open-mercato/ui/backend/inputs/EventSelect'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { Plus, Trash2, Edit2, Zap, AlertCircle, X } from 'lucide-react'\n\ninterface EventTrigger {\n id: string\n name: string\n description?: string | null\n eventPattern: string\n config?: {\n filterConditions?: Array<{\n field: string\n operator: string\n value: unknown\n }>\n contextMapping?: Array<{\n targetKey: string\n sourceExpression: string\n defaultValue?: unknown\n }>\n debounceMs?: number\n maxConcurrentInstances?: number\n } | null\n enabled: boolean\n priority: number\n workflowDefinitionId: string\n createdAt: string\n updatedAt: string\n}\n\ninterface EventTriggersEditorProps {\n workflowDefinitionId: string\n workflowId?: string\n className?: string\n}\n\nconst FILTER_OPERATORS = [\n { value: 'eq', label: 'Equals' },\n { value: 'neq', label: 'Not Equals' },\n { value: 'gt', label: 'Greater Than' },\n { value: 'gte', label: 'Greater Than or Equal' },\n { value: 'lt', label: 'Less Than' },\n { value: 'lte', label: 'Less Than or Equal' },\n { value: 'contains', label: 'Contains' },\n { value: 'startsWith', label: 'Starts With' },\n { value: 'endsWith', label: 'Ends With' },\n { value: 'in', label: 'In (array)' },\n { value: 'notIn', label: 'Not In (array)' },\n { value: 'exists', label: 'Exists' },\n { value: 'notExists', label: 'Not Exists' },\n { value: 'regex', label: 'Regex Match' },\n]\n\ntype TriggerFormValues = {\n name: string\n description: string\n eventPattern: string\n enabled: boolean\n priority: number\n filterConditions: Array<{ field: string; operator: string; value: string }>\n contextMappings: Array<{ targetKey: string; sourceExpression: string; defaultValue: string }>\n debounceMs: string\n maxConcurrentInstances: string\n}\n\nconst defaultFormValues: TriggerFormValues = {\n name: '',\n description: '',\n eventPattern: '',\n enabled: true,\n priority: 0,\n filterConditions: [],\n contextMappings: [],\n debounceMs: '',\n maxConcurrentInstances: '',\n}\n\nexport function EventTriggersEditor({\n workflowDefinitionId,\n workflowId,\n className,\n}: EventTriggersEditorProps) {\n const t = useT()\n const queryClient = useQueryClient()\n const [showDialog, setShowDialog] = useState(false)\n const [editingTrigger, setEditingTrigger] = useState<EventTrigger | null>(null)\n const [formValues, setFormValues] = useState<TriggerFormValues>(defaultFormValues)\n const [deleteConfirmId, setDeleteConfirmId] = useState<string | null>(null)\n\n // Fetch triggers for this workflow definition\n const { data: triggersData, isLoading } = useQuery({\n queryKey: ['workflow-triggers', workflowDefinitionId],\n queryFn: async () => {\n const result = await apiCall<{ data: EventTrigger[]; pagination: any }>(\n `/api/workflows/triggers?workflowDefinitionId=${workflowDefinitionId}&limit=100`\n )\n if (!result.ok) {\n throw new Error('Failed to load triggers')\n }\n return result.result?.data || []\n },\n enabled: !!workflowDefinitionId,\n })\n\n const triggers = triggersData || []\n\n // Create trigger mutation\n const createMutation = useMutation({\n mutationFn: async (values: TriggerFormValues) => {\n const payload = buildTriggerPayload(values)\n const result = await apiCall<{ data: EventTrigger; error?: string }>(\n '/api/workflows/triggers',\n {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n ...payload,\n workflowDefinitionId,\n }),\n }\n )\n if (!result.ok) {\n throw new Error(result.result?.error || 'Failed to create trigger')\n }\n return result.result?.data\n },\n onSuccess: () => {\n queryClient.invalidateQueries({ queryKey: ['workflow-triggers', workflowDefinitionId] })\n flash('Event trigger created successfully', 'success')\n handleCloseDialog()\n },\n onError: (error: Error) => {\n flash(error.message, 'error')\n },\n })\n\n // Update trigger mutation\n const updateMutation = useMutation({\n mutationFn: async ({ id, values }: { id: string; values: TriggerFormValues }) => {\n const payload = buildTriggerPayload(values)\n const result = await apiCall<{ data: EventTrigger; error?: string }>(\n `/api/workflows/triggers/${id}`,\n {\n method: 'PUT',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(payload),\n }\n )\n if (!result.ok) {\n throw new Error(result.result?.error || 'Failed to update trigger')\n }\n return result.result?.data\n },\n onSuccess: () => {\n queryClient.invalidateQueries({ queryKey: ['workflow-triggers', workflowDefinitionId] })\n flash('Event trigger updated successfully', 'success')\n handleCloseDialog()\n },\n onError: (error: Error) => {\n flash(error.message, 'error')\n },\n })\n\n // Delete trigger mutation\n const deleteMutation = useMutation({\n mutationFn: async (id: string) => {\n const result = await apiCall<{ error?: string }>(\n `/api/workflows/triggers/${id}`,\n { method: 'DELETE' }\n )\n if (!result.ok) {\n throw new Error(result.result?.error || 'Failed to delete trigger')\n }\n },\n onSuccess: () => {\n queryClient.invalidateQueries({ queryKey: ['workflow-triggers', workflowDefinitionId] })\n flash('Event trigger deleted successfully', 'success')\n setDeleteConfirmId(null)\n },\n onError: (error: Error) => {\n flash(error.message, 'error')\n },\n })\n\n // Build API payload from form values\n const buildTriggerPayload = useCallback((values: TriggerFormValues) => {\n const config: EventTrigger['config'] = {}\n\n if (values.filterConditions.length > 0) {\n config.filterConditions = values.filterConditions.map(fc => ({\n field: fc.field,\n operator: fc.operator,\n value: parseConditionValue(fc.value),\n }))\n }\n\n if (values.contextMappings.length > 0) {\n config.contextMapping = values.contextMappings.map(cm => ({\n targetKey: cm.targetKey,\n sourceExpression: cm.sourceExpression,\n defaultValue: cm.defaultValue ? parseConditionValue(cm.defaultValue) : undefined,\n }))\n }\n\n if (values.debounceMs) {\n config.debounceMs = parseInt(values.debounceMs, 10)\n }\n\n if (values.maxConcurrentInstances) {\n config.maxConcurrentInstances = parseInt(values.maxConcurrentInstances, 10)\n }\n\n return {\n name: values.name,\n description: values.description || null,\n eventPattern: values.eventPattern,\n enabled: values.enabled,\n priority: values.priority,\n config: Object.keys(config).length > 0 ? config : null,\n }\n }, [])\n\n // Parse condition value (try JSON, fallback to string)\n const parseConditionValue = (value: string): unknown => {\n try {\n return JSON.parse(value)\n } catch {\n return value\n }\n }\n\n // Open dialog for creating new trigger\n const handleCreateNew = useCallback(() => {\n setEditingTrigger(null)\n setFormValues(defaultFormValues)\n setShowDialog(true)\n }, [])\n\n // Open dialog for editing trigger\n const handleEdit = useCallback((trigger: EventTrigger) => {\n setEditingTrigger(trigger)\n setFormValues({\n name: trigger.name,\n description: trigger.description || '',\n eventPattern: trigger.eventPattern,\n enabled: trigger.enabled,\n priority: trigger.priority,\n filterConditions: trigger.config?.filterConditions?.map(fc => ({\n field: fc.field,\n operator: fc.operator,\n value: typeof fc.value === 'string' ? fc.value : JSON.stringify(fc.value),\n })) || [],\n contextMappings: trigger.config?.contextMapping?.map(cm => ({\n targetKey: cm.targetKey,\n sourceExpression: cm.sourceExpression,\n defaultValue: cm.defaultValue !== undefined\n ? (typeof cm.defaultValue === 'string' ? cm.defaultValue : JSON.stringify(cm.defaultValue))\n : '',\n })) || [],\n debounceMs: trigger.config?.debounceMs?.toString() || '',\n maxConcurrentInstances: trigger.config?.maxConcurrentInstances?.toString() || '',\n })\n setShowDialog(true)\n }, [])\n\n // Close dialog\n const handleCloseDialog = useCallback(() => {\n setShowDialog(false)\n setEditingTrigger(null)\n setFormValues(defaultFormValues)\n }, [])\n\n // Submit form\n const handleSubmit = useCallback(() => {\n if (!formValues.name.trim()) {\n flash('Name is required', 'error')\n return\n }\n if (!formValues.eventPattern.trim()) {\n flash('Event pattern is required', 'error')\n return\n }\n\n if (editingTrigger) {\n updateMutation.mutate({ id: editingTrigger.id, values: formValues })\n } else {\n createMutation.mutate(formValues)\n }\n }, [formValues, editingTrigger, createMutation, updateMutation])\n\n // Add filter condition\n const addFilterCondition = useCallback(() => {\n setFormValues(prev => ({\n ...prev,\n filterConditions: [...prev.filterConditions, { field: '', operator: 'eq', value: '' }],\n }))\n }, [])\n\n // Remove filter condition\n const removeFilterCondition = useCallback((index: number) => {\n setFormValues(prev => ({\n ...prev,\n filterConditions: prev.filterConditions.filter((_, i) => i !== index),\n }))\n }, [])\n\n // Update filter condition\n const updateFilterCondition = useCallback((index: number, field: string, value: string) => {\n setFormValues(prev => ({\n ...prev,\n filterConditions: prev.filterConditions.map((fc, i) =>\n i === index ? { ...fc, [field]: value } : fc\n ),\n }))\n }, [])\n\n // Add context mapping\n const addContextMapping = useCallback(() => {\n setFormValues(prev => ({\n ...prev,\n contextMappings: [...prev.contextMappings, { targetKey: '', sourceExpression: '', defaultValue: '' }],\n }))\n }, [])\n\n // Remove context mapping\n const removeContextMapping = useCallback((index: number) => {\n setFormValues(prev => ({\n ...prev,\n contextMappings: prev.contextMappings.filter((_, i) => i !== index),\n }))\n }, [])\n\n // Update context mapping\n const updateContextMapping = useCallback((index: number, field: string, value: string) => {\n setFormValues(prev => ({\n ...prev,\n contextMappings: prev.contextMappings.map((cm, i) =>\n i === index ? { ...cm, [field]: value } : cm\n ),\n }))\n }, [])\n\n const isSaving = createMutation.isPending || updateMutation.isPending\n\n return (\n <div className={className}>\n <div className=\"rounded-lg border bg-card p-4\">\n <div className=\"flex items-center justify-between mb-4\">\n <div className=\"flex items-center gap-2\">\n <Zap className=\"w-5 h-5 text-amber-500\" />\n <h3 className=\"text-sm font-semibold uppercase text-muted-foreground\">Event Triggers</h3>\n </div>\n <Button size=\"sm\" variant=\"outline\" onClick={handleCreateNew}>\n <Plus className=\"w-4 h-4 mr-1\" />\n Add Trigger\n </Button>\n </div>\n\n <p className=\"text-xs text-muted-foreground mb-4\">\n Configure events that automatically start this workflow. When a matching event occurs in the system,\n a new workflow instance will be created with the mapped context.\n </p>\n\n {isLoading ? (\n <div className=\"flex items-center justify-center py-8\">\n <Spinner className=\"w-6 h-6\" />\n </div>\n ) : triggers.length === 0 ? (\n <Alert variant=\"info\">\n <AlertCircle className=\"w-4 h-4\" />\n <AlertTitle>No triggers configured</AlertTitle>\n <AlertDescription>\n Click \"Add Trigger\" to create an event trigger that automatically starts this workflow.\n </AlertDescription>\n </Alert>\n ) : (\n <div className=\"space-y-2\">\n {triggers.map(trigger => (\n <div\n key={trigger.id}\n className=\"flex items-center justify-between p-3 rounded-lg border bg-background hover:bg-accent/50 transition-colors\"\n >\n <div className=\"flex items-center gap-3\">\n <Badge variant={trigger.enabled ? 'default' : 'secondary'}>\n {trigger.enabled ? 'Active' : 'Disabled'}\n </Badge>\n <div>\n <div className=\"font-medium text-sm\">{trigger.name}</div>\n <code className=\"text-xs text-muted-foreground\">{trigger.eventPattern}</code>\n </div>\n </div>\n <div className=\"flex items-center gap-2\">\n <Button size=\"sm\" variant=\"ghost\" onClick={() => handleEdit(trigger)}>\n <Edit2 className=\"w-4 h-4\" />\n </Button>\n <Button\n size=\"sm\"\n variant=\"ghost\"\n className=\"text-destructive hover:text-destructive\"\n onClick={() => setDeleteConfirmId(trigger.id)}\n >\n <Trash2 className=\"w-4 h-4\" />\n </Button>\n </div>\n </div>\n ))}\n </div>\n )}\n </div>\n\n {/* Create/Edit Dialog */}\n <Dialog open={showDialog} onOpenChange={setShowDialog}>\n <DialogContent className=\"max-w-2xl max-h-[90vh] overflow-y-auto\">\n <DialogHeader>\n <DialogTitle>\n {editingTrigger ? 'Edit Event Trigger' : 'Create Event Trigger'}\n </DialogTitle>\n <DialogDescription>\n Configure when this workflow should be automatically started based on system events.\n </DialogDescription>\n </DialogHeader>\n\n <div className=\"space-y-4 py-4\">\n {/* Basic Info */}\n <div className=\"grid grid-cols-2 gap-4\">\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-name\">Name *</Label>\n <Input\n id=\"trigger-name\"\n value={formValues.name}\n onChange={e => setFormValues(prev => ({ ...prev, name: e.target.value }))}\n placeholder=\"Order Approval Trigger\"\n />\n </div>\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-priority\">Priority</Label>\n <Input\n id=\"trigger-priority\"\n type=\"number\"\n value={formValues.priority}\n onChange={e => setFormValues(prev => ({ ...prev, priority: parseInt(e.target.value) || 0 }))}\n placeholder=\"0\"\n />\n <p className=\"text-xs text-muted-foreground\">Higher priority triggers execute first</p>\n </div>\n </div>\n\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-description\">Description</Label>\n <Textarea\n id=\"trigger-description\"\n value={formValues.description}\n onChange={e => setFormValues(prev => ({ ...prev, description: e.target.value }))}\n placeholder=\"Describe when this trigger should fire...\"\n rows={2}\n />\n </div>\n\n {/* Event Pattern */}\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-pattern\">Event Pattern *</Label>\n <div className=\"flex gap-2\">\n <Input\n id=\"trigger-pattern\"\n value={formValues.eventPattern}\n onChange={e => setFormValues(prev => ({ ...prev, eventPattern: e.target.value }))}\n placeholder=\"sales.orders.updated\"\n className=\"flex-1\"\n />\n <EventSelect\n value=\"\"\n onChange={(eventId) => setFormValues(prev => ({ ...prev, eventPattern: eventId }))}\n placeholder=\"Quick select...\"\n className=\"w-[200px]\"\n />\n </div>\n <p className=\"text-xs text-muted-foreground\">\n Use * as wildcard: \"sales.orders.*\" matches any order event\n </p>\n </div>\n\n {/* Enabled Switch */}\n <div className=\"flex items-center gap-2\">\n <Switch\n id=\"trigger-enabled\"\n checked={formValues.enabled}\n onCheckedChange={checked => setFormValues(prev => ({ ...prev, enabled: checked }))}\n />\n <Label htmlFor=\"trigger-enabled\">Enabled</Label>\n </div>\n\n {/* Filter Conditions */}\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between\">\n <Label>Filter Conditions</Label>\n <Button size=\"sm\" variant=\"ghost\" onClick={addFilterCondition}>\n <Plus className=\"w-4 h-4 mr-1\" />\n Add Condition\n </Button>\n </div>\n <p className=\"text-xs text-muted-foreground\">\n Only trigger when the event payload matches these conditions (all must match)\n </p>\n {formValues.filterConditions.map((fc, index) => (\n <div key={index} className=\"flex items-center gap-2\">\n <Input\n value={fc.field}\n onChange={e => updateFilterCondition(index, 'field', e.target.value)}\n placeholder=\"status\"\n className=\"w-1/3\"\n />\n <select\n value={fc.operator}\n onChange={(e: React.ChangeEvent<HTMLSelectElement>) => updateFilterCondition(index, 'operator', e.target.value)}\n className=\"h-10 w-[140px] rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2\"\n >\n {FILTER_OPERATORS.map(op => (\n <option key={op.value} value={op.value}>\n {op.label}\n </option>\n ))}\n </select>\n <Input\n value={fc.value}\n onChange={e => updateFilterCondition(index, 'value', e.target.value)}\n placeholder=\"submitted\"\n className=\"flex-1\"\n />\n <Button size=\"icon\" variant=\"ghost\" onClick={() => removeFilterCondition(index)}>\n <X className=\"w-4 h-4\" />\n </Button>\n </div>\n ))}\n </div>\n\n {/* Context Mapping */}\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between\">\n <Label>Context Mapping</Label>\n <Button size=\"sm\" variant=\"ghost\" onClick={addContextMapping}>\n <Plus className=\"w-4 h-4 mr-1\" />\n Add Mapping\n </Button>\n </div>\n <p className=\"text-xs text-muted-foreground\">\n Map values from the event payload to the workflow's initial context\n </p>\n {formValues.contextMappings.map((cm, index) => (\n <div key={index} className=\"flex items-center gap-2\">\n <Input\n value={cm.targetKey}\n onChange={e => updateContextMapping(index, 'targetKey', e.target.value)}\n placeholder=\"orderId\"\n className=\"w-1/3\"\n />\n <span className=\"text-muted-foreground\">=</span>\n <Input\n value={cm.sourceExpression}\n onChange={e => updateContextMapping(index, 'sourceExpression', e.target.value)}\n placeholder=\"id\"\n className=\"flex-1\"\n />\n <Input\n value={cm.defaultValue}\n onChange={e => updateContextMapping(index, 'defaultValue', e.target.value)}\n placeholder=\"default\"\n className=\"w-24\"\n />\n <Button size=\"icon\" variant=\"ghost\" onClick={() => removeContextMapping(index)}>\n <X className=\"w-4 h-4\" />\n </Button>\n </div>\n ))}\n </div>\n\n {/* Advanced Options */}\n <div className=\"grid grid-cols-2 gap-4\">\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-debounce\">Debounce (ms)</Label>\n <Input\n id=\"trigger-debounce\"\n type=\"number\"\n value={formValues.debounceMs}\n onChange={e => setFormValues(prev => ({ ...prev, debounceMs: e.target.value }))}\n placeholder=\"0\"\n />\n <p className=\"text-xs text-muted-foreground\">Delay to prevent rapid re-triggers</p>\n </div>\n <div className=\"space-y-1\">\n <Label htmlFor=\"trigger-max-concurrent\">Max Concurrent Instances</Label>\n <Input\n id=\"trigger-max-concurrent\"\n type=\"number\"\n value={formValues.maxConcurrentInstances}\n onChange={e => setFormValues(prev => ({ ...prev, maxConcurrentInstances: e.target.value }))}\n placeholder=\"Unlimited\"\n />\n <p className=\"text-xs text-muted-foreground\">Limit simultaneous workflow instances</p>\n </div>\n </div>\n </div>\n\n <DialogFooter>\n <Button variant=\"outline\" onClick={handleCloseDialog} disabled={isSaving}>\n Cancel\n </Button>\n <Button onClick={handleSubmit} disabled={isSaving}>\n {isSaving ? <Spinner className=\"w-4 h-4 mr-2\" /> : null}\n {editingTrigger ? 'Update' : 'Create'}\n </Button>\n </DialogFooter>\n </DialogContent>\n </Dialog>\n\n {/* Delete Confirmation Dialog */}\n <Dialog open={!!deleteConfirmId} onOpenChange={() => setDeleteConfirmId(null)}>\n <DialogContent>\n <DialogHeader>\n <DialogTitle>Delete Event Trigger?</DialogTitle>\n <DialogDescription>\n This will permanently delete the event trigger. Workflows will no longer be automatically started by this trigger.\n </DialogDescription>\n </DialogHeader>\n <DialogFooter>\n <Button variant=\"outline\" onClick={() => setDeleteConfirmId(null)}>\n Cancel\n </Button>\n <Button\n variant=\"destructive\"\n onClick={() => deleteConfirmId && deleteMutation.mutate(deleteConfirmId)}\n disabled={deleteMutation.isPending}\n >\n {deleteMutation.isPending ? <Spinner className=\"w-4 h-4 mr-2\" /> : null}\n Delete\n </Button>\n </DialogFooter>\n </DialogContent>\n </Dialog>\n </div>\n )\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|