@open-mercato/core 0.4.8-develop-bc5be31b5c → 0.4.8-develop-409bb4a065

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.
@@ -236,7 +236,6 @@ function AttachmentFilesField({
236
236
  value,
237
237
  setValue,
238
238
  disabled,
239
- error,
240
239
  labels,
241
240
  uploading
242
241
  }) {
@@ -345,8 +344,7 @@ function AttachmentFilesField({
345
344
  ]
346
345
  }
347
346
  ),
348
- renderFileList(),
349
- error ? /* @__PURE__ */ jsx("p", { className: "text-xs font-medium text-red-600", children: error }) : null
347
+ renderFileList()
350
348
  ] });
351
349
  }
352
350
  function AttachmentUploadForm({ partitions, availableTags, onUploaded, onCancel }) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/attachments/components/AttachmentLibrary.tsx"],
4
- "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Dialog, DialogContent, DialogHeader, DialogTitle } from '@open-mercato/ui/primitives/dialog'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { TagsInput } from '@open-mercato/ui/backend/inputs/TagsInput'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { CrudForm, type CrudField, type CrudFormGroup, type CrudCustomFieldRenderProps } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { z } from 'zod'\nimport { E } from '#generated/entities.ids.generated'\nimport type { LucideIcon } from 'lucide-react'\nimport { Download, Plus, Upload, Trash2, File, FileText, FileSpreadsheet, FileArchive, FileAudio, FileVideo, FileCode } from 'lucide-react'\nimport { buildAttachmentFileUrl, buildAttachmentImageUrl, slugifyAttachmentFileName } from '@open-mercato/core/modules/attachments/lib/imageUrls'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { AttachmentDeleteDialog, AttachmentMetadataDialog, type AttachmentItem, type AttachmentMetadataSavePayload, type AssignmentDraft } from '@open-mercato/ui/backend/detail'\n\ntype AttachmentAssignment = {\n type: string\n id: string\n href?: string | null\n label?: string | null\n}\n\ntype AttachmentRow = AttachmentItem\n\ntype AttachmentLibraryResponse = {\n items: AttachmentRow[]\n page: number\n pageSize: number\n total: number\n totalPages: number\n availableTags: string[]\n partitions: Array<{ code: string; title: string; description?: string | null; isPublic?: boolean }>\n error?: string\n}\n\nconst PAGE_SIZE = 25\nconst ENV_APP_URL = (process.env.NEXT_PUBLIC_APP_URL || '').replace(/\\/$/, '')\nconst LIBRARY_ENTITY_ID = 'attachments:library'\n\nfunction filterLibraryAssignments(assignments?: AttachmentAssignment[] | null): AttachmentAssignment[] {\n return (assignments ?? []).filter((assignment) => assignment.type !== LIBRARY_ENTITY_ID)\n}\n\nfunction formatFileSize(value: number): string {\n if (!Number.isFinite(value)) return '\u2014'\n if (value <= 0) return '0 B'\n const units = ['B', 'KB', 'MB', 'GB', 'TB']\n let idx = 0\n let current = value\n while (current >= 1024 && idx < units.length - 1) {\n current /= 1024\n idx += 1\n }\n return `${current.toFixed(idx === 0 ? 0 : 1)} ${units[idx]}`\n}\n\nfunction humanDate(value: string, locale?: string): string {\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return value\n return date.toLocaleString(locale ?? undefined)\n}\n\nfunction buildFilterSignature(values: FilterValues): string {\n return JSON.stringify(values, Object.keys(values).sort((a, b) => a.localeCompare(b)))\n}\n\nfunction resolveAbsoluteUrl(path: string): string {\n if (!path) return path\n if (/^https?:\\/\\//i.test(path)) return path\n const base =\n ENV_APP_URL ||\n (typeof window !== 'undefined' && window.location?.origin ? window.location.origin : '')\n if (!base) return path\n const normalizedBase = base.replace(/\\/$/, '')\n return `${normalizedBase}${path.startsWith('/') ? path : `/${path}`}`\n}\n\nfunction resolveFileExtension(fileName?: string | null): string {\n if (!fileName) return ''\n const normalized = fileName.trim()\n if (!normalized) return ''\n const lastDot = normalized.lastIndexOf('.')\n if (lastDot === -1 || lastDot === normalized.length - 1) return ''\n return normalized.slice(lastDot + 1).toLowerCase()\n}\n\nconst EXTENSION_ICON_MAP: Record<string, LucideIcon> = {\n pdf: FileText,\n doc: FileText,\n docx: FileText,\n txt: FileText,\n md: FileText,\n rtf: FileText,\n xls: FileSpreadsheet,\n xlsx: FileSpreadsheet,\n csv: FileSpreadsheet,\n ods: FileSpreadsheet,\n ppt: FileText,\n pptx: FileText,\n zip: FileArchive,\n gz: FileArchive,\n rar: FileArchive,\n tgz: FileArchive,\n '7z': FileArchive,\n tar: FileArchive,\n json: FileCode,\n js: FileCode,\n ts: FileCode,\n jsx: FileCode,\n tsx: FileCode,\n html: FileCode,\n css: FileCode,\n xml: FileCode,\n yaml: FileCode,\n yml: FileCode,\n mp3: FileAudio,\n wav: FileAudio,\n flac: FileAudio,\n ogg: FileAudio,\n mp4: FileVideo,\n mov: FileVideo,\n avi: FileVideo,\n webm: FileVideo,\n}\n\nconst MIME_FALLBACK_ICONS: Record<string, LucideIcon> = {\n audio: FileAudio,\n video: FileVideo,\n text: FileText,\n application: FileText,\n}\n\nfunction resolveAttachmentPlaceholder(mimeType?: string | null, fileName?: string | null): { icon: LucideIcon; label: string } {\n const extension = resolveFileExtension(fileName)\n const normalizedMime = typeof mimeType === 'string' ? mimeType.toLowerCase() : ''\n if (extension && EXTENSION_ICON_MAP[extension]) {\n return { icon: EXTENSION_ICON_MAP[extension], label: extension.toUpperCase() }\n }\n if (!extension && normalizedMime.includes('pdf')) {\n return { icon: FileText, label: 'PDF' }\n }\n if (!extension && normalizedMime.includes('zip')) {\n return { icon: FileArchive, label: 'ZIP' }\n }\n if (!extension && normalizedMime.includes('json')) {\n return { icon: FileCode, label: 'JSON' }\n }\n const mimeRoot = normalizedMime.split('/')[0] || ''\n if (mimeRoot && MIME_FALLBACK_ICONS[mimeRoot]) {\n return { icon: MIME_FALLBACK_ICONS[mimeRoot], label: mimeRoot.toUpperCase() }\n }\n const fallbackSource = extension || mimeRoot || 'file'\n const fallbackLabel = fallbackSource.slice(0, 6).toUpperCase()\n return { icon: File, label: fallbackLabel }\n}\n\nfunction normalizeCustomFieldSubmitValue(value: unknown): unknown {\n if (Array.isArray(value)) {\n return value.filter((entry) => entry !== undefined)\n }\n if (value === undefined) return null\n return value\n}\n\n\ntype AttachmentUploadFormValues = {\n files: File[]\n partitionCode?: string\n tags?: string[]\n assignments?: AssignmentDraft[]\n} & Record<string, unknown>\n\ntype AssignmentsEditorProps = {\n value: AssignmentDraft[]\n onChange: (next: AssignmentDraft[]) => void\n labels: {\n title: string\n description: string\n type: string\n id: string\n href: string\n label?: string\n add: string\n remove: string\n }\n disabled?: boolean\n}\n\nfunction AttachmentAssignmentsEditor({ value, onChange, labels, disabled }: AssignmentsEditorProps) {\n const handleChange = React.useCallback(\n (index: number, patch: Partial<AssignmentDraft>) => {\n onChange(value.map((entry, idx) => (idx === index ? { ...entry, ...patch } : entry)))\n },\n [onChange, value],\n )\n\n const handleRemove = React.useCallback(\n (index: number) => {\n onChange(value.filter((_, idx) => idx !== index))\n },\n [onChange, value],\n )\n\n const handleAdd = React.useCallback(() => {\n onChange([...value, { type: '', id: '', href: '', label: '' }])\n }, [onChange, value])\n\n return (\n <div className=\"space-y-2\">\n <div>\n <div className=\"text-sm font-medium\">{labels.title}</div>\n <div className=\"text-xs text-muted-foreground\">{labels.description}</div>\n </div>\n <div className=\"space-y-3\">\n {value.length === 0 ? (\n <div className=\"text-xs text-muted-foreground\">No assignments yet.</div>\n ) : (\n value.map((entry, index) => (\n <div key={`${index}-${entry.type}-${entry.id}`} className=\"rounded border p-3 space-y-2\">\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.type}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.type}\n disabled={disabled}\n onChange={(event) => handleChange(index, { type: event.target.value })}\n />\n </div>\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.id}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.id}\n disabled={disabled}\n onChange={(event) => handleChange(index, { id: event.target.value })}\n />\n </div>\n </div>\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.href}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.href ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { href: event.target.value })}\n />\n </div>\n {labels.label ? (\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.label}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.label ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { label: event.target.value })}\n />\n </div>\n ) : null}\n </div>\n <div className=\"flex justify-end\">\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n disabled={disabled}\n onClick={() => handleRemove(index)}\n className=\"inline-flex items-center gap-1 text-muted-foreground\"\n >\n <Trash2 className=\"h-4 w-4\" />\n {labels.remove}\n </Button>\n </div>\n </div>\n ))\n )}\n </div>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" disabled={disabled} onClick={handleAdd} className=\"inline-flex items-center gap-1\">\n <Plus className=\"h-4 w-4\" />\n {labels.add}\n </Button>\n </div>\n )\n}\n\ntype AttachmentFilesFieldProps = CrudCustomFieldRenderProps & {\n labels: {\n dropHint: string\n choose: string\n uploading: string\n empty: string\n }\n uploading: boolean\n}\n\nfunction AttachmentFilesField({\n value,\n setValue,\n disabled,\n error,\n labels,\n uploading,\n}: AttachmentFilesFieldProps) {\n const files = React.useMemo(() => (Array.isArray(value) ? (value as File[]) : []), [value])\n const [isDragOver, setDragOver] = React.useState(false)\n const fileInputRef = React.useRef<HTMLInputElement | null>(null)\n\n const acceptFiles = React.useCallback(\n (list: FileList | null) => {\n if (!list?.length) return\n const dedupe = new Map<string, File>(files.map((file) => [`${file.name}:${file.size}`, file]))\n Array.from(list).forEach((file) => {\n dedupe.set(`${file.name}:${file.size}`, file)\n })\n setValue(Array.from(dedupe.values()))\n },\n [files, setValue],\n )\n\n const handleDrop = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n acceptFiles(event.dataTransfer?.files ?? null)\n },\n [acceptFiles, disabled, uploading],\n )\n\n const handleDragOver = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(true)\n },\n [disabled, uploading],\n )\n\n const handleDragLeave = React.useCallback((event: React.DragEvent<HTMLDivElement>) => {\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n }, [])\n\n const removeFile = React.useCallback(\n (name: string, size: number) => {\n if (disabled || uploading) return\n setValue(files.filter((file) => !(file.name === name && file.size === size)))\n },\n [disabled, files, setValue, uploading],\n )\n\n const pickFiles = React.useCallback(() => {\n if (disabled || uploading) return\n fileInputRef.current?.click()\n }, [disabled, uploading])\n\n const renderFileList = () => {\n if (!files.length) {\n return <p className=\"text-xs text-muted-foreground\">{labels.empty}</p>\n }\n return (\n <div className=\"space-y-2\">\n {files.map((candidate) => (\n <div key={`${candidate.name}-${candidate.size}`} className=\"flex items-center justify-between rounded border px-3 py-2 text-sm\">\n <div>\n <div className=\"font-medium\">{candidate.name}</div>\n <div className=\"text-xs text-muted-foreground\">{formatFileSize(candidate.size)}</div>\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => removeFile(candidate.name, candidate.size)}\n disabled={disabled || uploading}\n >\n <Trash2 className=\"h-4 w-4\" />\n </Button>\n </div>\n ))}\n </div>\n )\n }\n\n return (\n <div className=\"space-y-2\">\n <div\n className={cn(\n 'flex flex-col items-center justify-center rounded-lg border border-dashed p-6 text-center transition-colors',\n isDragOver ? 'border-primary bg-primary/5' : 'border-muted-foreground/30',\n disabled || uploading ? 'opacity-70' : '',\n )}\n onDrop={handleDrop}\n onDragOver={handleDragOver}\n onDragLeave={handleDragLeave}\n role=\"presentation\"\n >\n <Upload className=\"mx-auto h-6 w-6 text-muted-foreground\" />\n <p className=\"mt-2 text-sm text-muted-foreground\">{labels.dropHint}</p>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" className=\"mt-4\" onClick={pickFiles} disabled={disabled || uploading}>\n {uploading ? labels.uploading : labels.choose}\n </Button>\n <input\n ref={fileInputRef}\n type=\"file\"\n className=\"hidden\"\n multiple\n onChange={(event) => {\n acceptFiles(event.target.files)\n event.currentTarget.value = ''\n }}\n disabled={disabled || uploading}\n />\n </div>\n {renderFileList()}\n {error ? <p className=\"text-xs font-medium text-red-600\">{error}</p> : null}\n </div>\n )\n}\n\ntype AttachmentUploadFormProps = {\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n onCancel: () => void\n}\n\nfunction AttachmentUploadForm({ partitions, availableTags, onUploaded, onCancel }: AttachmentUploadFormProps) {\n const t = useT()\n const [isUploading, setIsUploading] = React.useState(false)\n const [uploadProgress, setUploadProgress] = React.useState<{ completed: number; total: number }>({ completed: 0, total: 0 })\n\n const partitionOptions = React.useMemo(\n () =>\n partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n })),\n [partitions],\n )\n\n const assignmentLabels = React.useMemo(\n () => ({\n title: t('attachments.library.upload.assignments.title', 'Assignments'),\n description: t(\n 'attachments.library.upload.assignments.description',\n 'Optionally link this file to existing records now or add them later.',\n ),\n type: t('attachments.library.upload.assignments.type', 'Type'),\n id: t('attachments.library.upload.assignments.id', 'Record ID'),\n href: t('attachments.library.upload.assignments.href', 'Link'),\n label: t('attachments.library.upload.assignments.label', 'Label'),\n add: t('attachments.library.upload.assignments.add', 'Add assignment'),\n remove: t('attachments.library.upload.assignments.remove', 'Remove'),\n }),\n [t],\n )\n\n const formSchema = React.useMemo(\n () =>\n z\n .object({\n files: z.array(z.any()).min(1, { message: t('attachments.library.upload.fileRequired', 'Select at least one file to upload.') }),\n partitionCode: z.string().optional(),\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().optional(),\n label: z.string().optional(),\n }),\n )\n .optional(),\n })\n .passthrough(),\n [t],\n )\n\n const fields = React.useMemo<CrudField[]>(() => {\n return [\n {\n id: 'files',\n label: t('attachments.library.upload.file', 'Files'),\n type: 'custom',\n component: (props) => (\n <AttachmentFilesField\n {...props}\n uploading={isUploading}\n labels={{\n dropHint: t('attachments.library.upload.dropHint', 'Drag and drop files here or click to upload.'),\n choose: t('attachments.library.upload.choose', 'Choose files'),\n uploading: t('attachments.library.upload.submitting', 'Uploading\u2026'),\n empty: t('attachments.library.upload.noFiles', 'No files selected yet.'),\n }}\n />\n ),\n },\n {\n id: 'partitionCode',\n label: t('attachments.library.upload.partition', 'Partition'),\n type: 'select',\n options: [\n { value: '', label: t('attachments.library.upload.partitionDefault', 'Default (private)') },\n ...partitionOptions,\n ],\n },\n {\n id: 'tags',\n label: t('attachments.library.table.tags', 'Tags'),\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <TagsInput\n value={Array.isArray(value) ? (value as string[]) : []}\n onChange={(next) => setValue(next)}\n suggestions={availableTags}\n placeholder={t('attachments.library.upload.tagsPlaceholder', 'Add tags')}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n {\n id: 'assignments',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <AttachmentAssignmentsEditor\n value={Array.isArray(value) ? (value as AssignmentDraft[]) : []}\n onChange={(next) => setValue(next)}\n labels={assignmentLabels}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n ]\n }, [assignmentLabels, availableTags, isUploading, partitionOptions, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n return [\n {\n id: 'details',\n title: t('attachments.library.upload.title', 'Upload attachment'),\n column: 1,\n fields: ['files', 'partitionCode', 'tags', 'assignments'],\n },\n {\n id: 'customFields',\n title: t('entities.customFields.title', 'Custom attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n }, [t])\n\n const uploadPercentage = uploadProgress.total\n ? Math.min(100, Math.round((uploadProgress.completed / uploadProgress.total) * 100))\n : 0\n\n const handleSubmit = React.useCallback(\n async (values: AttachmentUploadFormValues) => {\n const files = Array.isArray(values.files) ? values.files : []\n if (!files.length) {\n throw new Error(t('attachments.library.upload.fileRequired', 'Select at least one file to upload.'))\n }\n setUploadProgress({ completed: 0, total: files.length })\n setIsUploading(true)\n try {\n const tags = Array.isArray(values.tags)\n ? values.tags\n .map((tag) => (typeof tag === 'string' ? tag.trim() : ''))\n .filter((tag) => tag.length > 0)\n : []\n const cleanedAssignments =\n Array.isArray(values.assignments) && values.assignments.length\n ? values.assignments\n .map((assignment) => ({\n type: assignment.type?.trim() ?? '',\n id: assignment.id?.trim() ?? '',\n href: assignment.href?.trim() || undefined,\n label: assignment.label?.trim() || undefined,\n }))\n .filter((assignment) => assignment.type && assignment.id)\n : []\n const customFields = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n let completed = 0\n for (const file of files) {\n const fd = new FormData()\n fd.set('entityId', LIBRARY_ENTITY_ID)\n fd.set(\n 'recordId',\n typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function' ? crypto.randomUUID() : String(Date.now()),\n )\n fd.set('file', file)\n if (typeof values.partitionCode === 'string' && values.partitionCode.trim().length) {\n fd.set('partitionCode', values.partitionCode.trim())\n }\n if (tags.length) fd.set('tags', JSON.stringify(tags))\n if (cleanedAssignments.length) fd.set('assignments', JSON.stringify(cleanedAssignments))\n if (Object.keys(customFields).length) fd.set('customFields', JSON.stringify(customFields))\n const call = await apiCall<{ error?: string }>('/api/attachments', {\n method: 'POST',\n body: fd,\n })\n if (!call.ok) {\n const message = call.result?.error || t('attachments.library.upload.failed', 'Upload failed.')\n throw new Error(message)\n }\n completed += 1\n setUploadProgress({ completed, total: files.length })\n }\n flash(t('attachments.library.upload.success', 'Attachment uploaded.'), 'success')\n onUploaded()\n onCancel()\n } catch (err: any) {\n const message = err?.message || t('attachments.library.upload.failed', 'Upload failed.')\n flash(message, 'error')\n throw new Error(message)\n } finally {\n setIsUploading(false)\n }\n },\n [onCancel, onUploaded, t],\n )\n\n return (\n <div className=\"relative\">\n <CrudForm<AttachmentUploadFormValues>\n embedded\n schema={formSchema}\n entityId={E.attachments.attachment}\n fields={fields}\n groups={groups}\n initialValues={{ files: [], tags: [], assignments: [], partitionCode: '' }}\n submitLabel={\n isUploading\n ? t('attachments.library.upload.submitting', 'Uploading\u2026')\n : t('attachments.library.upload.submit', 'Upload')\n }\n extraActions={\n <Button type=\"button\" variant=\"outline\" onClick={onCancel} disabled={isUploading}>\n {t('attachments.library.upload.cancel', 'Cancel')}\n </Button>\n }\n onSubmit={handleSubmit}\n />\n {isUploading ? (\n <div className=\"pointer-events-none absolute inset-0 z-20 flex items-center justify-center bg-background/90 px-6 text-center backdrop-blur\">\n <div className=\"flex w-full max-w-sm flex-col items-center gap-4 rounded-xl border border-border/50 bg-card/95 px-6 py-8 shadow-2xl\">\n <Spinner size=\"lg\" className=\"border-primary/50 border-t-primary\" />\n <div className=\"w-full space-y-3\">\n <p className=\"text-base font-semibold\">\n {t('attachments.library.upload.progressLabel', 'Uploading files')}\n </p>\n {uploadProgress.total > 0 ? (\n <>\n <p className=\"text-sm text-muted-foreground\">\n {uploadProgress.completed}/{uploadProgress.total}\n </p>\n <div className=\"h-2 w-full rounded bg-muted\">\n <div\n className=\"h-2 rounded bg-primary transition-all\"\n style={{\n width: `${uploadPercentage}%`,\n }}\n />\n </div>\n </>\n ) : null}\n </div>\n </div>\n </div>\n ) : null}\n </div>\n )\n}\ntype UploadDialogProps = {\n open: boolean\n onOpenChange: (next: boolean) => void\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n}\n\nfunction AttachmentUploadDialog({ open, onOpenChange, partitions, availableTags, onUploaded }: UploadDialogProps) {\n const t = useT()\n const [formResetKey, setFormResetKey] = React.useState(0)\n const previousOpen = React.useRef(open)\n\n React.useEffect(() => {\n if (previousOpen.current && !open) {\n setFormResetKey((prev) => prev + 1)\n }\n previousOpen.current = open\n }, [open])\n\n const handleDialogChange = React.useCallback(\n (next: boolean) => {\n onOpenChange(next)\n },\n [onOpenChange],\n )\n\n const handleUploaded = React.useCallback(() => {\n onUploaded()\n }, [onUploaded])\n\n return (\n <Dialog open={open} onOpenChange={handleDialogChange}>\n <DialogContent className=\"sm:max-w-[54.6rem]\">\n <DialogHeader>\n <DialogTitle>{t('attachments.library.upload.title', 'Upload attachment')}</DialogTitle>\n </DialogHeader>\n <AttachmentUploadForm\n key={formResetKey}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploaded}\n onCancel={() => handleDialogChange(false)}\n />\n </DialogContent>\n </Dialog>\n )\n}\n\nexport function AttachmentLibrary() {\n const t = useT()\n const queryClient = useQueryClient()\n const [page, setPage] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'createdAt', desc: true }])\n const [metadataDialogOpen, setMetadataDialogOpen] = React.useState(false)\n const [selectedRow, setSelectedRow] = React.useState<AttachmentRow | null>(null)\n const [uploadDialogOpen, setUploadDialogOpen] = React.useState(false)\n const [deleteDialogOpen, setDeleteDialogOpen] = React.useState(false)\n const [deleteTarget, setDeleteTarget] = React.useState<AttachmentRow | null>(null)\n const [deletePending, setDeletePending] = React.useState(false)\n const filterSignature = React.useMemo(() => buildFilterSignature(filterValues), [filterValues])\n const sortingSignature = React.useMemo(() => JSON.stringify(sorting), [sorting])\n\n const { data, isLoading, error, refetch } = useQuery({\n queryKey: ['attachments-library', page, search, filterSignature, sortingSignature],\n queryFn: async () => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(PAGE_SIZE))\n if (search.trim().length > 0) params.set('search', search.trim())\n const partition = typeof filterValues.partition === 'string' ? filterValues.partition : ''\n if (partition) params.set('partition', partition)\n const tags = Array.isArray(filterValues.tags) ? filterValues.tags : []\n if (tags.length > 0) params.set('tags', tags.join(','))\n if (sorting.length > 0) {\n const primary = sorting[0]\n params.set('sortField', primary.id)\n params.set('sortDir', primary.desc ? 'desc' : 'asc')\n }\n const call = await apiCall<AttachmentLibraryResponse>(`/api/attachments/library?${params.toString()}`)\n if (!call.ok || !call.result) {\n const message = call.result?.error || t('attachments.library.errors.load', 'Failed to load attachments.')\n throw new Error(message)\n }\n return call.result\n },\n })\n\n const partitions = data?.partitions ?? []\n const availableTags = data?.availableTags ?? []\n\n const filters = React.useMemo<FilterDef[]>(() => {\n const partitionOptions = partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n }))\n return [\n {\n id: 'partition',\n label: t('attachments.library.filters.partition', 'Partition'),\n type: 'select',\n options: partitionOptions,\n },\n {\n id: 'tags',\n label: t('attachments.library.filters.tags', 'Tags'),\n type: 'tags',\n placeholder: t('attachments.library.filters.tagsPlaceholder', 'Filter by tag'),\n options: availableTags.map((tag) => ({ value: tag, label: tag })),\n },\n ]\n }, [availableTags, partitions, t])\n\n const items = data?.items ?? []\n\n const columns = React.useMemo<ColumnDef<AttachmentRow>[]>(() => {\n return [\n {\n id: 'preview',\n header: '',\n enableSorting: false,\n cell: ({ row }) => {\n const value = row.original\n if (value.thumbnailUrl) {\n return (\n <div className=\"h-16 w-16 overflow-hidden rounded border bg-muted\">\n <img\n src={value.thumbnailUrl}\n alt={value.fileName}\n className=\"h-full w-full object-cover\"\n loading=\"lazy\"\n />\n </div>\n )\n }\n const placeholder = resolveAttachmentPlaceholder(value.mimeType, value.fileName)\n const PlaceholderIcon = placeholder.icon\n return (\n <div className=\"flex h-16 w-16 flex-col items-center justify-center rounded border bg-muted text-[10px] font-semibold uppercase text-muted-foreground\">\n <PlaceholderIcon className=\"mb-1 h-5 w-5 text-muted-foreground\" aria-hidden />\n {placeholder.label}\n </div>\n )\n },\n },\n {\n id: 'fileName',\n accessorKey: 'fileName',\n header: t('attachments.library.table.file', 'File'),\n cell: ({ row }) => {\n const value = row.original\n return (\n <div className=\"space-y-1 min-w-0 max-w-[280px]\">\n <div className=\"font-medium truncate\" title={value.fileName}>\n {value.fileName}\n </div>\n <div className=\"text-xs text-muted-foreground\">\n {formatFileSize(value.fileSize)} \u2022 {value.mimeType || 'application/octet-stream'}\n </div>\n <div className=\"text-xs text-muted-foreground line-clamp-2\">\n {value.content?.trim()\n ? value.content\n : t('attachments.library.metadata.noContent', 'No text extracted')}\n </div>\n </div>\n )\n },\n },\n {\n id: 'tags',\n accessorKey: 'tags',\n header: t('attachments.library.table.tags', 'Tags'),\n enableSorting: false,\n cell: ({ row }) => {\n const tags = row.original.tags ?? []\n if (!tags.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-wrap gap-1\">\n {tags.map((tag) => (\n <Badge key={tag} variant=\"outline\">\n {tag}\n </Badge>\n ))}\n </div>\n )\n },\n },\n {\n id: 'assignments',\n accessorKey: 'assignments',\n header: t('attachments.library.table.assignments', 'Assignments'),\n enableSorting: false,\n cell: ({ row }) => {\n const assignments = filterLibraryAssignments(row.original.assignments)\n if (!assignments.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-col gap-1\">\n {assignments.map((assignment) => {\n const label = assignment.label?.trim() || assignment.id\n const hideType =\n assignment.type === (E as any).catalog?.catalog_product ||\n assignment.type === (E as any).catalog?.catalog_product_variant\n const content = hideType ? label : `${assignment.type}: ${label}`\n return assignment.href ? (\n <a\n key={`${assignment.type}-${assignment.id}-${assignment.href}`}\n href={assignment.href}\n className=\"text-sm text-blue-600 underline\"\n target=\"_blank\"\n rel=\"noreferrer\"\n >\n {content}\n </a>\n ) : (\n <div key={`${assignment.type}-${assignment.id}`} className=\"text-sm\">\n {content}\n </div>\n )\n })}\n </div>\n )\n },\n },\n {\n id: 'partitionCode',\n accessorKey: 'partitionCode',\n header: t('attachments.library.table.partition', 'Partition'),\n cell: ({ row }) => (\n <div className=\"text-sm text-muted-foreground\">\n {row.original.partitionTitle ?? row.original.partitionCode}\n </div>\n ),\n },\n {\n id: 'createdAt',\n accessorKey: 'createdAt',\n header: t('attachments.library.table.created', 'Created'),\n cell: ({ row }) => {\n const createdAt = row.original.createdAt\n return (\n <div className=\"text-sm text-muted-foreground\">\n {createdAt ? humanDate(createdAt) : '\u2014'}\n </div>\n )\n },\n },\n {\n id: 'download',\n header: t('attachments.library.table.download', 'Download'),\n enableSorting: false,\n cell: ({ row }) => {\n const downloadPath = buildAttachmentFileUrl(row.original.id, { download: true })\n const absolute = resolveAbsoluteUrl(downloadPath)\n return (\n <Button variant=\"ghost\" size=\"icon\" asChild>\n <a href={absolute} download aria-label={t('attachments.library.table.download', 'Download')}>\n <Download className=\"h-4 w-4\" />\n </a>\n </Button>\n )\n },\n },\n ]\n }, [t])\n\n const openMetadataDialog = React.useCallback((row: AttachmentRow) => {\n setSelectedRow(row)\n setMetadataDialogOpen(true)\n }, [])\n\n const openDeleteDialog = React.useCallback((row: AttachmentRow) => {\n setDeleteTarget(row)\n setDeleteDialogOpen(true)\n }, [])\n\n const handleMetadataSave = React.useCallback(\n async (id: string, payload: AttachmentMetadataSavePayload) => {\n try {\n const body: Record<string, unknown> = {\n tags: payload.tags,\n assignments: payload.assignments,\n }\n if (payload.customFields && Object.keys(payload.customFields).length) {\n body.customFields = payload.customFields\n }\n const call = await apiCall<{ error?: string }>(`/api/attachments/library/${encodeURIComponent(id)}`, {\n method: 'PATCH',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify(body),\n })\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.metadata.error', 'Failed to update metadata.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.metadata.success', 'Attachment updated.'), 'success')\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n setMetadataDialogOpen(false)\n } catch (err: any) {\n flash(err?.message || t('attachments.library.metadata.error', 'Failed to update metadata.'), 'error')\n }\n },\n [queryClient, t],\n )\n\n const handleUploadCompleted = React.useCallback(async () => {\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n }, [queryClient])\n\n const handleDelete = React.useCallback(async () => {\n if (!deleteTarget) return\n try {\n setDeletePending(true)\n const call = await apiCall<{ error?: string }>(\n `/api/attachments/library/${encodeURIComponent(deleteTarget.id)}`,\n { method: 'DELETE' },\n )\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.errors.delete', 'Failed to delete attachment.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.messages.deleted', 'Attachment removed.'), 'success')\n if (selectedRow?.id === deleteTarget.id) {\n setSelectedRow(null)\n setMetadataDialogOpen(false)\n }\n setDeleteDialogOpen(false)\n setDeleteTarget(null)\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n } catch (err: any) {\n flash(err?.message || t('attachments.library.errors.delete', 'Failed to delete attachment.'), 'error')\n } finally {\n setDeletePending(false)\n }\n }, [deleteTarget, queryClient, selectedRow, t])\n\n const total = data?.total ?? 0\n const totalPages = data?.totalPages ?? 0\n return (\n <>\n <DataTable<AttachmentRow>\n title={t('attachments.library.title', 'Attachments')}\n refreshButton={{\n label: t('attachments.library.actions.refresh', 'Refresh'),\n onRefresh: () => { void refetch() },\n isRefreshing: isLoading,\n }}\n actions={(\n <Button onClick={() => setUploadDialogOpen(true)}>\n {t('attachments.library.actions.upload', 'Upload')}\n </Button>\n )}\n columns={columns}\n data={items}\n sorting={sorting}\n onSortingChange={setSorting}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'open',\n label: t('attachments.library.actions.open', 'Open'),\n onSelect: () => {\n if (!row.url) return\n window.open(row.url, '_blank', 'noopener,noreferrer')\n },\n },\n {\n id: 'edit',\n label: t('attachments.library.actions.edit', 'Edit metadata'),\n onSelect: () => openMetadataDialog(row),\n },\n {\n id: 'copy-url',\n label: t('attachments.library.actions.copyUrl', 'Copy URL'),\n onSelect: () => {\n if (!row.url) {\n flash(t('attachments.library.actions.copyError', 'Unable to copy link.'), 'error')\n return\n }\n const absolute = resolveAbsoluteUrl(row.url)\n navigator.clipboard\n .writeText(absolute)\n .then(() =>\n flash(\n t('attachments.library.actions.copied', 'Link copied.'),\n 'success',\n ),\n )\n .catch(() =>\n flash(\n t('attachments.library.actions.copyError', 'Unable to copy link.'),\n 'error',\n ),\n )\n },\n },\n {\n id: 'delete',\n label: t('attachments.library.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => openDeleteDialog(row),\n },\n ]}\n />\n )}\n onRowClick={(row) => openMetadataDialog(row)}\n isLoading={isLoading}\n error={error?.message}\n emptyState={\n <div className=\"py-10 text-center text-sm text-muted-foreground\">\n {t('attachments.library.table.empty', 'No attachments found.')}\n </div>\n }\n searchValue={search}\n onSearchChange={(value) => {\n setPage(1)\n setSearch(value)\n }}\n searchPlaceholder={t('attachments.library.table.search', 'Search files\u2026')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={(values) => {\n setFilterValues(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilterValues({})\n setPage(1)\n }}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: (next) => setPage(next),\n }}\n />\n <AttachmentMetadataDialog\n open={metadataDialogOpen}\n onOpenChange={setMetadataDialogOpen}\n item={selectedRow}\n availableTags={availableTags}\n onSave={handleMetadataSave}\n />\n <AttachmentDeleteDialog\n open={deleteDialogOpen}\n onOpenChange={setDeleteDialogOpen}\n fileName={deleteTarget?.fileName}\n onConfirm={handleDelete}\n isDeleting={deletePending}\n />\n <AttachmentUploadDialog\n open={uploadDialogOpen}\n onOpenChange={setUploadDialogOpen}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploadCompleted}\n />\n </>\n )\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["\"use client\"\n\nimport * as React from 'react'\nimport type { ColumnDef, SortingState } from '@tanstack/react-table'\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { DataTable } from '@open-mercato/ui/backend/DataTable'\nimport { RowActions } from '@open-mercato/ui/backend/RowActions'\nimport type { FilterDef, FilterValues } from '@open-mercato/ui/backend/FilterBar'\nimport { Button } from '@open-mercato/ui/primitives/button'\nimport { Badge } from '@open-mercato/ui/primitives/badge'\nimport { Dialog, DialogContent, DialogHeader, DialogTitle } from '@open-mercato/ui/primitives/dialog'\nimport { Input } from '@open-mercato/ui/primitives/input'\nimport { TagsInput } from '@open-mercato/ui/backend/inputs/TagsInput'\nimport { Spinner } from '@open-mercato/ui/primitives/spinner'\nimport { CrudForm, type CrudField, type CrudFormGroup, type CrudCustomFieldRenderProps } from '@open-mercato/ui/backend/CrudForm'\nimport { collectCustomFieldValues } from '@open-mercato/ui/backend/utils/customFieldValues'\nimport { apiCall } from '@open-mercato/ui/backend/utils/apiCall'\nimport { flash } from '@open-mercato/ui/backend/FlashMessages'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\nimport { z } from 'zod'\nimport { E } from '#generated/entities.ids.generated'\nimport type { LucideIcon } from 'lucide-react'\nimport { Download, Plus, Upload, Trash2, File, FileText, FileSpreadsheet, FileArchive, FileAudio, FileVideo, FileCode } from 'lucide-react'\nimport { buildAttachmentFileUrl, buildAttachmentImageUrl, slugifyAttachmentFileName } from '@open-mercato/core/modules/attachments/lib/imageUrls'\nimport { cn } from '@open-mercato/shared/lib/utils'\nimport { AttachmentDeleteDialog, AttachmentMetadataDialog, type AttachmentItem, type AttachmentMetadataSavePayload, type AssignmentDraft } from '@open-mercato/ui/backend/detail'\n\ntype AttachmentAssignment = {\n type: string\n id: string\n href?: string | null\n label?: string | null\n}\n\ntype AttachmentRow = AttachmentItem\n\ntype AttachmentLibraryResponse = {\n items: AttachmentRow[]\n page: number\n pageSize: number\n total: number\n totalPages: number\n availableTags: string[]\n partitions: Array<{ code: string; title: string; description?: string | null; isPublic?: boolean }>\n error?: string\n}\n\nconst PAGE_SIZE = 25\nconst ENV_APP_URL = (process.env.NEXT_PUBLIC_APP_URL || '').replace(/\\/$/, '')\nconst LIBRARY_ENTITY_ID = 'attachments:library'\n\nfunction filterLibraryAssignments(assignments?: AttachmentAssignment[] | null): AttachmentAssignment[] {\n return (assignments ?? []).filter((assignment) => assignment.type !== LIBRARY_ENTITY_ID)\n}\n\nfunction formatFileSize(value: number): string {\n if (!Number.isFinite(value)) return '\u2014'\n if (value <= 0) return '0 B'\n const units = ['B', 'KB', 'MB', 'GB', 'TB']\n let idx = 0\n let current = value\n while (current >= 1024 && idx < units.length - 1) {\n current /= 1024\n idx += 1\n }\n return `${current.toFixed(idx === 0 ? 0 : 1)} ${units[idx]}`\n}\n\nfunction humanDate(value: string, locale?: string): string {\n const date = new Date(value)\n if (Number.isNaN(date.getTime())) return value\n return date.toLocaleString(locale ?? undefined)\n}\n\nfunction buildFilterSignature(values: FilterValues): string {\n return JSON.stringify(values, Object.keys(values).sort((a, b) => a.localeCompare(b)))\n}\n\nfunction resolveAbsoluteUrl(path: string): string {\n if (!path) return path\n if (/^https?:\\/\\//i.test(path)) return path\n const base =\n ENV_APP_URL ||\n (typeof window !== 'undefined' && window.location?.origin ? window.location.origin : '')\n if (!base) return path\n const normalizedBase = base.replace(/\\/$/, '')\n return `${normalizedBase}${path.startsWith('/') ? path : `/${path}`}`\n}\n\nfunction resolveFileExtension(fileName?: string | null): string {\n if (!fileName) return ''\n const normalized = fileName.trim()\n if (!normalized) return ''\n const lastDot = normalized.lastIndexOf('.')\n if (lastDot === -1 || lastDot === normalized.length - 1) return ''\n return normalized.slice(lastDot + 1).toLowerCase()\n}\n\nconst EXTENSION_ICON_MAP: Record<string, LucideIcon> = {\n pdf: FileText,\n doc: FileText,\n docx: FileText,\n txt: FileText,\n md: FileText,\n rtf: FileText,\n xls: FileSpreadsheet,\n xlsx: FileSpreadsheet,\n csv: FileSpreadsheet,\n ods: FileSpreadsheet,\n ppt: FileText,\n pptx: FileText,\n zip: FileArchive,\n gz: FileArchive,\n rar: FileArchive,\n tgz: FileArchive,\n '7z': FileArchive,\n tar: FileArchive,\n json: FileCode,\n js: FileCode,\n ts: FileCode,\n jsx: FileCode,\n tsx: FileCode,\n html: FileCode,\n css: FileCode,\n xml: FileCode,\n yaml: FileCode,\n yml: FileCode,\n mp3: FileAudio,\n wav: FileAudio,\n flac: FileAudio,\n ogg: FileAudio,\n mp4: FileVideo,\n mov: FileVideo,\n avi: FileVideo,\n webm: FileVideo,\n}\n\nconst MIME_FALLBACK_ICONS: Record<string, LucideIcon> = {\n audio: FileAudio,\n video: FileVideo,\n text: FileText,\n application: FileText,\n}\n\nfunction resolveAttachmentPlaceholder(mimeType?: string | null, fileName?: string | null): { icon: LucideIcon; label: string } {\n const extension = resolveFileExtension(fileName)\n const normalizedMime = typeof mimeType === 'string' ? mimeType.toLowerCase() : ''\n if (extension && EXTENSION_ICON_MAP[extension]) {\n return { icon: EXTENSION_ICON_MAP[extension], label: extension.toUpperCase() }\n }\n if (!extension && normalizedMime.includes('pdf')) {\n return { icon: FileText, label: 'PDF' }\n }\n if (!extension && normalizedMime.includes('zip')) {\n return { icon: FileArchive, label: 'ZIP' }\n }\n if (!extension && normalizedMime.includes('json')) {\n return { icon: FileCode, label: 'JSON' }\n }\n const mimeRoot = normalizedMime.split('/')[0] || ''\n if (mimeRoot && MIME_FALLBACK_ICONS[mimeRoot]) {\n return { icon: MIME_FALLBACK_ICONS[mimeRoot], label: mimeRoot.toUpperCase() }\n }\n const fallbackSource = extension || mimeRoot || 'file'\n const fallbackLabel = fallbackSource.slice(0, 6).toUpperCase()\n return { icon: File, label: fallbackLabel }\n}\n\nfunction normalizeCustomFieldSubmitValue(value: unknown): unknown {\n if (Array.isArray(value)) {\n return value.filter((entry) => entry !== undefined)\n }\n if (value === undefined) return null\n return value\n}\n\n\ntype AttachmentUploadFormValues = {\n files: File[]\n partitionCode?: string\n tags?: string[]\n assignments?: AssignmentDraft[]\n} & Record<string, unknown>\n\ntype AssignmentsEditorProps = {\n value: AssignmentDraft[]\n onChange: (next: AssignmentDraft[]) => void\n labels: {\n title: string\n description: string\n type: string\n id: string\n href: string\n label?: string\n add: string\n remove: string\n }\n disabled?: boolean\n}\n\nfunction AttachmentAssignmentsEditor({ value, onChange, labels, disabled }: AssignmentsEditorProps) {\n const handleChange = React.useCallback(\n (index: number, patch: Partial<AssignmentDraft>) => {\n onChange(value.map((entry, idx) => (idx === index ? { ...entry, ...patch } : entry)))\n },\n [onChange, value],\n )\n\n const handleRemove = React.useCallback(\n (index: number) => {\n onChange(value.filter((_, idx) => idx !== index))\n },\n [onChange, value],\n )\n\n const handleAdd = React.useCallback(() => {\n onChange([...value, { type: '', id: '', href: '', label: '' }])\n }, [onChange, value])\n\n return (\n <div className=\"space-y-2\">\n <div>\n <div className=\"text-sm font-medium\">{labels.title}</div>\n <div className=\"text-xs text-muted-foreground\">{labels.description}</div>\n </div>\n <div className=\"space-y-3\">\n {value.length === 0 ? (\n <div className=\"text-xs text-muted-foreground\">No assignments yet.</div>\n ) : (\n value.map((entry, index) => (\n <div key={`${index}-${entry.type}-${entry.id}`} className=\"rounded border p-3 space-y-2\">\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.type}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.type}\n disabled={disabled}\n onChange={(event) => handleChange(index, { type: event.target.value })}\n />\n </div>\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.id}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.id}\n disabled={disabled}\n onChange={(event) => handleChange(index, { id: event.target.value })}\n />\n </div>\n </div>\n <div className=\"grid gap-2 sm:grid-cols-2\">\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.href}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.href ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { href: event.target.value })}\n />\n </div>\n {labels.label ? (\n <div className=\"space-y-1\">\n <label className=\"text-xs font-medium\">{labels.label}</label>\n <input\n className=\"w-full rounded border px-2 py-1 text-sm\"\n value={entry.label ?? ''}\n disabled={disabled}\n onChange={(event) => handleChange(index, { label: event.target.value })}\n />\n </div>\n ) : null}\n </div>\n <div className=\"flex justify-end\">\n <Button\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n disabled={disabled}\n onClick={() => handleRemove(index)}\n className=\"inline-flex items-center gap-1 text-muted-foreground\"\n >\n <Trash2 className=\"h-4 w-4\" />\n {labels.remove}\n </Button>\n </div>\n </div>\n ))\n )}\n </div>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" disabled={disabled} onClick={handleAdd} className=\"inline-flex items-center gap-1\">\n <Plus className=\"h-4 w-4\" />\n {labels.add}\n </Button>\n </div>\n )\n}\n\ntype AttachmentFilesFieldProps = CrudCustomFieldRenderProps & {\n labels: {\n dropHint: string\n choose: string\n uploading: string\n empty: string\n }\n uploading: boolean\n}\n\nfunction AttachmentFilesField({\n value,\n setValue,\n disabled,\n labels,\n uploading,\n}: AttachmentFilesFieldProps) {\n const files = React.useMemo(() => (Array.isArray(value) ? (value as File[]) : []), [value])\n const [isDragOver, setDragOver] = React.useState(false)\n const fileInputRef = React.useRef<HTMLInputElement | null>(null)\n\n const acceptFiles = React.useCallback(\n (list: FileList | null) => {\n if (!list?.length) return\n const dedupe = new Map<string, File>(files.map((file) => [`${file.name}:${file.size}`, file]))\n Array.from(list).forEach((file) => {\n dedupe.set(`${file.name}:${file.size}`, file)\n })\n setValue(Array.from(dedupe.values()))\n },\n [files, setValue],\n )\n\n const handleDrop = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n acceptFiles(event.dataTransfer?.files ?? null)\n },\n [acceptFiles, disabled, uploading],\n )\n\n const handleDragOver = React.useCallback(\n (event: React.DragEvent<HTMLDivElement>) => {\n if (disabled || uploading) return\n event.preventDefault()\n event.stopPropagation()\n setDragOver(true)\n },\n [disabled, uploading],\n )\n\n const handleDragLeave = React.useCallback((event: React.DragEvent<HTMLDivElement>) => {\n event.preventDefault()\n event.stopPropagation()\n setDragOver(false)\n }, [])\n\n const removeFile = React.useCallback(\n (name: string, size: number) => {\n if (disabled || uploading) return\n setValue(files.filter((file) => !(file.name === name && file.size === size)))\n },\n [disabled, files, setValue, uploading],\n )\n\n const pickFiles = React.useCallback(() => {\n if (disabled || uploading) return\n fileInputRef.current?.click()\n }, [disabled, uploading])\n\n const renderFileList = () => {\n if (!files.length) {\n return <p className=\"text-xs text-muted-foreground\">{labels.empty}</p>\n }\n return (\n <div className=\"space-y-2\">\n {files.map((candidate) => (\n <div key={`${candidate.name}-${candidate.size}`} className=\"flex items-center justify-between rounded border px-3 py-2 text-sm\">\n <div>\n <div className=\"font-medium\">{candidate.name}</div>\n <div className=\"text-xs text-muted-foreground\">{formatFileSize(candidate.size)}</div>\n </div>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n onClick={() => removeFile(candidate.name, candidate.size)}\n disabled={disabled || uploading}\n >\n <Trash2 className=\"h-4 w-4\" />\n </Button>\n </div>\n ))}\n </div>\n )\n }\n\n return (\n <div className=\"space-y-2\">\n <div\n className={cn(\n 'flex flex-col items-center justify-center rounded-lg border border-dashed p-6 text-center transition-colors',\n isDragOver ? 'border-primary bg-primary/5' : 'border-muted-foreground/30',\n disabled || uploading ? 'opacity-70' : '',\n )}\n onDrop={handleDrop}\n onDragOver={handleDragOver}\n onDragLeave={handleDragLeave}\n role=\"presentation\"\n >\n <Upload className=\"mx-auto h-6 w-6 text-muted-foreground\" />\n <p className=\"mt-2 text-sm text-muted-foreground\">{labels.dropHint}</p>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" className=\"mt-4\" onClick={pickFiles} disabled={disabled || uploading}>\n {uploading ? labels.uploading : labels.choose}\n </Button>\n <input\n ref={fileInputRef}\n type=\"file\"\n className=\"hidden\"\n multiple\n onChange={(event) => {\n acceptFiles(event.target.files)\n event.currentTarget.value = ''\n }}\n disabled={disabled || uploading}\n />\n </div>\n {renderFileList()}\n </div>\n )\n}\n\ntype AttachmentUploadFormProps = {\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n onCancel: () => void\n}\n\nfunction AttachmentUploadForm({ partitions, availableTags, onUploaded, onCancel }: AttachmentUploadFormProps) {\n const t = useT()\n const [isUploading, setIsUploading] = React.useState(false)\n const [uploadProgress, setUploadProgress] = React.useState<{ completed: number; total: number }>({ completed: 0, total: 0 })\n\n const partitionOptions = React.useMemo(\n () =>\n partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n })),\n [partitions],\n )\n\n const assignmentLabels = React.useMemo(\n () => ({\n title: t('attachments.library.upload.assignments.title', 'Assignments'),\n description: t(\n 'attachments.library.upload.assignments.description',\n 'Optionally link this file to existing records now or add them later.',\n ),\n type: t('attachments.library.upload.assignments.type', 'Type'),\n id: t('attachments.library.upload.assignments.id', 'Record ID'),\n href: t('attachments.library.upload.assignments.href', 'Link'),\n label: t('attachments.library.upload.assignments.label', 'Label'),\n add: t('attachments.library.upload.assignments.add', 'Add assignment'),\n remove: t('attachments.library.upload.assignments.remove', 'Remove'),\n }),\n [t],\n )\n\n const formSchema = React.useMemo(\n () =>\n z\n .object({\n files: z.array(z.any()).min(1, { message: t('attachments.library.upload.fileRequired', 'Select at least one file to upload.') }),\n partitionCode: z.string().optional(),\n tags: z.array(z.string()).optional(),\n assignments: z\n .array(\n z.object({\n type: z.string().min(1),\n id: z.string().min(1),\n href: z.string().optional(),\n label: z.string().optional(),\n }),\n )\n .optional(),\n })\n .passthrough(),\n [t],\n )\n\n const fields = React.useMemo<CrudField[]>(() => {\n return [\n {\n id: 'files',\n label: t('attachments.library.upload.file', 'Files'),\n type: 'custom',\n component: (props) => (\n <AttachmentFilesField\n {...props}\n uploading={isUploading}\n labels={{\n dropHint: t('attachments.library.upload.dropHint', 'Drag and drop files here or click to upload.'),\n choose: t('attachments.library.upload.choose', 'Choose files'),\n uploading: t('attachments.library.upload.submitting', 'Uploading\u2026'),\n empty: t('attachments.library.upload.noFiles', 'No files selected yet.'),\n }}\n />\n ),\n },\n {\n id: 'partitionCode',\n label: t('attachments.library.upload.partition', 'Partition'),\n type: 'select',\n options: [\n { value: '', label: t('attachments.library.upload.partitionDefault', 'Default (private)') },\n ...partitionOptions,\n ],\n },\n {\n id: 'tags',\n label: t('attachments.library.table.tags', 'Tags'),\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <TagsInput\n value={Array.isArray(value) ? (value as string[]) : []}\n onChange={(next) => setValue(next)}\n suggestions={availableTags}\n placeholder={t('attachments.library.upload.tagsPlaceholder', 'Add tags')}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n {\n id: 'assignments',\n label: '',\n type: 'custom',\n component: ({ value, setValue, disabled }) => (\n <AttachmentAssignmentsEditor\n value={Array.isArray(value) ? (value as AssignmentDraft[]) : []}\n onChange={(next) => setValue(next)}\n labels={assignmentLabels}\n disabled={Boolean(disabled) || isUploading}\n />\n ),\n },\n ]\n }, [assignmentLabels, availableTags, isUploading, partitionOptions, t])\n\n const groups = React.useMemo<CrudFormGroup[]>(() => {\n return [\n {\n id: 'details',\n title: t('attachments.library.upload.title', 'Upload attachment'),\n column: 1,\n fields: ['files', 'partitionCode', 'tags', 'assignments'],\n },\n {\n id: 'customFields',\n title: t('entities.customFields.title', 'Custom attributes'),\n column: 2,\n kind: 'customFields',\n },\n ]\n }, [t])\n\n const uploadPercentage = uploadProgress.total\n ? Math.min(100, Math.round((uploadProgress.completed / uploadProgress.total) * 100))\n : 0\n\n const handleSubmit = React.useCallback(\n async (values: AttachmentUploadFormValues) => {\n const files = Array.isArray(values.files) ? values.files : []\n if (!files.length) {\n throw new Error(t('attachments.library.upload.fileRequired', 'Select at least one file to upload.'))\n }\n setUploadProgress({ completed: 0, total: files.length })\n setIsUploading(true)\n try {\n const tags = Array.isArray(values.tags)\n ? values.tags\n .map((tag) => (typeof tag === 'string' ? tag.trim() : ''))\n .filter((tag) => tag.length > 0)\n : []\n const cleanedAssignments =\n Array.isArray(values.assignments) && values.assignments.length\n ? values.assignments\n .map((assignment) => ({\n type: assignment.type?.trim() ?? '',\n id: assignment.id?.trim() ?? '',\n href: assignment.href?.trim() || undefined,\n label: assignment.label?.trim() || undefined,\n }))\n .filter((assignment) => assignment.type && assignment.id)\n : []\n const customFields = collectCustomFieldValues(values, {\n transform: (value) => normalizeCustomFieldSubmitValue(value),\n })\n let completed = 0\n for (const file of files) {\n const fd = new FormData()\n fd.set('entityId', LIBRARY_ENTITY_ID)\n fd.set(\n 'recordId',\n typeof crypto !== 'undefined' && typeof crypto.randomUUID === 'function' ? crypto.randomUUID() : String(Date.now()),\n )\n fd.set('file', file)\n if (typeof values.partitionCode === 'string' && values.partitionCode.trim().length) {\n fd.set('partitionCode', values.partitionCode.trim())\n }\n if (tags.length) fd.set('tags', JSON.stringify(tags))\n if (cleanedAssignments.length) fd.set('assignments', JSON.stringify(cleanedAssignments))\n if (Object.keys(customFields).length) fd.set('customFields', JSON.stringify(customFields))\n const call = await apiCall<{ error?: string }>('/api/attachments', {\n method: 'POST',\n body: fd,\n })\n if (!call.ok) {\n const message = call.result?.error || t('attachments.library.upload.failed', 'Upload failed.')\n throw new Error(message)\n }\n completed += 1\n setUploadProgress({ completed, total: files.length })\n }\n flash(t('attachments.library.upload.success', 'Attachment uploaded.'), 'success')\n onUploaded()\n onCancel()\n } catch (err: any) {\n const message = err?.message || t('attachments.library.upload.failed', 'Upload failed.')\n flash(message, 'error')\n throw new Error(message)\n } finally {\n setIsUploading(false)\n }\n },\n [onCancel, onUploaded, t],\n )\n\n return (\n <div className=\"relative\">\n <CrudForm<AttachmentUploadFormValues>\n embedded\n schema={formSchema}\n entityId={E.attachments.attachment}\n fields={fields}\n groups={groups}\n initialValues={{ files: [], tags: [], assignments: [], partitionCode: '' }}\n submitLabel={\n isUploading\n ? t('attachments.library.upload.submitting', 'Uploading\u2026')\n : t('attachments.library.upload.submit', 'Upload')\n }\n extraActions={\n <Button type=\"button\" variant=\"outline\" onClick={onCancel} disabled={isUploading}>\n {t('attachments.library.upload.cancel', 'Cancel')}\n </Button>\n }\n onSubmit={handleSubmit}\n />\n {isUploading ? (\n <div className=\"pointer-events-none absolute inset-0 z-20 flex items-center justify-center bg-background/90 px-6 text-center backdrop-blur\">\n <div className=\"flex w-full max-w-sm flex-col items-center gap-4 rounded-xl border border-border/50 bg-card/95 px-6 py-8 shadow-2xl\">\n <Spinner size=\"lg\" className=\"border-primary/50 border-t-primary\" />\n <div className=\"w-full space-y-3\">\n <p className=\"text-base font-semibold\">\n {t('attachments.library.upload.progressLabel', 'Uploading files')}\n </p>\n {uploadProgress.total > 0 ? (\n <>\n <p className=\"text-sm text-muted-foreground\">\n {uploadProgress.completed}/{uploadProgress.total}\n </p>\n <div className=\"h-2 w-full rounded bg-muted\">\n <div\n className=\"h-2 rounded bg-primary transition-all\"\n style={{\n width: `${uploadPercentage}%`,\n }}\n />\n </div>\n </>\n ) : null}\n </div>\n </div>\n </div>\n ) : null}\n </div>\n )\n}\ntype UploadDialogProps = {\n open: boolean\n onOpenChange: (next: boolean) => void\n partitions: Array<{ code: string; title: string }>\n availableTags: string[]\n onUploaded: () => void\n}\n\nfunction AttachmentUploadDialog({ open, onOpenChange, partitions, availableTags, onUploaded }: UploadDialogProps) {\n const t = useT()\n const [formResetKey, setFormResetKey] = React.useState(0)\n const previousOpen = React.useRef(open)\n\n React.useEffect(() => {\n if (previousOpen.current && !open) {\n setFormResetKey((prev) => prev + 1)\n }\n previousOpen.current = open\n }, [open])\n\n const handleDialogChange = React.useCallback(\n (next: boolean) => {\n onOpenChange(next)\n },\n [onOpenChange],\n )\n\n const handleUploaded = React.useCallback(() => {\n onUploaded()\n }, [onUploaded])\n\n return (\n <Dialog open={open} onOpenChange={handleDialogChange}>\n <DialogContent className=\"sm:max-w-[54.6rem]\">\n <DialogHeader>\n <DialogTitle>{t('attachments.library.upload.title', 'Upload attachment')}</DialogTitle>\n </DialogHeader>\n <AttachmentUploadForm\n key={formResetKey}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploaded}\n onCancel={() => handleDialogChange(false)}\n />\n </DialogContent>\n </Dialog>\n )\n}\n\nexport function AttachmentLibrary() {\n const t = useT()\n const queryClient = useQueryClient()\n const [page, setPage] = React.useState(1)\n const [search, setSearch] = React.useState('')\n const [filterValues, setFilterValues] = React.useState<FilterValues>({})\n const [sorting, setSorting] = React.useState<SortingState>([{ id: 'createdAt', desc: true }])\n const [metadataDialogOpen, setMetadataDialogOpen] = React.useState(false)\n const [selectedRow, setSelectedRow] = React.useState<AttachmentRow | null>(null)\n const [uploadDialogOpen, setUploadDialogOpen] = React.useState(false)\n const [deleteDialogOpen, setDeleteDialogOpen] = React.useState(false)\n const [deleteTarget, setDeleteTarget] = React.useState<AttachmentRow | null>(null)\n const [deletePending, setDeletePending] = React.useState(false)\n const filterSignature = React.useMemo(() => buildFilterSignature(filterValues), [filterValues])\n const sortingSignature = React.useMemo(() => JSON.stringify(sorting), [sorting])\n\n const { data, isLoading, error, refetch } = useQuery({\n queryKey: ['attachments-library', page, search, filterSignature, sortingSignature],\n queryFn: async () => {\n const params = new URLSearchParams()\n params.set('page', String(page))\n params.set('pageSize', String(PAGE_SIZE))\n if (search.trim().length > 0) params.set('search', search.trim())\n const partition = typeof filterValues.partition === 'string' ? filterValues.partition : ''\n if (partition) params.set('partition', partition)\n const tags = Array.isArray(filterValues.tags) ? filterValues.tags : []\n if (tags.length > 0) params.set('tags', tags.join(','))\n if (sorting.length > 0) {\n const primary = sorting[0]\n params.set('sortField', primary.id)\n params.set('sortDir', primary.desc ? 'desc' : 'asc')\n }\n const call = await apiCall<AttachmentLibraryResponse>(`/api/attachments/library?${params.toString()}`)\n if (!call.ok || !call.result) {\n const message = call.result?.error || t('attachments.library.errors.load', 'Failed to load attachments.')\n throw new Error(message)\n }\n return call.result\n },\n })\n\n const partitions = data?.partitions ?? []\n const availableTags = data?.availableTags ?? []\n\n const filters = React.useMemo<FilterDef[]>(() => {\n const partitionOptions = partitions.map((entry) => ({\n value: entry.code,\n label: entry.title || entry.code,\n }))\n return [\n {\n id: 'partition',\n label: t('attachments.library.filters.partition', 'Partition'),\n type: 'select',\n options: partitionOptions,\n },\n {\n id: 'tags',\n label: t('attachments.library.filters.tags', 'Tags'),\n type: 'tags',\n placeholder: t('attachments.library.filters.tagsPlaceholder', 'Filter by tag'),\n options: availableTags.map((tag) => ({ value: tag, label: tag })),\n },\n ]\n }, [availableTags, partitions, t])\n\n const items = data?.items ?? []\n\n const columns = React.useMemo<ColumnDef<AttachmentRow>[]>(() => {\n return [\n {\n id: 'preview',\n header: '',\n enableSorting: false,\n cell: ({ row }) => {\n const value = row.original\n if (value.thumbnailUrl) {\n return (\n <div className=\"h-16 w-16 overflow-hidden rounded border bg-muted\">\n <img\n src={value.thumbnailUrl}\n alt={value.fileName}\n className=\"h-full w-full object-cover\"\n loading=\"lazy\"\n />\n </div>\n )\n }\n const placeholder = resolveAttachmentPlaceholder(value.mimeType, value.fileName)\n const PlaceholderIcon = placeholder.icon\n return (\n <div className=\"flex h-16 w-16 flex-col items-center justify-center rounded border bg-muted text-[10px] font-semibold uppercase text-muted-foreground\">\n <PlaceholderIcon className=\"mb-1 h-5 w-5 text-muted-foreground\" aria-hidden />\n {placeholder.label}\n </div>\n )\n },\n },\n {\n id: 'fileName',\n accessorKey: 'fileName',\n header: t('attachments.library.table.file', 'File'),\n cell: ({ row }) => {\n const value = row.original\n return (\n <div className=\"space-y-1 min-w-0 max-w-[280px]\">\n <div className=\"font-medium truncate\" title={value.fileName}>\n {value.fileName}\n </div>\n <div className=\"text-xs text-muted-foreground\">\n {formatFileSize(value.fileSize)} \u2022 {value.mimeType || 'application/octet-stream'}\n </div>\n <div className=\"text-xs text-muted-foreground line-clamp-2\">\n {value.content?.trim()\n ? value.content\n : t('attachments.library.metadata.noContent', 'No text extracted')}\n </div>\n </div>\n )\n },\n },\n {\n id: 'tags',\n accessorKey: 'tags',\n header: t('attachments.library.table.tags', 'Tags'),\n enableSorting: false,\n cell: ({ row }) => {\n const tags = row.original.tags ?? []\n if (!tags.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-wrap gap-1\">\n {tags.map((tag) => (\n <Badge key={tag} variant=\"outline\">\n {tag}\n </Badge>\n ))}\n </div>\n )\n },\n },\n {\n id: 'assignments',\n accessorKey: 'assignments',\n header: t('attachments.library.table.assignments', 'Assignments'),\n enableSorting: false,\n cell: ({ row }) => {\n const assignments = filterLibraryAssignments(row.original.assignments)\n if (!assignments.length) return <span className=\"text-xs text-muted-foreground\">\u2014</span>\n return (\n <div className=\"flex flex-col gap-1\">\n {assignments.map((assignment) => {\n const label = assignment.label?.trim() || assignment.id\n const hideType =\n assignment.type === (E as any).catalog?.catalog_product ||\n assignment.type === (E as any).catalog?.catalog_product_variant\n const content = hideType ? label : `${assignment.type}: ${label}`\n return assignment.href ? (\n <a\n key={`${assignment.type}-${assignment.id}-${assignment.href}`}\n href={assignment.href}\n className=\"text-sm text-blue-600 underline\"\n target=\"_blank\"\n rel=\"noreferrer\"\n >\n {content}\n </a>\n ) : (\n <div key={`${assignment.type}-${assignment.id}`} className=\"text-sm\">\n {content}\n </div>\n )\n })}\n </div>\n )\n },\n },\n {\n id: 'partitionCode',\n accessorKey: 'partitionCode',\n header: t('attachments.library.table.partition', 'Partition'),\n cell: ({ row }) => (\n <div className=\"text-sm text-muted-foreground\">\n {row.original.partitionTitle ?? row.original.partitionCode}\n </div>\n ),\n },\n {\n id: 'createdAt',\n accessorKey: 'createdAt',\n header: t('attachments.library.table.created', 'Created'),\n cell: ({ row }) => {\n const createdAt = row.original.createdAt\n return (\n <div className=\"text-sm text-muted-foreground\">\n {createdAt ? humanDate(createdAt) : '\u2014'}\n </div>\n )\n },\n },\n {\n id: 'download',\n header: t('attachments.library.table.download', 'Download'),\n enableSorting: false,\n cell: ({ row }) => {\n const downloadPath = buildAttachmentFileUrl(row.original.id, { download: true })\n const absolute = resolveAbsoluteUrl(downloadPath)\n return (\n <Button variant=\"ghost\" size=\"icon\" asChild>\n <a href={absolute} download aria-label={t('attachments.library.table.download', 'Download')}>\n <Download className=\"h-4 w-4\" />\n </a>\n </Button>\n )\n },\n },\n ]\n }, [t])\n\n const openMetadataDialog = React.useCallback((row: AttachmentRow) => {\n setSelectedRow(row)\n setMetadataDialogOpen(true)\n }, [])\n\n const openDeleteDialog = React.useCallback((row: AttachmentRow) => {\n setDeleteTarget(row)\n setDeleteDialogOpen(true)\n }, [])\n\n const handleMetadataSave = React.useCallback(\n async (id: string, payload: AttachmentMetadataSavePayload) => {\n try {\n const body: Record<string, unknown> = {\n tags: payload.tags,\n assignments: payload.assignments,\n }\n if (payload.customFields && Object.keys(payload.customFields).length) {\n body.customFields = payload.customFields\n }\n const call = await apiCall<{ error?: string }>(`/api/attachments/library/${encodeURIComponent(id)}`, {\n method: 'PATCH',\n headers: { 'content-type': 'application/json' },\n body: JSON.stringify(body),\n })\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.metadata.error', 'Failed to update metadata.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.metadata.success', 'Attachment updated.'), 'success')\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n setMetadataDialogOpen(false)\n } catch (err: any) {\n flash(err?.message || t('attachments.library.metadata.error', 'Failed to update metadata.'), 'error')\n }\n },\n [queryClient, t],\n )\n\n const handleUploadCompleted = React.useCallback(async () => {\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n }, [queryClient])\n\n const handleDelete = React.useCallback(async () => {\n if (!deleteTarget) return\n try {\n setDeletePending(true)\n const call = await apiCall<{ error?: string }>(\n `/api/attachments/library/${encodeURIComponent(deleteTarget.id)}`,\n { method: 'DELETE' },\n )\n if (!call.ok) {\n const message =\n call.result?.error || t('attachments.library.errors.delete', 'Failed to delete attachment.')\n flash(message, 'error')\n return\n }\n flash(t('attachments.library.messages.deleted', 'Attachment removed.'), 'success')\n if (selectedRow?.id === deleteTarget.id) {\n setSelectedRow(null)\n setMetadataDialogOpen(false)\n }\n setDeleteDialogOpen(false)\n setDeleteTarget(null)\n await queryClient.invalidateQueries({ queryKey: ['attachments-library'], exact: false })\n } catch (err: any) {\n flash(err?.message || t('attachments.library.errors.delete', 'Failed to delete attachment.'), 'error')\n } finally {\n setDeletePending(false)\n }\n }, [deleteTarget, queryClient, selectedRow, t])\n\n const total = data?.total ?? 0\n const totalPages = data?.totalPages ?? 0\n return (\n <>\n <DataTable<AttachmentRow>\n title={t('attachments.library.title', 'Attachments')}\n refreshButton={{\n label: t('attachments.library.actions.refresh', 'Refresh'),\n onRefresh: () => { void refetch() },\n isRefreshing: isLoading,\n }}\n actions={(\n <Button onClick={() => setUploadDialogOpen(true)}>\n {t('attachments.library.actions.upload', 'Upload')}\n </Button>\n )}\n columns={columns}\n data={items}\n sorting={sorting}\n onSortingChange={setSorting}\n rowActions={(row) => (\n <RowActions\n items={[\n {\n id: 'open',\n label: t('attachments.library.actions.open', 'Open'),\n onSelect: () => {\n if (!row.url) return\n window.open(row.url, '_blank', 'noopener,noreferrer')\n },\n },\n {\n id: 'edit',\n label: t('attachments.library.actions.edit', 'Edit metadata'),\n onSelect: () => openMetadataDialog(row),\n },\n {\n id: 'copy-url',\n label: t('attachments.library.actions.copyUrl', 'Copy URL'),\n onSelect: () => {\n if (!row.url) {\n flash(t('attachments.library.actions.copyError', 'Unable to copy link.'), 'error')\n return\n }\n const absolute = resolveAbsoluteUrl(row.url)\n navigator.clipboard\n .writeText(absolute)\n .then(() =>\n flash(\n t('attachments.library.actions.copied', 'Link copied.'),\n 'success',\n ),\n )\n .catch(() =>\n flash(\n t('attachments.library.actions.copyError', 'Unable to copy link.'),\n 'error',\n ),\n )\n },\n },\n {\n id: 'delete',\n label: t('attachments.library.actions.delete', 'Delete'),\n destructive: true,\n onSelect: () => openDeleteDialog(row),\n },\n ]}\n />\n )}\n onRowClick={(row) => openMetadataDialog(row)}\n isLoading={isLoading}\n error={error?.message}\n emptyState={\n <div className=\"py-10 text-center text-sm text-muted-foreground\">\n {t('attachments.library.table.empty', 'No attachments found.')}\n </div>\n }\n searchValue={search}\n onSearchChange={(value) => {\n setPage(1)\n setSearch(value)\n }}\n searchPlaceholder={t('attachments.library.table.search', 'Search files\u2026')}\n filters={filters}\n filterValues={filterValues}\n onFiltersApply={(values) => {\n setFilterValues(values)\n setPage(1)\n }}\n onFiltersClear={() => {\n setFilterValues({})\n setPage(1)\n }}\n pagination={{\n page,\n pageSize: PAGE_SIZE,\n total,\n totalPages,\n onPageChange: (next) => setPage(next),\n }}\n />\n <AttachmentMetadataDialog\n open={metadataDialogOpen}\n onOpenChange={setMetadataDialogOpen}\n item={selectedRow}\n availableTags={availableTags}\n onSave={handleMetadataSave}\n />\n <AttachmentDeleteDialog\n open={deleteDialogOpen}\n onOpenChange={setDeleteDialogOpen}\n fileName={deleteTarget?.fileName}\n onConfirm={handleDelete}\n isDeleting={deletePending}\n />\n <AttachmentUploadDialog\n open={uploadDialogOpen}\n onOpenChange={setUploadDialogOpen}\n partitions={partitions}\n availableTags={availableTags}\n onUploaded={handleUploadCompleted}\n />\n </>\n )\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -91,7 +91,7 @@ async function POST(req, context) {
91
91
  } : null,
92
92
  executionTime: result.executionTime,
93
93
  error: result.error,
94
- logId: result.logId
94
+ logId: result.logId ? String(result.logId) : void 0
95
95
  };
96
96
  const status = result.success ? 200 : result.error === "Rule not found" ? 404 : 200;
97
97
  return NextResponse.json(response, { status });
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/modules/business_rules/api/execute/%5BruleId%5D/route.ts"],
4
- "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport * as ruleEngine from '../../../lib/rule-engine'\n\nconst executeByIdRequestSchema = z.object({\n data: z.any(),\n dryRun: z.boolean().optional().default(false),\n entityType: z.string().optional(),\n entityId: z.string().optional(),\n eventType: z.string().optional(),\n})\n\nconst executeByIdResponseSchema = z.object({\n success: z.boolean(),\n ruleId: z.string(),\n ruleName: z.string(),\n conditionResult: z.boolean(),\n actionsExecuted: z.object({\n success: z.boolean(),\n results: z.array(z.object({\n type: z.string(),\n success: z.boolean(),\n error: z.string().optional(),\n })),\n }).nullable(),\n executionTime: z.number(),\n error: z.string().optional(),\n logId: z.string().optional(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n POST: { requireAuth: true, requireFeatures: ['business_rules.execute'] },\n}\n\nexport const metadata = routeMetadata\n\ninterface RouteContext {\n params: Promise<{ ruleId: string }>\n}\n\nexport async function POST(req: Request, context: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const params = await context.params\n const ruleId = params.ruleId\n\n if (!ruleId || !z.uuid().safeParse(ruleId).success) {\n return NextResponse.json({ error: 'Invalid rule ID' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n let body: any\n try {\n body = await req.json()\n } catch {\n return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })\n }\n\n const parsed = executeByIdRequestSchema.safeParse(body)\n if (!parsed.success) {\n const errors = parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Validation failed: ${errors.join(', ')}` }, { status: 400 })\n }\n\n const { data, dryRun, entityType, entityId, eventType } = parsed.data\n\n const execContext: ruleEngine.DirectRuleExecutionContext = {\n ruleId,\n data,\n user: {\n id: auth.sub,\n email: auth.email,\n role: (auth.role as string) ?? undefined,\n },\n tenantId: auth.tenantId ?? '',\n organizationId: auth.orgId ?? '',\n executedBy: auth.sub ?? auth.email ?? undefined,\n dryRun,\n entityType,\n entityId,\n eventType,\n }\n\n try {\n const result = await ruleEngine.executeRuleById(em, execContext)\n\n const response = {\n success: result.success,\n ruleId: result.ruleId,\n ruleName: result.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted ? {\n success: result.actionsExecuted.success,\n results: result.actionsExecuted.results.map(ar => ({\n type: ar.action.type,\n success: ar.success,\n error: ar.error,\n })),\n } : null,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId,\n }\n\n // Return appropriate status based on result\n const status = result.success ? 200 : (result.error === 'Rule not found' ? 404 : 200)\n return NextResponse.json(response, { status })\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n return NextResponse.json(\n { error: `Rule execution failed: ${errorMessage}` },\n { status: 500 }\n )\n }\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Execute a specific business rule by ID',\n methods: {\n POST: {\n summary: 'Execute a specific rule by its database UUID',\n 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.',\n pathParams: z.object({\n ruleId: z.string().uuid().describe('The database UUID of the business rule to execute'),\n }),\n requestBody: {\n contentType: 'application/json',\n schema: executeByIdRequestSchema,\n },\n responses: [\n {\n status: 200,\n description: 'Rule executed successfully',\n schema: executeByIdResponseSchema,\n },\n {\n status: 404,\n description: 'Rule not found',\n schema: errorResponseSchema,\n },\n ],\n errors: [\n { status: 400, description: 'Invalid request payload or rule ID', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 500, description: 'Execution error', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport * as ruleEngine from '../../../lib/rule-engine'\n\nconst executeByIdRequestSchema = z.object({\n data: z.any(),\n dryRun: z.boolean().optional().default(false),\n entityType: z.string().optional(),\n entityId: z.string().optional(),\n eventType: z.string().optional(),\n})\n\nconst executeByIdResponseSchema = z.object({\n success: z.boolean(),\n ruleId: z.string(),\n ruleName: z.string(),\n conditionResult: z.boolean(),\n actionsExecuted: z.object({\n success: z.boolean(),\n results: z.array(z.object({\n type: z.string(),\n success: z.boolean(),\n error: z.string().optional(),\n })),\n }).nullable(),\n executionTime: z.number(),\n error: z.string().optional(),\n logId: z.string().optional(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n POST: { requireAuth: true, requireFeatures: ['business_rules.execute'] },\n}\n\nexport const metadata = routeMetadata\n\ninterface RouteContext {\n params: Promise<{ ruleId: string }>\n}\n\nexport async function POST(req: Request, context: RouteContext) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const params = await context.params\n const ruleId = params.ruleId\n\n if (!ruleId || !z.uuid().safeParse(ruleId).success) {\n return NextResponse.json({ error: 'Invalid rule ID' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n let body: any\n try {\n body = await req.json()\n } catch {\n return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })\n }\n\n const parsed = executeByIdRequestSchema.safeParse(body)\n if (!parsed.success) {\n const errors = parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Validation failed: ${errors.join(', ')}` }, { status: 400 })\n }\n\n const { data, dryRun, entityType, entityId, eventType } = parsed.data\n\n const execContext: ruleEngine.DirectRuleExecutionContext = {\n ruleId,\n data,\n user: {\n id: auth.sub,\n email: auth.email,\n role: (auth.role as string) ?? undefined,\n },\n tenantId: auth.tenantId ?? '',\n organizationId: auth.orgId ?? '',\n executedBy: auth.sub ?? auth.email ?? undefined,\n dryRun,\n entityType,\n entityId,\n eventType,\n }\n\n try {\n const result = await ruleEngine.executeRuleById(em, execContext)\n\n const response = {\n success: result.success,\n ruleId: result.ruleId,\n ruleName: result.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted ? {\n success: result.actionsExecuted.success,\n results: result.actionsExecuted.results.map(ar => ({\n type: ar.action.type,\n success: ar.success,\n error: ar.error,\n })),\n } : null,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId ? String(result.logId) : undefined,\n }\n\n // Return appropriate status based on result\n const status = result.success ? 200 : (result.error === 'Rule not found' ? 404 : 200)\n return NextResponse.json(response, { status })\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n return NextResponse.json(\n { error: `Rule execution failed: ${errorMessage}` },\n { status: 500 }\n )\n }\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Execute a specific business rule by ID',\n methods: {\n POST: {\n summary: 'Execute a specific rule by its database UUID',\n 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.',\n pathParams: z.object({\n ruleId: z.string().uuid().describe('The database UUID of the business rule to execute'),\n }),\n requestBody: {\n contentType: 'application/json',\n schema: executeByIdRequestSchema,\n },\n responses: [\n {\n status: 200,\n description: 'Rule executed successfully',\n schema: executeByIdResponseSchema,\n },\n {\n status: 404,\n description: 'Rule not found',\n schema: errorResponseSchema,\n },\n ],\n errors: [\n { status: 400, description: 'Invalid request payload or rule ID', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 500, description: 'Execution error', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -94,11 +94,11 @@ async function POST(req) {
94
94
  } : null,
95
95
  executionTime: r.executionTime,
96
96
  error: r.error,
97
- logId: r.logId
97
+ logId: r.logId ? String(r.logId) : void 0
98
98
  })),
99
99
  totalExecutionTime: result.totalExecutionTime,
100
100
  errors: result.errors,
101
- logIds: result.logIds
101
+ logIds: result.logIds?.map((entry) => String(entry))
102
102
  };
103
103
  return NextResponse.json(response, { status: 200 });
104
104
  } catch (error) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/business_rules/api/execute/route.ts"],
4
- "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EventBus } from '@open-mercato/events'\nimport { ruleEngineContextSchema } from '../../data/validators'\nimport * as ruleEngine from '../../lib/rule-engine'\n\nconst executeRequestSchema = z.object({\n entityType: z.string().min(1, 'entityType is required'),\n entityId: z.string().optional(),\n eventType: z.string().optional(),\n data: z.any(),\n dryRun: z.boolean().optional().default(false),\n})\n\nconst executeResponseSchema = z.object({\n allowed: z.boolean(),\n executedRules: z.array(z.object({\n ruleId: z.string(),\n ruleName: z.string(),\n conditionResult: z.boolean(),\n executionTime: z.number(),\n error: z.string().optional(),\n })),\n totalExecutionTime: z.number(),\n errors: z.array(z.string()).optional(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n POST: { requireAuth: true, requireFeatures: ['business_rules.execute'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function POST(req: Request) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n let eventBus: EventBus | null = null\n try {\n eventBus = container.resolve('eventBus') as EventBus\n } catch {\n eventBus = null\n }\n\n let body: any\n try {\n body = await req.json()\n } catch {\n return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })\n }\n\n const parsed = executeRequestSchema.safeParse(body)\n if (!parsed.success) {\n const errors = parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Validation failed: ${errors.join(', ')}` }, { status: 400 })\n }\n\n const { entityType, entityId, eventType, data, dryRun } = parsed.data\n\n const context: ruleEngine.RuleEngineContext = {\n entityType,\n entityId,\n eventType,\n data,\n user: {\n id: auth.sub,\n email: auth.email,\n role: (auth.role as string) ?? undefined,\n },\n tenantId: auth.tenantId ?? '',\n organizationId: auth.orgId ?? '',\n executedBy: auth.sub ?? auth.email ?? null,\n dryRun,\n }\n\n const validation = ruleEngineContextSchema.safeParse(context)\n if (!validation.success) {\n const errors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Invalid execution context: ${errors.join(', ')}` }, { status: 400 })\n }\n\n try {\n const result = await ruleEngine.executeRules(em, context, { eventBus })\n\n const response = {\n allowed: result.allowed,\n executedRules: result.executedRules.map(r => ({\n ruleId: r.rule.ruleId,\n ruleName: r.rule.ruleName,\n ruleType: r.rule.ruleType,\n conditionResult: r.conditionResult,\n actionsExecuted: r.actionsExecuted ? {\n success: r.actionsExecuted.success,\n results: r.actionsExecuted.results.map(ar => ({\n type: ar.action.type,\n success: ar.success,\n error: ar.error,\n })),\n } : null,\n executionTime: r.executionTime,\n error: r.error,\n logId: r.logId,\n })),\n totalExecutionTime: result.totalExecutionTime,\n errors: result.errors,\n logIds: result.logIds,\n }\n\n return NextResponse.json(response, { status: 200 })\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n return NextResponse.json(\n { error: `Rule execution failed: ${errorMessage}` },\n { status: 500 }\n )\n }\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Execute business rules',\n methods: {\n POST: {\n summary: 'Execute rules for given context',\n description: 'Manually executes applicable business rules for the specified entity type, event, and data. Supports dry-run mode to test rules without executing actions.',\n requestBody: {\n contentType: 'application/json',\n schema: executeRequestSchema,\n },\n responses: [\n {\n status: 200,\n description: 'Rules executed successfully',\n schema: executeResponseSchema,\n },\n ],\n errors: [\n { status: 400, description: 'Invalid request payload', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 500, description: 'Execution error', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EventBus } from '@open-mercato/events'\nimport { ruleEngineContextSchema } from '../../data/validators'\nimport * as ruleEngine from '../../lib/rule-engine'\n\nconst executeRequestSchema = z.object({\n entityType: z.string().min(1, 'entityType is required'),\n entityId: z.string().optional(),\n eventType: z.string().optional(),\n data: z.any(),\n dryRun: z.boolean().optional().default(false),\n})\n\nconst executeResponseSchema = z.object({\n allowed: z.boolean(),\n executedRules: z.array(z.object({\n ruleId: z.string(),\n ruleName: z.string(),\n conditionResult: z.boolean(),\n executionTime: z.number(),\n error: z.string().optional(),\n })),\n totalExecutionTime: z.number(),\n errors: z.array(z.string()).optional(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n POST: { requireAuth: true, requireFeatures: ['business_rules.execute'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function POST(req: Request) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n let eventBus: EventBus | null = null\n try {\n eventBus = container.resolve('eventBus') as EventBus\n } catch {\n eventBus = null\n }\n\n let body: any\n try {\n body = await req.json()\n } catch {\n return NextResponse.json({ error: 'Invalid JSON body' }, { status: 400 })\n }\n\n const parsed = executeRequestSchema.safeParse(body)\n if (!parsed.success) {\n const errors = parsed.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Validation failed: ${errors.join(', ')}` }, { status: 400 })\n }\n\n const { entityType, entityId, eventType, data, dryRun } = parsed.data\n\n const context: ruleEngine.RuleEngineContext = {\n entityType,\n entityId,\n eventType,\n data,\n user: {\n id: auth.sub,\n email: auth.email,\n role: (auth.role as string) ?? undefined,\n },\n tenantId: auth.tenantId ?? '',\n organizationId: auth.orgId ?? '',\n executedBy: auth.sub ?? auth.email ?? null,\n dryRun,\n }\n\n const validation = ruleEngineContextSchema.safeParse(context)\n if (!validation.success) {\n const errors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return NextResponse.json({ error: `Invalid execution context: ${errors.join(', ')}` }, { status: 400 })\n }\n\n try {\n const result = await ruleEngine.executeRules(em, context, { eventBus })\n\n const response = {\n allowed: result.allowed,\n executedRules: result.executedRules.map(r => ({\n ruleId: r.rule.ruleId,\n ruleName: r.rule.ruleName,\n ruleType: r.rule.ruleType,\n conditionResult: r.conditionResult,\n actionsExecuted: r.actionsExecuted ? {\n success: r.actionsExecuted.success,\n results: r.actionsExecuted.results.map(ar => ({\n type: ar.action.type,\n success: ar.success,\n error: ar.error,\n })),\n } : null,\n executionTime: r.executionTime,\n error: r.error,\n logId: r.logId ? String(r.logId) : undefined,\n })),\n totalExecutionTime: result.totalExecutionTime,\n errors: result.errors,\n logIds: result.logIds?.map((entry) => String(entry)),\n }\n\n return NextResponse.json(response, { status: 200 })\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n return NextResponse.json(\n { error: `Rule execution failed: ${errorMessage}` },\n { status: 500 }\n )\n }\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Execute business rules',\n methods: {\n POST: {\n summary: 'Execute rules for given context',\n description: 'Manually executes applicable business rules for the specified entity type, event, and data. Supports dry-run mode to test rules without executing actions.',\n requestBody: {\n contentType: 'application/json',\n schema: executeRequestSchema,\n },\n responses: [\n {\n status: 200,\n description: 'Rules executed successfully',\n schema: executeResponseSchema,\n },\n ],\n errors: [\n { status: 400, description: 'Invalid request payload', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 500, description: 'Execution error', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -67,7 +67,7 @@ async function GET(_req, ctx) {
67
67
  return NextResponse.json({ error: "Log entry not found" }, { status: 404 });
68
68
  }
69
69
  const response = {
70
- id: log.id,
70
+ id: String(log.id),
71
71
  rule: {
72
72
  id: log.rule.id,
73
73
  ruleId: log.rule.ruleId,
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../../src/modules/business_rules/api/logs/%5Bid%5D/route.ts"],
4
- "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport { RuleExecutionLog } from '../../../data/entities'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { executionResultSchema } from '../../../data/validators'\nimport { findOneWithDecryption } from '@open-mercato/shared/lib/encryption/find'\n\nconst paramsSchema = z.object({\n id: z.string().regex(/^\\d+$/, 'Invalid log id'),\n})\n\nconst logDetailSchema = z.object({\n id: z.string(),\n rule: z.object({\n id: z.string().uuid(),\n ruleId: z.string(),\n ruleName: z.string(),\n ruleType: z.string(),\n entityType: z.string(),\n }),\n entityId: z.string().uuid(),\n entityType: z.string(),\n executionResult: executionResultSchema,\n inputContext: z.any().nullable(),\n outputContext: z.any().nullable(),\n errorMessage: z.string().nullable(),\n executionTimeMs: z.number(),\n executedAt: z.string(),\n tenantId: z.string().uuid(),\n organizationId: z.string().uuid().nullable(),\n executedBy: z.string().nullable(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['business_rules.view_logs'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function GET(_req: Request, ctx: { params?: { id?: string } }) {\n const auth = await getAuthFromRequest(_req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const parse = paramsSchema.safeParse({ id: ctx.params?.id })\n if (!parse.success) {\n return NextResponse.json({ error: 'Invalid log id' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n const filters: Record<string, any> = {\n id: parse.data.id,\n tenantId: auth.tenantId,\n }\n\n // Organization filter is optional for logs (can be null)\n if (auth.orgId) {\n filters.organizationId = auth.orgId\n }\n\n const log = await findOneWithDecryption(\n em,\n RuleExecutionLog,\n filters,\n {\n populate: ['rule'],\n },\n { tenantId: auth.tenantId ?? null, organizationId: auth.orgId ?? null },\n )\n\n if (!log) {\n return NextResponse.json({ error: 'Log entry not found' }, { status: 404 })\n }\n\n const response = {\n id: log.id,\n rule: {\n id: log.rule.id,\n ruleId: log.rule.ruleId,\n ruleName: log.rule.ruleName,\n ruleType: log.rule.ruleType,\n entityType: log.rule.entityType,\n },\n entityId: log.entityId,\n entityType: log.entityType,\n executionResult: log.executionResult,\n inputContext: log.inputContext ?? null,\n outputContext: log.outputContext ?? null,\n errorMessage: log.errorMessage ?? null,\n executionTimeMs: log.executionTimeMs,\n executedAt: log.executedAt.toISOString(),\n tenantId: log.tenantId,\n organizationId: log.organizationId ?? null,\n executedBy: log.executedBy ?? null,\n }\n\n return NextResponse.json(response)\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Rule execution log detail',\n methods: {\n GET: {\n summary: 'Get execution log detail',\n description: 'Returns detailed information about a specific rule execution, including full context and results.',\n responses: [\n { status: 200, description: 'Log entry details', schema: logDetailSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid log id', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 404, description: 'Log entry not found', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport { RuleExecutionLog } from '../../../data/entities'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { executionResultSchema } from '../../../data/validators'\nimport { findOneWithDecryption } from '@open-mercato/shared/lib/encryption/find'\n\nconst paramsSchema = z.object({\n id: z.string().regex(/^\\d+$/, 'Invalid log id'),\n})\n\nconst logDetailSchema = z.object({\n id: z.string(),\n rule: z.object({\n id: z.string().uuid(),\n ruleId: z.string(),\n ruleName: z.string(),\n ruleType: z.string(),\n entityType: z.string(),\n }),\n entityId: z.string().uuid(),\n entityType: z.string(),\n executionResult: executionResultSchema,\n inputContext: z.any().nullable(),\n outputContext: z.any().nullable(),\n errorMessage: z.string().nullable(),\n executionTimeMs: z.number(),\n executedAt: z.string(),\n tenantId: z.string().uuid(),\n organizationId: z.string().uuid().nullable(),\n executedBy: z.string().nullable(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['business_rules.view_logs'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function GET(_req: Request, ctx: { params?: { id?: string } }) {\n const auth = await getAuthFromRequest(_req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const parse = paramsSchema.safeParse({ id: ctx.params?.id })\n if (!parse.success) {\n return NextResponse.json({ error: 'Invalid log id' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n const filters: Record<string, any> = {\n id: parse.data.id,\n tenantId: auth.tenantId,\n }\n\n // Organization filter is optional for logs (can be null)\n if (auth.orgId) {\n filters.organizationId = auth.orgId\n }\n\n const log = await findOneWithDecryption(\n em,\n RuleExecutionLog,\n filters,\n {\n populate: ['rule'],\n },\n { tenantId: auth.tenantId ?? null, organizationId: auth.orgId ?? null },\n )\n\n if (!log) {\n return NextResponse.json({ error: 'Log entry not found' }, { status: 404 })\n }\n\n const response = {\n id: String(log.id),\n rule: {\n id: log.rule.id,\n ruleId: log.rule.ruleId,\n ruleName: log.rule.ruleName,\n ruleType: log.rule.ruleType,\n entityType: log.rule.entityType,\n },\n entityId: log.entityId,\n entityType: log.entityType,\n executionResult: log.executionResult,\n inputContext: log.inputContext ?? null,\n outputContext: log.outputContext ?? null,\n errorMessage: log.errorMessage ?? null,\n executionTimeMs: log.executionTimeMs,\n executedAt: log.executedAt.toISOString(),\n tenantId: log.tenantId,\n organizationId: log.organizationId ?? null,\n executedBy: log.executedBy ?? null,\n }\n\n return NextResponse.json(response)\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Rule execution log detail',\n methods: {\n GET: {\n summary: 'Get execution log detail',\n description: 'Returns detailed information about a specific rule execution, including full context and results.',\n responses: [\n { status: 200, description: 'Log entry details', schema: logDetailSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid log id', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n { status: 404, description: 'Log entry not found', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -114,7 +114,7 @@ async function GET(req) {
114
114
  { tenantId: auth.tenantId ?? null, organizationId: auth.orgId ?? null }
115
115
  );
116
116
  const items = rows.map((log) => ({
117
- id: log.id,
117
+ id: String(log.id),
118
118
  ruleId: log.rule.id,
119
119
  ruleName: log.rule.ruleName,
120
120
  ruleType: log.rule.ruleType,
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/business_rules/api/logs/route.ts"],
4
- "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport { RuleExecutionLog } from '../../data/entities'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { executionResultSchema } from '../../data/validators'\nimport { findAndCountWithDecryption } from '@open-mercato/shared/lib/encryption/find'\n\nconst querySchema = z.looseObject({\n id: z.coerce.bigint().optional(),\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n ruleId: z.string().uuid().optional(),\n entityId: z.string().uuid().optional(),\n entityType: z.string().optional(),\n executionResult: executionResultSchema.optional(),\n executedBy: z.string().optional(),\n executedAtFrom: z.coerce.date().optional(),\n executedAtTo: z.coerce.date().optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional().default('desc'),\n})\n\nconst logListItemSchema = z.object({\n id: z.string(),\n ruleId: z.string(),\n ruleName: z.string(),\n ruleType: z.string(),\n entityId: z.string().uuid(),\n entityType: z.string(),\n executionResult: executionResultSchema,\n inputContext: z.any().nullable(),\n outputContext: z.any().nullable(),\n errorMessage: z.string().nullable(),\n executionTimeMs: z.number(),\n executedAt: z.string(),\n tenantId: z.string().uuid(),\n organizationId: z.string().uuid().nullable(),\n executedBy: z.string().nullable(),\n})\n\nconst logListResponseSchema = z.object({\n items: z.array(logListItemSchema),\n total: z.number().int().nonnegative(),\n totalPages: z.number().int().positive(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['business_rules.view_logs'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function GET(req: Request) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const url = new URL(req.url)\n const parsed = querySchema.safeParse({\n id: url.searchParams.get('id') || undefined,\n page: url.searchParams.get('page') || undefined,\n pageSize: url.searchParams.get('pageSize') || undefined,\n ruleId: url.searchParams.get('ruleId') || undefined,\n entityId: url.searchParams.get('entityId') || undefined,\n entityType: url.searchParams.get('entityType') || undefined,\n executionResult: url.searchParams.get('executionResult') || undefined,\n executedBy: url.searchParams.get('executedBy') || undefined,\n executedAtFrom: url.searchParams.get('executedAtFrom') || undefined,\n executedAtTo: url.searchParams.get('executedAtTo') || undefined,\n sortField: url.searchParams.get('sortField') || undefined,\n sortDir: url.searchParams.get('sortDir') || undefined,\n })\n\n if (!parsed.success) {\n return NextResponse.json({ error: 'Invalid query parameters' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n const { id, page, pageSize, ruleId, entityId, entityType, executionResult, executedBy, executedAtFrom, executedAtTo, sortField, sortDir } = parsed.data\n\n const filters: Record<string, any> = {\n tenantId: auth.tenantId,\n }\n\n // Organization filter is optional for logs (can be null)\n if (auth.orgId) {\n filters.organizationId = auth.orgId\n }\n\n if (id) filters.id = id.toString()\n if (ruleId) filters.rule = { id: ruleId }\n if (entityId) filters.entityId = entityId\n if (entityType) filters.entityType = entityType\n if (executionResult) filters.executionResult = executionResult\n if (executedBy) filters.executedBy = executedBy\n\n // Date range filter\n if (executedAtFrom || executedAtTo) {\n filters.executedAt = {}\n if (executedAtFrom) filters.executedAt.$gte = executedAtFrom\n if (executedAtTo) filters.executedAt.$lte = executedAtTo\n }\n\n const sortFieldMap: Record<string, string> = {\n id: 'id',\n ruleId: 'rule.ruleId',\n entityType: 'entityType',\n executionResult: 'executionResult',\n executionTimeMs: 'executionTimeMs',\n executedAt: 'executedAt',\n }\n\n const orderByField = sortField && sortFieldMap[sortField] ? sortFieldMap[sortField] : 'executedAt'\n const orderBy = { [orderByField]: sortDir }\n\n const [rows, count] = await findAndCountWithDecryption(\n em,\n RuleExecutionLog,\n filters,\n {\n limit: pageSize,\n offset: (page - 1) * pageSize,\n orderBy,\n populate: ['rule'],\n },\n { tenantId: auth.tenantId ?? null, organizationId: auth.orgId ?? null },\n )\n\n const items = rows.map((log) => ({\n id: log.id,\n ruleId: log.rule.id,\n ruleName: log.rule.ruleName,\n ruleType: log.rule.ruleType,\n entityId: log.entityId,\n entityType: log.entityType,\n executionResult: log.executionResult,\n inputContext: log.inputContext ?? null,\n outputContext: log.outputContext ?? null,\n errorMessage: log.errorMessage ?? null,\n executionTimeMs: log.executionTimeMs,\n executedAt: log.executedAt.toISOString(),\n tenantId: log.tenantId,\n organizationId: log.organizationId ?? null,\n executedBy: log.executedBy ?? null,\n }))\n\n const totalPages = Math.max(1, Math.ceil(count / pageSize))\n\n return NextResponse.json({ items, total: count, totalPages })\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Rule execution logs',\n methods: {\n GET: {\n summary: 'List rule execution logs',\n description: 'Returns rule execution history for the current tenant and organization with filtering and pagination. Useful for audit trails and debugging.',\n query: querySchema,\n responses: [\n { status: 200, description: 'Rule execution logs collection', schema: logListResponseSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid query parameters', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { NextResponse } from 'next/server'\nimport { z } from 'zod'\nimport type { OpenApiRouteDoc } from '@open-mercato/shared/lib/openapi'\nimport { getAuthFromRequest } from '@open-mercato/shared/lib/auth/server'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport { RuleExecutionLog } from '../../data/entities'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { executionResultSchema } from '../../data/validators'\nimport { findAndCountWithDecryption } from '@open-mercato/shared/lib/encryption/find'\n\nconst querySchema = z.looseObject({\n id: z.coerce.bigint().optional(),\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n ruleId: z.string().uuid().optional(),\n entityId: z.string().uuid().optional(),\n entityType: z.string().optional(),\n executionResult: executionResultSchema.optional(),\n executedBy: z.string().optional(),\n executedAtFrom: z.coerce.date().optional(),\n executedAtTo: z.coerce.date().optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional().default('desc'),\n})\n\nconst logListItemSchema = z.object({\n id: z.string(),\n ruleId: z.string(),\n ruleName: z.string(),\n ruleType: z.string(),\n entityId: z.string().uuid(),\n entityType: z.string(),\n executionResult: executionResultSchema,\n inputContext: z.any().nullable(),\n outputContext: z.any().nullable(),\n errorMessage: z.string().nullable(),\n executionTimeMs: z.number(),\n executedAt: z.string(),\n tenantId: z.string().uuid(),\n organizationId: z.string().uuid().nullable(),\n executedBy: z.string().nullable(),\n})\n\nconst logListResponseSchema = z.object({\n items: z.array(logListItemSchema),\n total: z.number().int().nonnegative(),\n totalPages: z.number().int().positive(),\n})\n\nconst errorResponseSchema = z.object({\n error: z.string(),\n})\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['business_rules.view_logs'] },\n}\n\nexport const metadata = routeMetadata\n\nexport async function GET(req: Request) {\n const auth = await getAuthFromRequest(req)\n if (!auth) {\n return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })\n }\n\n const url = new URL(req.url)\n const parsed = querySchema.safeParse({\n id: url.searchParams.get('id') || undefined,\n page: url.searchParams.get('page') || undefined,\n pageSize: url.searchParams.get('pageSize') || undefined,\n ruleId: url.searchParams.get('ruleId') || undefined,\n entityId: url.searchParams.get('entityId') || undefined,\n entityType: url.searchParams.get('entityType') || undefined,\n executionResult: url.searchParams.get('executionResult') || undefined,\n executedBy: url.searchParams.get('executedBy') || undefined,\n executedAtFrom: url.searchParams.get('executedAtFrom') || undefined,\n executedAtTo: url.searchParams.get('executedAtTo') || undefined,\n sortField: url.searchParams.get('sortField') || undefined,\n sortDir: url.searchParams.get('sortDir') || undefined,\n })\n\n if (!parsed.success) {\n return NextResponse.json({ error: 'Invalid query parameters' }, { status: 400 })\n }\n\n const container = await createRequestContainer()\n const em = container.resolve('em') as EntityManager\n\n const { id, page, pageSize, ruleId, entityId, entityType, executionResult, executedBy, executedAtFrom, executedAtTo, sortField, sortDir } = parsed.data\n\n const filters: Record<string, any> = {\n tenantId: auth.tenantId,\n }\n\n // Organization filter is optional for logs (can be null)\n if (auth.orgId) {\n filters.organizationId = auth.orgId\n }\n\n if (id) filters.id = id.toString()\n if (ruleId) filters.rule = { id: ruleId }\n if (entityId) filters.entityId = entityId\n if (entityType) filters.entityType = entityType\n if (executionResult) filters.executionResult = executionResult\n if (executedBy) filters.executedBy = executedBy\n\n // Date range filter\n if (executedAtFrom || executedAtTo) {\n filters.executedAt = {}\n if (executedAtFrom) filters.executedAt.$gte = executedAtFrom\n if (executedAtTo) filters.executedAt.$lte = executedAtTo\n }\n\n const sortFieldMap: Record<string, string> = {\n id: 'id',\n ruleId: 'rule.ruleId',\n entityType: 'entityType',\n executionResult: 'executionResult',\n executionTimeMs: 'executionTimeMs',\n executedAt: 'executedAt',\n }\n\n const orderByField = sortField && sortFieldMap[sortField] ? sortFieldMap[sortField] : 'executedAt'\n const orderBy = { [orderByField]: sortDir }\n\n const [rows, count] = await findAndCountWithDecryption(\n em,\n RuleExecutionLog,\n filters,\n {\n limit: pageSize,\n offset: (page - 1) * pageSize,\n orderBy,\n populate: ['rule'],\n },\n { tenantId: auth.tenantId ?? null, organizationId: auth.orgId ?? null },\n )\n\n const items = rows.map((log) => ({\n id: String(log.id),\n ruleId: log.rule.id,\n ruleName: log.rule.ruleName,\n ruleType: log.rule.ruleType,\n entityId: log.entityId,\n entityType: log.entityType,\n executionResult: log.executionResult,\n inputContext: log.inputContext ?? null,\n outputContext: log.outputContext ?? null,\n errorMessage: log.errorMessage ?? null,\n executionTimeMs: log.executionTimeMs,\n executedAt: log.executedAt.toISOString(),\n tenantId: log.tenantId,\n organizationId: log.organizationId ?? null,\n executedBy: log.executedBy ?? null,\n }))\n\n const totalPages = Math.max(1, Math.ceil(count / pageSize))\n\n return NextResponse.json({ items, total: count, totalPages })\n}\n\nexport const openApi: OpenApiRouteDoc = {\n tag: 'Business Rules',\n summary: 'Rule execution logs',\n methods: {\n GET: {\n summary: 'List rule execution logs',\n description: 'Returns rule execution history for the current tenant and organization with filtering and pagination. Useful for audit trails and debugging.',\n query: querySchema,\n responses: [\n { status: 200, description: 'Rule execution logs collection', schema: logListResponseSchema },\n ],\n errors: [\n { status: 400, description: 'Invalid query parameters', schema: errorResponseSchema },\n { status: 401, description: 'Unauthorized', schema: errorResponseSchema },\n ],\n },\n },\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -527,7 +527,7 @@ async function logRuleExecution(em, options) {
527
527
  executedBy: context.executedBy || null
528
528
  });
529
529
  await em.persistAndFlush(log);
530
- return log.id;
530
+ return String(log.id);
531
531
  }
532
532
  async function emitRuleExecutionFailed(eventBus, payload) {
533
533
  if (!eventBus?.emitEvent) return;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/business_rules/lib/rule-engine.ts"],
4
- "sourcesContent": ["import type { EntityManager } from '@mikro-orm/core'\nimport type { EventBus } from '@open-mercato/events'\nimport { BusinessRule, RuleExecutionLog, type RuleType } from '../data/entities'\nimport * as ruleEvaluator from './rule-evaluator'\nimport * as actionExecutor from './action-executor'\nimport type { RuleEvaluationContext } from './rule-evaluator'\nimport type { ActionContext, ActionExecutionOutcome } from './action-executor'\nimport { ruleEngineContextSchema, ruleDiscoveryOptionsSchema, directRuleExecutionContextSchema, ruleIdExecutionContextSchema } from '../data/validators'\n\n/**\n * Constants\n */\nconst DEFAULT_ENTITY_ID = 'unknown'\nconst RULE_TYPE_GUARD = 'GUARD'\nconst EXECUTION_RESULT_ERROR = 'ERROR'\nconst EXECUTION_RESULT_SUCCESS = 'SUCCESS'\nconst EXECUTION_RESULT_FAILURE = 'FAILURE'\n\n/**\n * Execution limits\n */\nconst MAX_RULES_PER_EXECUTION = 100\nconst MAX_SINGLE_RULE_TIMEOUT_MS = 30000 // 30 seconds\nconst MAX_TOTAL_EXECUTION_TIMEOUT_MS = 60000 // 60 seconds\n\n/**\n * Rule execution context\n */\nexport interface RuleEngineContext {\n entityType: string\n entityId?: string\n eventType?: string\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenant?: {\n id?: string\n [key: string]: any\n }\n organization?: {\n id?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n [key: string]: any\n}\n\n/**\n * Single rule execution result\n */\nexport interface RuleExecutionResult {\n rule: BusinessRule\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n logId?: string // Database log ID (if logged)\n}\n\n/**\n * Overall rule engine result\n */\nexport interface RuleEngineResult {\n allowed: boolean\n executedRules: RuleExecutionResult[]\n totalExecutionTime: number\n errors?: string[]\n logIds?: string[]\n}\n\n/**\n * Rule discovery options\n */\nexport interface RuleDiscoveryOptions {\n entityType: string\n eventType?: string\n tenantId: string\n organizationId: string\n ruleType?: RuleType\n}\n\nexport type RuleEngineExecutionOptions = {\n eventBus?: Pick<EventBus, 'emitEvent'> | null\n}\n\ntype RuleExecutionFailedPayload = {\n ruleId: string\n ruleName: string\n entityType?: string | null\n errorMessage?: string | null\n tenantId: string\n organizationId?: string | null\n}\n\n/**\n * Direct rule execution context (for executing a specific rule by ID)\n */\nexport interface DirectRuleExecutionContext {\n ruleId: string // Database UUID of the rule\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n // Optional for logging (falls back to rule's entityType)\n entityType?: string\n entityId?: string\n eventType?: string\n}\n\n/**\n * Direct rule execution result\n */\nexport interface DirectRuleExecutionResult {\n success: boolean\n ruleId: string\n ruleName: string\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n logId?: string\n}\n\n/**\n * Context for executing a rule by its string rule_id identifier\n * Unlike DirectRuleExecutionContext which uses database UUID,\n * this uses the string identifier (e.g., \"workflow_checkout_inventory_available\")\n */\nexport interface RuleIdExecutionContext {\n ruleId: string // String identifier (e.g., \"workflow_checkout_inventory_available\")\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n entityType?: string\n entityId?: string\n eventType?: string\n}\n\n/**\n * Execute a function with a timeout\n */\nasync function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n errorMessage: string\n): Promise<T> {\n let timeoutId: NodeJS.Timeout\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(new Error(`${errorMessage} (timeout: ${timeoutMs}ms)`))\n }, timeoutMs)\n })\n\n try {\n return await Promise.race([promise, timeoutPromise])\n } finally {\n clearTimeout(timeoutId!)\n }\n}\n\n/**\n * Execute all applicable rules for the given context\n */\nexport async function executeRules(\n em: EntityManager,\n context: RuleEngineContext,\n options: RuleEngineExecutionOptions = {}\n): Promise<RuleEngineResult> {\n // Validate input\n const validation = ruleEngineContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n allowed: false,\n executedRules: [],\n totalExecutionTime: 0,\n errors: validationErrors,\n }\n }\n\n const startTime = Date.now()\n const executedRules: RuleExecutionResult[] = []\n const errors: string[] = []\n const logIds: string[] = []\n\n try {\n // Discover applicable rules\n const rules = await findApplicableRules(em, {\n entityType: context.entityType,\n eventType: context.eventType,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n })\n\n // Check rule count limit\n if (rules.length > MAX_RULES_PER_EXECUTION) {\n errors.push(\n `Rule count limit exceeded: ${rules.length} rules found, maximum is ${MAX_RULES_PER_EXECUTION}`\n )\n return {\n allowed: false,\n executedRules: [],\n totalExecutionTime: Date.now() - startTime,\n errors,\n }\n }\n\n // Rules already sorted by database query (priority DESC, ruleId ASC)\n // Execute each rule with total timeout\n const executionPromise = (async () => {\n for (const rule of rules) {\n try {\n const ruleResult = await executeSingleRule(em, rule, context, options)\n executedRules.push(ruleResult)\n\n if (ruleResult.logId) {\n logIds.push(ruleResult.logId)\n }\n\n if (ruleResult.error) {\n errors.push(\n `Rule execution failed [ruleId=${rule.ruleId}, type=${rule.ruleType}, entityType=${context.entityType}]: ${ruleResult.error}`\n )\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n errors.push(\n `Unexpected error in rule execution [ruleId=${rule.ruleId}, type=${rule.ruleType}]: ${errorMessage}`\n )\n\n if (!context.dryRun) {\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n executedRules.push({\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: 0,\n error: errorMessage,\n })\n }\n }\n })()\n\n // Execute with timeout\n await withTimeout(\n executionPromise,\n MAX_TOTAL_EXECUTION_TIMEOUT_MS,\n `Total rule execution timeout [entityType=${context.entityType}]`\n )\n\n // Determine overall allowed status\n // For GUARD rules: all must pass for operation to be allowed\n const guardRules = executedRules.filter((r) => r.rule.ruleType === RULE_TYPE_GUARD)\n const allowed = guardRules.length === 0 || guardRules.every((r) => r.conditionResult)\n\n const totalExecutionTime = Date.now() - startTime\n\n return {\n allowed,\n executedRules,\n totalExecutionTime,\n errors: errors.length > 0 ? errors : undefined,\n logIds: logIds.length > 0 ? logIds : undefined,\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n const stack = error instanceof Error ? error.stack : undefined\n errors.push(\n `Critical rule engine error [entityType=${context.entityType}, entityId=${context.entityId || 'unknown'}]: ${errorMessage}${stack ? `\\nStack: ${stack}` : ''}`\n )\n\n const totalExecutionTime = Date.now() - startTime\n\n return {\n allowed: false,\n executedRules,\n totalExecutionTime,\n errors,\n }\n }\n}\n\n/**\n * Execute a single rule\n */\nexport async function executeSingleRule(\n em: EntityManager,\n rule: BusinessRule,\n context: RuleEngineContext,\n options: RuleEngineExecutionOptions = {}\n): Promise<RuleExecutionResult> {\n const startTime = Date.now()\n\n try {\n // Wrap execution in timeout\n const executeWithTimeout = async () => {\n // Build evaluation context\n const evalContext: RuleEvaluationContext = {\n entityType: context.entityType,\n entityId: context.entityId,\n eventType: context.eventType,\n user: context.user,\n tenant: context.tenant,\n organization: context.organization,\n }\n\n // Evaluate rule conditions\n const result = await ruleEvaluator.evaluateSingleRule(rule, context.data, evalContext)\n\n // Check if evaluation completed (not just if conditions passed)\n if (!result.evaluationCompleted) {\n const executionTime = Date.now() - startTime\n\n let logId: string | undefined\n\n // Log failure if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: result.error,\n })\n\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage: result.error ?? null,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n return {\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: result.error,\n logId,\n }\n }\n\n // Evaluation completed successfully - determine which actions to execute\n const actions = result.conditionsPassed ? rule.successActions : rule.failureActions\n\n let actionsExecuted: ActionExecutionOutcome | null = null\n\n if (actions && Array.isArray(actions) && actions.length > 0) {\n // Build action context\n const actionContext: ActionContext = {\n ...evalContext,\n data: context.data,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n }\n\n // Execute actions\n actionsExecuted = await actionExecutor.executeActions(actions, actionContext)\n }\n\n const executionTime = Date.now() - startTime\n\n let logId: string | undefined\n\n // Log execution if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: result.conditionsPassed,\n actionsExecuted,\n executionTime,\n })\n }\n\n return {\n rule,\n conditionResult: result.conditionsPassed,\n actionsExecuted,\n executionTime,\n logId,\n }\n }\n\n // Execute with single rule timeout\n return await withTimeout(\n executeWithTimeout(),\n MAX_SINGLE_RULE_TIMEOUT_MS,\n `Single rule execution timeout [ruleId=${rule.ruleId}]`\n )\n } catch (error) {\n const executionTime = Date.now() - startTime\n const errorMessage = error instanceof Error ? error.message : String(error)\n const enhancedError = `Failed to execute rule [ruleId=${rule.ruleId}, entityType=${context.entityType}]: ${errorMessage}`\n\n let logId: string | undefined\n\n // Log error if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: enhancedError,\n })\n\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage: enhancedError,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n return {\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: enhancedError,\n logId,\n }\n }\n}\n\n/**\n * Find all applicable rules for the given criteria\n */\nexport async function findApplicableRules(\n em: EntityManager,\n options: RuleDiscoveryOptions\n): Promise<BusinessRule[]> {\n // Validate input\n ruleDiscoveryOptionsSchema.parse(options)\n\n const { entityType, eventType, tenantId, organizationId, ruleType } = options\n\n const where: Partial<BusinessRule> = {\n entityType,\n tenantId,\n organizationId,\n enabled: true,\n deletedAt: null,\n }\n\n if (eventType) {\n where.eventType = eventType\n }\n\n if (ruleType) {\n where.ruleType = ruleType\n }\n\n const rules = await em.find(BusinessRule, where, {\n orderBy: { priority: 'DESC', ruleId: 'ASC' },\n })\n\n // Filter by effective date range\n const now = new Date()\n return rules.filter((rule) => {\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return false\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return false\n }\n return true\n })\n}\n\n/**\n * Execute a specific rule by its database UUID\n * This bypasses the entityType/eventType discovery mechanism and directly executes the rule\n */\nexport async function executeRuleById(\n em: EntityManager,\n context: DirectRuleExecutionContext\n): Promise<DirectRuleExecutionResult> {\n const startTime = Date.now()\n\n // Validate input\n const validation = directRuleExecutionContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Validation failed: ${validationErrors.join(', ')}`,\n }\n }\n\n // Fetch rule by ID with tenant/org validation\n const rule = await em.findOne(BusinessRule, {\n id: context.ruleId,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n deletedAt: null,\n })\n\n if (!rule) {\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule not found',\n }\n }\n\n if (!rule.enabled) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule is disabled',\n }\n }\n\n // Check effective date range\n const now = new Date()\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule is not yet effective (starts ${rule.effectiveFrom.toISOString()})`,\n }\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule has expired (ended ${rule.effectiveTo.toISOString()})`,\n }\n }\n\n // Build RuleEngineContext (use provided entityType or fall back to rule's)\n const engineContext: RuleEngineContext = {\n entityType: context.entityType || rule.entityType,\n entityId: context.entityId,\n eventType: context.eventType || rule.eventType || undefined,\n data: context.data,\n user: context.user,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy,\n dryRun: context.dryRun,\n }\n\n // Execute via existing executeSingleRule\n const result = await executeSingleRule(em, rule, engineContext)\n\n return {\n success: !result.error,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId,\n }\n}\n\n/**\n * Execute a rule by its string rule_id identifier\n * Looks up rule by rule_id (string column) + tenant_id (unique constraint)\n * This is useful for workflow conditions that reference rules by their string identifiers\n */\nexport async function executeRuleByRuleId(\n em: EntityManager,\n context: RuleIdExecutionContext\n): Promise<DirectRuleExecutionResult> {\n const startTime = Date.now()\n\n // Validate input\n const validation = ruleIdExecutionContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n success: false,\n ruleId: context.ruleId || 'unknown',\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Validation failed: ${validationErrors.join(', ')}`,\n }\n }\n\n // Fetch rule by rule_id (string identifier) + tenant/org\n const rule = await em.findOne(BusinessRule, {\n ruleId: context.ruleId, // String identifier column\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n deletedAt: null,\n })\n\n if (!rule) {\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule not found',\n }\n }\n\n if (!rule.enabled) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule is disabled',\n }\n }\n\n // Check effective date range\n const now = new Date()\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule is not yet effective (starts ${rule.effectiveFrom.toISOString()})`,\n }\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule has expired (ended ${rule.effectiveTo.toISOString()})`,\n }\n }\n\n // Build RuleEngineContext (use provided entityType or fall back to rule's)\n const engineContext: RuleEngineContext = {\n entityType: context.entityType || rule.entityType,\n entityId: context.entityId,\n eventType: context.eventType || rule.eventType || undefined,\n data: context.data,\n user: context.user,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy,\n dryRun: context.dryRun,\n }\n\n // Execute via existing executeSingleRule\n const result = await executeSingleRule(em, rule, engineContext)\n\n return {\n success: !result.error,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId,\n }\n}\n\n/**\n * Sensitive field patterns to exclude from logs\n */\nconst SENSITIVE_FIELD_PATTERNS = [\n /password/i,\n /passwd/i,\n /pwd/i,\n /secret/i,\n /token/i,\n /api[_-]?key/i,\n /auth/i,\n /credit[_-]?card/i,\n /card[_-]?number/i,\n /cvv/i,\n /ssn/i,\n /social[_-]?security/i,\n /tax[_-]?id/i,\n /driver[_-]?license/i,\n /passport/i,\n]\n\n/**\n * Maximum depth for nested object sanitization\n */\nconst MAX_SANITIZATION_DEPTH = 5\n\n/**\n * Sanitize data for logging by removing sensitive fields\n */\nfunction sanitizeForLogging(data: any, depth: number = 0): any {\n // Prevent infinite recursion\n if (depth > MAX_SANITIZATION_DEPTH) {\n return '[Max depth exceeded]'\n }\n\n // Handle null/undefined\n if (data === null || data === undefined) {\n return data\n }\n\n // Handle primitives\n if (typeof data !== 'object') {\n return data\n }\n\n // Handle arrays\n if (Array.isArray(data)) {\n return data.map(item => sanitizeForLogging(item, depth + 1))\n }\n\n // Handle objects\n const sanitized: Record<string, any> = {}\n\n for (const [key, value] of Object.entries(data)) {\n // Check if field name matches sensitive patterns\n const isSensitive = SENSITIVE_FIELD_PATTERNS.some(pattern => pattern.test(key))\n\n if (isSensitive) {\n sanitized[key] = '[REDACTED]'\n } else if (typeof value === 'object' && value !== null) {\n sanitized[key] = sanitizeForLogging(value, depth + 1)\n } else {\n sanitized[key] = value\n }\n }\n\n return sanitized\n}\n\n/**\n * Sanitize user object for logging (keep only safe fields)\n */\nfunction sanitizeUser(user: any): any {\n if (!user) {\n return undefined\n }\n\n // Only log safe user fields\n return {\n id: user.id,\n role: user.role,\n // Don't log: email, name, phone, address, etc.\n }\n}\n\n/**\n * Log rule execution to database\n */\ninterface LogExecutionOptions {\n rule: BusinessRule\n context: RuleEngineContext\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n}\n\nexport async function logRuleExecution(\n em: EntityManager,\n options: LogExecutionOptions\n): Promise<string> {\n const { rule, context, conditionResult, actionsExecuted, executionTime, error } = options\n\n const executionResult: 'SUCCESS' | 'FAILURE' | 'ERROR' = error\n ? EXECUTION_RESULT_ERROR\n : conditionResult\n ? EXECUTION_RESULT_SUCCESS\n : EXECUTION_RESULT_FAILURE\n\n const log = em.create(RuleExecutionLog, {\n rule,\n entityId: context.entityId || DEFAULT_ENTITY_ID,\n entityType: context.entityType,\n executionResult,\n inputContext: {\n data: sanitizeForLogging(context.data),\n eventType: context.eventType,\n user: sanitizeUser(context.user),\n },\n outputContext: actionsExecuted\n ? {\n conditionResult,\n actionsExecuted: actionsExecuted.results.map((r) => ({\n type: r.action.type,\n success: r.success,\n error: r.error,\n })),\n }\n : null,\n errorMessage: error || null,\n executionTimeMs: executionTime,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy || null,\n })\n\n await em.persistAndFlush(log)\n\n return log.id\n}\n\nasync function emitRuleExecutionFailed(\n eventBus: Pick<EventBus, 'emitEvent'> | null | undefined,\n payload: RuleExecutionFailedPayload\n): Promise<void> {\n if (!eventBus?.emitEvent) return\n\n await eventBus.emitEvent('business_rules.rule.execution_failed', payload).catch(() => undefined)\n}\n"],
5
- "mappings": "AAEA,SAAS,cAAc,wBAAuC;AAC9D,YAAY,mBAAmB;AAC/B,YAAY,oBAAoB;AAGhC,SAAS,yBAAyB,4BAA4B,kCAAkC,oCAAoC;AAKpI,MAAM,oBAAoB;AAC1B,MAAM,kBAAkB;AACxB,MAAM,yBAAyB;AAC/B,MAAM,2BAA2B;AACjC,MAAM,2BAA2B;AAKjC,MAAM,0BAA0B;AAChC,MAAM,6BAA6B;AACnC,MAAM,iCAAiC;AA4IvC,eAAe,YACb,SACA,WACA,cACY;AACZ,MAAI;AAEJ,QAAM,iBAAiB,IAAI,QAAe,CAAC,GAAG,WAAW;AACvD,gBAAY,WAAW,MAAM;AAC3B,aAAO,IAAI,MAAM,GAAG,YAAY,cAAc,SAAS,KAAK,CAAC;AAAA,IAC/D,GAAG,SAAS;AAAA,EACd,CAAC;AAED,MAAI;AACF,WAAO,MAAM,QAAQ,KAAK,CAAC,SAAS,cAAc,CAAC;AAAA,EACrD,UAAE;AACA,iBAAa,SAAU;AAAA,EACzB;AACF;AAKA,eAAsB,aACpB,IACA,SACA,UAAsC,CAAC,GACZ;AAE3B,QAAM,aAAa,wBAAwB,UAAU,OAAO;AAC5D,MAAI,CAAC,WAAW,SAAS;AACvB,UAAM,mBAAmB,WAAW,MAAM,OAAO,IAAI,OAAK,GAAG,EAAE,KAAK,KAAK,GAAG,CAAC,KAAK,EAAE,OAAO,EAAE;AAC7F,WAAO;AAAA,MACL,SAAS;AAAA,MACT,eAAe,CAAC;AAAA,MAChB,oBAAoB;AAAA,MACpB,QAAQ;AAAA,IACV;AAAA,EACF;AAEA,QAAM,YAAY,KAAK,IAAI;AAC3B,QAAM,gBAAuC,CAAC;AAC9C,QAAM,SAAmB,CAAC;AAC1B,QAAM,SAAmB,CAAC;AAE1B,MAAI;AAEF,UAAM,QAAQ,MAAM,oBAAoB,IAAI;AAAA,MAC1C,YAAY,QAAQ;AAAA,MACpB,WAAW,QAAQ;AAAA,MACnB,UAAU,QAAQ;AAAA,MAClB,gBAAgB,QAAQ;AAAA,IAC1B,CAAC;AAGD,QAAI,MAAM,SAAS,yBAAyB;AAC1C,aAAO;AAAA,QACL,8BAA8B,MAAM,MAAM,4BAA4B,uBAAuB;AAAA,MAC/F;AACA,aAAO;AAAA,QACL,SAAS;AAAA,QACT,eAAe,CAAC;AAAA,QAChB,oBAAoB,KAAK,IAAI,IAAI;AAAA,QACjC;AAAA,MACF;AAAA,IACF;AAIA,UAAM,oBAAoB,YAAY;AACpC,iBAAW,QAAQ,OAAO;AAC1B,YAAI;AACF,gBAAM,aAAa,MAAM,kBAAkB,IAAI,MAAM,SAAS,OAAO;AACrE,wBAAc,KAAK,UAAU;AAE7B,cAAI,WAAW,OAAO;AACpB,mBAAO,KAAK,WAAW,KAAK;AAAA,UAC9B;AAEA,cAAI,WAAW,OAAO;AACpB,mBAAO;AAAA,cACL,iCAAiC,KAAK,MAAM,UAAU,KAAK,QAAQ,gBAAgB,QAAQ,UAAU,MAAM,WAAW,KAAK;AAAA,YAC7H;AAAA,UACF;AAAA,QACF,SAAS,OAAO;AACd,gBAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,iBAAO;AAAA,YACL,8CAA8C,KAAK,MAAM,UAAU,KAAK,QAAQ,MAAM,YAAY;AAAA,UACpG;AAEA,cAAI,CAAC,QAAQ,QAAQ;AACnB,kBAAM,wBAAwB,QAAQ,UAAU;AAAA,cAC9C,QAAQ,KAAK;AAAA,cACb,UAAU,KAAK;AAAA,cACf,YAAY,QAAQ,cAAc;AAAA,cAClC;AAAA,cACA,UAAU,QAAQ;AAAA,cAClB,gBAAgB,QAAQ,kBAAkB;AAAA,YAC5C,CAAC;AAAA,UACH;AAEA,wBAAc,KAAK;AAAA,YACjB;AAAA,YACA,iBAAiB;AAAA,YACjB,iBAAiB;AAAA,YACjB,eAAe;AAAA,YACf,OAAO;AAAA,UACT,CAAC;AAAA,QACH;AAAA,MACA;AAAA,IACF,GAAG;AAGH,UAAM;AAAA,MACJ;AAAA,MACA;AAAA,MACA,4CAA4C,QAAQ,UAAU;AAAA,IAChE;AAIA,UAAM,aAAa,cAAc,OAAO,CAAC,MAAM,EAAE,KAAK,aAAa,eAAe;AAClF,UAAM,UAAU,WAAW,WAAW,KAAK,WAAW,MAAM,CAAC,MAAM,EAAE,eAAe;AAEpF,UAAM,qBAAqB,KAAK,IAAI,IAAI;AAExC,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA,QAAQ,OAAO,SAAS,IAAI,SAAS;AAAA,MACrC,QAAQ,OAAO,SAAS,IAAI,SAAS;AAAA,IACvC;AAAA,EACF,SAAS,OAAO;AACd,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,UAAM,QAAQ,iBAAiB,QAAQ,MAAM,QAAQ;AACrD,WAAO;AAAA,MACL,0CAA0C,QAAQ,UAAU,cAAc,QAAQ,YAAY,SAAS,MAAM,YAAY,GAAG,QAAQ;AAAA,SAAY,KAAK,KAAK,EAAE;AAAA,IAC9J;AAEA,UAAM,qBAAqB,KAAK,IAAI,IAAI;AAExC,WAAO;AAAA,MACL,SAAS;AAAA,MACT;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AACF;AAKA,eAAsB,kBACpB,IACA,MACA,SACA,UAAsC,CAAC,GACT;AAC9B,QAAM,YAAY,KAAK,IAAI;AAE3B,MAAI;AAEF,UAAM,qBAAqB,YAAY;AAErC,YAAM,cAAqC;AAAA,QACzC,YAAY,QAAQ;AAAA,QACpB,UAAU,QAAQ;AAAA,QAClB,WAAW,QAAQ;AAAA,QACnB,MAAM,QAAQ;AAAA,QACd,QAAQ,QAAQ;AAAA,QAChB,cAAc,QAAQ;AAAA,MACxB;AAGA,YAAM,SAAS,MAAM,cAAc,mBAAmB,MAAM,QAAQ,MAAM,WAAW;AAGrF,UAAI,CAAC,OAAO,qBAAqB;AAC/B,cAAMA,iBAAgB,KAAK,IAAI,IAAI;AAEnC,YAAIC;AAGJ,YAAI,CAAC,QAAQ,QAAQ;AACnB,UAAAA,SAAQ,MAAM,iBAAiB,IAAI;AAAA,YACjC;AAAA,YACA;AAAA,YACA,iBAAiB;AAAA,YACjB,iBAAiB;AAAA,YACjB,eAAAD;AAAA,YACA,OAAO,OAAO;AAAA,UAChB,CAAC;AAED,gBAAM,wBAAwB,QAAQ,UAAU;AAAA,YAC9C,QAAQ,KAAK;AAAA,YACb,UAAU,KAAK;AAAA,YACf,YAAY,QAAQ,cAAc;AAAA,YAClC,cAAc,OAAO,SAAS;AAAA,YAC9B,UAAU,QAAQ;AAAA,YAClB,gBAAgB,QAAQ,kBAAkB;AAAA,UAC5C,CAAC;AAAA,QACH;AAEA,eAAO;AAAA,UACL;AAAA,UACA,iBAAiB;AAAA,UACjB,iBAAiB;AAAA,UACjB,eAAAA;AAAA,UACA,OAAO,OAAO;AAAA,UACd,OAAAC;AAAA,QACF;AAAA,MACF;AAGA,YAAM,UAAU,OAAO,mBAAmB,KAAK,iBAAiB,KAAK;AAErE,UAAI,kBAAiD;AAErD,UAAI,WAAW,MAAM,QAAQ,OAAO,KAAK,QAAQ,SAAS,GAAG;AAE3D,cAAM,gBAA+B;AAAA,UACnC,GAAG;AAAA,UACH,MAAM,QAAQ;AAAA,UACd,QAAQ,KAAK;AAAA,UACb,UAAU,KAAK;AAAA,QACjB;AAGA,0BAAkB,MAAM,eAAe,eAAe,SAAS,aAAa;AAAA,MAC9E;AAEA,YAAM,gBAAgB,KAAK,IAAI,IAAI;AAEnC,UAAI;AAGJ,UAAI,CAAC,QAAQ,QAAQ;AACnB,gBAAQ,MAAM,iBAAiB,IAAI;AAAA,UACjC;AAAA,UACA;AAAA,UACA,iBAAiB,OAAO;AAAA,UACxB;AAAA,UACA;AAAA,QACF,CAAC;AAAA,MACH;AAEA,aAAO;AAAA,QACL;AAAA,QACA,iBAAiB,OAAO;AAAA,QACxB;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAGA,WAAO,MAAM;AAAA,MACX,mBAAmB;AAAA,MACnB;AAAA,MACA,yCAAyC,KAAK,MAAM;AAAA,IACtD;AAAA,EACF,SAAS,OAAO;AACd,UAAM,gBAAgB,KAAK,IAAI,IAAI;AACnC,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,UAAM,gBAAgB,kCAAkC,KAAK,MAAM,gBAAgB,QAAQ,UAAU,MAAM,YAAY;AAEvH,QAAI;AAGJ,QAAI,CAAC,QAAQ,QAAQ;AACnB,cAAQ,MAAM,iBAAiB,IAAI;AAAA,QACjC;AAAA,QACA;AAAA,QACA,iBAAiB;AAAA,QACjB,iBAAiB;AAAA,QACjB;AAAA,QACA,OAAO;AAAA,MACT,CAAC;AAED,YAAM,wBAAwB,QAAQ,UAAU;AAAA,QAC9C,QAAQ,KAAK;AAAA,QACb,UAAU,KAAK;AAAA,QACf,YAAY,QAAQ,cAAc;AAAA,QAClC,cAAc;AAAA,QACd,UAAU,QAAQ;AAAA,QAClB,gBAAgB,QAAQ,kBAAkB;AAAA,MAC5C,CAAC;AAAA,IACH;AAEA,WAAO;AAAA,MACL;AAAA,MACA,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB;AAAA,MACA,OAAO;AAAA,MACP;AAAA,IACF;AAAA,EACF;AACF;AAKA,eAAsB,oBACpB,IACA,SACyB;AAEzB,6BAA2B,MAAM,OAAO;AAExC,QAAM,EAAE,YAAY,WAAW,UAAU,gBAAgB,SAAS,IAAI;AAEtE,QAAM,QAA+B;AAAA,IACnC;AAAA,IACA;AAAA,IACA;AAAA,IACA,SAAS;AAAA,IACT,WAAW;AAAA,EACb;AAEA,MAAI,WAAW;AACb,UAAM,YAAY;AAAA,EACpB;AAEA,MAAI,UAAU;AACZ,UAAM,WAAW;AAAA,EACnB;AAEA,QAAM,QAAQ,MAAM,GAAG,KAAK,cAAc,OAAO;AAAA,IAC/C,SAAS,EAAE,UAAU,QAAQ,QAAQ,MAAM;AAAA,EAC7C,CAAC;AAGD,QAAM,MAAM,oBAAI,KAAK;AACrB,SAAO,MAAM,OAAO,CAAC,SAAS;AAC5B,QAAI,KAAK,iBAAiB,KAAK,gBAAgB,KAAK;AAClD,aAAO;AAAA,IACT;AACA,QAAI,KAAK,eAAe,KAAK,cAAc,KAAK;AAC9C,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT,CAAC;AACH;AAMA,eAAsB,gBACpB,IACA,SACoC;AACpC,QAAM,YAAY,KAAK,IAAI;AAG3B,QAAM,aAAa,iCAAiC,UAAU,OAAO;AACrE,MAAI,CAAC,WAAW,SAAS;AACvB,UAAM,mBAAmB,WAAW,MAAM,OAAO,IAAI,OAAK,GAAG,EAAE,KAAK,KAAK,GAAG,CAAC,KAAK,EAAE,OAAO,EAAE;AAC7F,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,QAAQ;AAAA,MAChB,UAAU;AAAA,MACV,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,sBAAsB,iBAAiB,KAAK,IAAI,CAAC;AAAA,IAC1D;AAAA,EACF;AAGA,QAAM,OAAO,MAAM,GAAG,QAAQ,cAAc;AAAA,IAC1C,IAAI,QAAQ;AAAA,IACZ,UAAU,QAAQ;AAAA,IAClB,gBAAgB,QAAQ;AAAA,IACxB,WAAW;AAAA,EACb,CAAC;AAED,MAAI,CAAC,MAAM;AACT,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,QAAQ;AAAA,MAChB,UAAU;AAAA,MACV,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO;AAAA,IACT;AAAA,EACF;AAEA,MAAI,CAAC,KAAK,SAAS;AACjB,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO;AAAA,IACT;AAAA,EACF;AAGA,QAAM,MAAM,oBAAI,KAAK;AACrB,MAAI,KAAK,iBAAiB,KAAK,gBAAgB,KAAK;AAClD,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,qCAAqC,KAAK,cAAc,YAAY,CAAC;AAAA,IAC9E;AAAA,EACF;AACA,MAAI,KAAK,eAAe,KAAK,cAAc,KAAK;AAC9C,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,2BAA2B,KAAK,YAAY,YAAY,CAAC;AAAA,IAClE;AAAA,EACF;AAGA,QAAM,gBAAmC;AAAA,IACvC,YAAY,QAAQ,cAAc,KAAK;AAAA,IACvC,UAAU,QAAQ;AAAA,IAClB,WAAW,QAAQ,aAAa,KAAK,aAAa;AAAA,IAClD,MAAM,QAAQ;AAAA,IACd,MAAM,QAAQ;AAAA,IACd,UAAU,QAAQ;AAAA,IAClB,gBAAgB,QAAQ;AAAA,IACxB,YAAY,QAAQ;AAAA,IACpB,QAAQ,QAAQ;AAAA,EAClB;AAGA,QAAM,SAAS,MAAM,kBAAkB,IAAI,MAAM,aAAa;AAE9D,SAAO;AAAA,IACL,SAAS,CAAC,OAAO;AAAA,IACjB,QAAQ,KAAK;AAAA,IACb,UAAU,KAAK;AAAA,IACf,iBAAiB,OAAO;AAAA,IACxB,iBAAiB,OAAO;AAAA,IACxB,eAAe,OAAO;AAAA,IACtB,OAAO,OAAO;AAAA,IACd,OAAO,OAAO;AAAA,EAChB;AACF;AAOA,eAAsB,oBACpB,IACA,SACoC;AACpC,QAAM,YAAY,KAAK,IAAI;AAG3B,QAAM,aAAa,6BAA6B,UAAU,OAAO;AACjE,MAAI,CAAC,WAAW,SAAS;AACvB,UAAM,mBAAmB,WAAW,MAAM,OAAO,IAAI,OAAK,GAAG,EAAE,KAAK,KAAK,GAAG,CAAC,KAAK,EAAE,OAAO,EAAE;AAC7F,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,QAAQ,UAAU;AAAA,MAC1B,UAAU;AAAA,MACV,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,sBAAsB,iBAAiB,KAAK,IAAI,CAAC;AAAA,IAC1D;AAAA,EACF;AAGA,QAAM,OAAO,MAAM,GAAG,QAAQ,cAAc;AAAA,IAC1C,QAAQ,QAAQ;AAAA;AAAA,IAChB,UAAU,QAAQ;AAAA,IAClB,gBAAgB,QAAQ;AAAA,IACxB,WAAW;AAAA,EACb,CAAC;AAED,MAAI,CAAC,MAAM;AACT,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,QAAQ;AAAA,MAChB,UAAU;AAAA,MACV,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO;AAAA,IACT;AAAA,EACF;AAEA,MAAI,CAAC,KAAK,SAAS;AACjB,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO;AAAA,IACT;AAAA,EACF;AAGA,QAAM,MAAM,oBAAI,KAAK;AACrB,MAAI,KAAK,iBAAiB,KAAK,gBAAgB,KAAK;AAClD,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,qCAAqC,KAAK,cAAc,YAAY,CAAC;AAAA,IAC9E;AAAA,EACF;AACA,MAAI,KAAK,eAAe,KAAK,cAAc,KAAK;AAC9C,WAAO;AAAA,MACL,SAAS;AAAA,MACT,QAAQ,KAAK;AAAA,MACb,UAAU,KAAK;AAAA,MACf,iBAAiB;AAAA,MACjB,iBAAiB;AAAA,MACjB,eAAe,KAAK,IAAI,IAAI;AAAA,MAC5B,OAAO,2BAA2B,KAAK,YAAY,YAAY,CAAC;AAAA,IAClE;AAAA,EACF;AAGA,QAAM,gBAAmC;AAAA,IACvC,YAAY,QAAQ,cAAc,KAAK;AAAA,IACvC,UAAU,QAAQ;AAAA,IAClB,WAAW,QAAQ,aAAa,KAAK,aAAa;AAAA,IAClD,MAAM,QAAQ;AAAA,IACd,MAAM,QAAQ;AAAA,IACd,UAAU,QAAQ;AAAA,IAClB,gBAAgB,QAAQ;AAAA,IACxB,YAAY,QAAQ;AAAA,IACpB,QAAQ,QAAQ;AAAA,EAClB;AAGA,QAAM,SAAS,MAAM,kBAAkB,IAAI,MAAM,aAAa;AAE9D,SAAO;AAAA,IACL,SAAS,CAAC,OAAO;AAAA,IACjB,QAAQ,KAAK;AAAA,IACb,UAAU,KAAK;AAAA,IACf,iBAAiB,OAAO;AAAA,IACxB,iBAAiB,OAAO;AAAA,IACxB,eAAe,OAAO;AAAA,IACtB,OAAO,OAAO;AAAA,IACd,OAAO,OAAO;AAAA,EAChB;AACF;AAKA,MAAM,2BAA2B;AAAA,EAC/B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;AAKA,MAAM,yBAAyB;AAK/B,SAAS,mBAAmB,MAAW,QAAgB,GAAQ;AAE7D,MAAI,QAAQ,wBAAwB;AAClC,WAAO;AAAA,EACT;AAGA,MAAI,SAAS,QAAQ,SAAS,QAAW;AACvC,WAAO;AAAA,EACT;AAGA,MAAI,OAAO,SAAS,UAAU;AAC5B,WAAO;AAAA,EACT;AAGA,MAAI,MAAM,QAAQ,IAAI,GAAG;AACvB,WAAO,KAAK,IAAI,UAAQ,mBAAmB,MAAM,QAAQ,CAAC,CAAC;AAAA,EAC7D;AAGA,QAAM,YAAiC,CAAC;AAExC,aAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,IAAI,GAAG;AAE/C,UAAM,cAAc,yBAAyB,KAAK,aAAW,QAAQ,KAAK,GAAG,CAAC;AAE9E,QAAI,aAAa;AACf,gBAAU,GAAG,IAAI;AAAA,IACnB,WAAW,OAAO,UAAU,YAAY,UAAU,MAAM;AACtD,gBAAU,GAAG,IAAI,mBAAmB,OAAO,QAAQ,CAAC;AAAA,IACtD,OAAO;AACL,gBAAU,GAAG,IAAI;AAAA,IACnB;AAAA,EACF;AAEA,SAAO;AACT;AAKA,SAAS,aAAa,MAAgB;AACpC,MAAI,CAAC,MAAM;AACT,WAAO;AAAA,EACT;AAGA,SAAO;AAAA,IACL,IAAI,KAAK;AAAA,IACT,MAAM,KAAK;AAAA;AAAA,EAEb;AACF;AAcA,eAAsB,iBACpB,IACA,SACiB;AACjB,QAAM,EAAE,MAAM,SAAS,iBAAiB,iBAAiB,eAAe,MAAM,IAAI;AAElF,QAAM,kBAAmD,QACrD,yBACA,kBACE,2BACA;AAEN,QAAM,MAAM,GAAG,OAAO,kBAAkB;AAAA,IACtC;AAAA,IACA,UAAU,QAAQ,YAAY;AAAA,IAC9B,YAAY,QAAQ;AAAA,IACpB;AAAA,IACA,cAAc;AAAA,MACZ,MAAM,mBAAmB,QAAQ,IAAI;AAAA,MACrC,WAAW,QAAQ;AAAA,MACnB,MAAM,aAAa,QAAQ,IAAI;AAAA,IACjC;AAAA,IACA,eAAe,kBACX;AAAA,MACE;AAAA,MACA,iBAAiB,gBAAgB,QAAQ,IAAI,CAAC,OAAO;AAAA,QACnD,MAAM,EAAE,OAAO;AAAA,QACf,SAAS,EAAE;AAAA,QACX,OAAO,EAAE;AAAA,MACX,EAAE;AAAA,IACJ,IACA;AAAA,IACJ,cAAc,SAAS;AAAA,IACvB,iBAAiB;AAAA,IACjB,UAAU,QAAQ;AAAA,IAClB,gBAAgB,QAAQ;AAAA,IACxB,YAAY,QAAQ,cAAc;AAAA,EACpC,CAAC;AAED,QAAM,GAAG,gBAAgB,GAAG;AAE5B,SAAO,IAAI;AACb;AAEA,eAAe,wBACb,UACA,SACe;AACf,MAAI,CAAC,UAAU,UAAW;AAE1B,QAAM,SAAS,UAAU,wCAAwC,OAAO,EAAE,MAAM,MAAM,MAAS;AACjG;",
4
+ "sourcesContent": ["import type { EntityManager } from '@mikro-orm/core'\nimport type { EventBus } from '@open-mercato/events'\nimport { BusinessRule, RuleExecutionLog, type RuleType } from '../data/entities'\nimport * as ruleEvaluator from './rule-evaluator'\nimport * as actionExecutor from './action-executor'\nimport type { RuleEvaluationContext } from './rule-evaluator'\nimport type { ActionContext, ActionExecutionOutcome } from './action-executor'\nimport { ruleEngineContextSchema, ruleDiscoveryOptionsSchema, directRuleExecutionContextSchema, ruleIdExecutionContextSchema } from '../data/validators'\n\n/**\n * Constants\n */\nconst DEFAULT_ENTITY_ID = 'unknown'\nconst RULE_TYPE_GUARD = 'GUARD'\nconst EXECUTION_RESULT_ERROR = 'ERROR'\nconst EXECUTION_RESULT_SUCCESS = 'SUCCESS'\nconst EXECUTION_RESULT_FAILURE = 'FAILURE'\n\n/**\n * Execution limits\n */\nconst MAX_RULES_PER_EXECUTION = 100\nconst MAX_SINGLE_RULE_TIMEOUT_MS = 30000 // 30 seconds\nconst MAX_TOTAL_EXECUTION_TIMEOUT_MS = 60000 // 60 seconds\n\n/**\n * Rule execution context\n */\nexport interface RuleEngineContext {\n entityType: string\n entityId?: string\n eventType?: string\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenant?: {\n id?: string\n [key: string]: any\n }\n organization?: {\n id?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n [key: string]: any\n}\n\n/**\n * Single rule execution result\n */\nexport interface RuleExecutionResult {\n rule: BusinessRule\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n logId?: string // Database log ID (if logged)\n}\n\n/**\n * Overall rule engine result\n */\nexport interface RuleEngineResult {\n allowed: boolean\n executedRules: RuleExecutionResult[]\n totalExecutionTime: number\n errors?: string[]\n logIds?: string[]\n}\n\n/**\n * Rule discovery options\n */\nexport interface RuleDiscoveryOptions {\n entityType: string\n eventType?: string\n tenantId: string\n organizationId: string\n ruleType?: RuleType\n}\n\nexport type RuleEngineExecutionOptions = {\n eventBus?: Pick<EventBus, 'emitEvent'> | null\n}\n\ntype RuleExecutionFailedPayload = {\n ruleId: string\n ruleName: string\n entityType?: string | null\n errorMessage?: string | null\n tenantId: string\n organizationId?: string | null\n}\n\n/**\n * Direct rule execution context (for executing a specific rule by ID)\n */\nexport interface DirectRuleExecutionContext {\n ruleId: string // Database UUID of the rule\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n // Optional for logging (falls back to rule's entityType)\n entityType?: string\n entityId?: string\n eventType?: string\n}\n\n/**\n * Direct rule execution result\n */\nexport interface DirectRuleExecutionResult {\n success: boolean\n ruleId: string\n ruleName: string\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n logId?: string\n}\n\n/**\n * Context for executing a rule by its string rule_id identifier\n * Unlike DirectRuleExecutionContext which uses database UUID,\n * this uses the string identifier (e.g., \"workflow_checkout_inventory_available\")\n */\nexport interface RuleIdExecutionContext {\n ruleId: string // String identifier (e.g., \"workflow_checkout_inventory_available\")\n data: any\n user?: {\n id?: string\n email?: string\n role?: string\n [key: string]: any\n }\n tenantId: string\n organizationId: string\n executedBy?: string\n dryRun?: boolean\n entityType?: string\n entityId?: string\n eventType?: string\n}\n\n/**\n * Execute a function with a timeout\n */\nasync function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n errorMessage: string\n): Promise<T> {\n let timeoutId: NodeJS.Timeout\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n timeoutId = setTimeout(() => {\n reject(new Error(`${errorMessage} (timeout: ${timeoutMs}ms)`))\n }, timeoutMs)\n })\n\n try {\n return await Promise.race([promise, timeoutPromise])\n } finally {\n clearTimeout(timeoutId!)\n }\n}\n\n/**\n * Execute all applicable rules for the given context\n */\nexport async function executeRules(\n em: EntityManager,\n context: RuleEngineContext,\n options: RuleEngineExecutionOptions = {}\n): Promise<RuleEngineResult> {\n // Validate input\n const validation = ruleEngineContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n allowed: false,\n executedRules: [],\n totalExecutionTime: 0,\n errors: validationErrors,\n }\n }\n\n const startTime = Date.now()\n const executedRules: RuleExecutionResult[] = []\n const errors: string[] = []\n const logIds: string[] = []\n\n try {\n // Discover applicable rules\n const rules = await findApplicableRules(em, {\n entityType: context.entityType,\n eventType: context.eventType,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n })\n\n // Check rule count limit\n if (rules.length > MAX_RULES_PER_EXECUTION) {\n errors.push(\n `Rule count limit exceeded: ${rules.length} rules found, maximum is ${MAX_RULES_PER_EXECUTION}`\n )\n return {\n allowed: false,\n executedRules: [],\n totalExecutionTime: Date.now() - startTime,\n errors,\n }\n }\n\n // Rules already sorted by database query (priority DESC, ruleId ASC)\n // Execute each rule with total timeout\n const executionPromise = (async () => {\n for (const rule of rules) {\n try {\n const ruleResult = await executeSingleRule(em, rule, context, options)\n executedRules.push(ruleResult)\n\n if (ruleResult.logId) {\n logIds.push(ruleResult.logId)\n }\n\n if (ruleResult.error) {\n errors.push(\n `Rule execution failed [ruleId=${rule.ruleId}, type=${rule.ruleType}, entityType=${context.entityType}]: ${ruleResult.error}`\n )\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n errors.push(\n `Unexpected error in rule execution [ruleId=${rule.ruleId}, type=${rule.ruleType}]: ${errorMessage}`\n )\n\n if (!context.dryRun) {\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n executedRules.push({\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: 0,\n error: errorMessage,\n })\n }\n }\n })()\n\n // Execute with timeout\n await withTimeout(\n executionPromise,\n MAX_TOTAL_EXECUTION_TIMEOUT_MS,\n `Total rule execution timeout [entityType=${context.entityType}]`\n )\n\n // Determine overall allowed status\n // For GUARD rules: all must pass for operation to be allowed\n const guardRules = executedRules.filter((r) => r.rule.ruleType === RULE_TYPE_GUARD)\n const allowed = guardRules.length === 0 || guardRules.every((r) => r.conditionResult)\n\n const totalExecutionTime = Date.now() - startTime\n\n return {\n allowed,\n executedRules,\n totalExecutionTime,\n errors: errors.length > 0 ? errors : undefined,\n logIds: logIds.length > 0 ? logIds : undefined,\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error)\n const stack = error instanceof Error ? error.stack : undefined\n errors.push(\n `Critical rule engine error [entityType=${context.entityType}, entityId=${context.entityId || 'unknown'}]: ${errorMessage}${stack ? `\\nStack: ${stack}` : ''}`\n )\n\n const totalExecutionTime = Date.now() - startTime\n\n return {\n allowed: false,\n executedRules,\n totalExecutionTime,\n errors,\n }\n }\n}\n\n/**\n * Execute a single rule\n */\nexport async function executeSingleRule(\n em: EntityManager,\n rule: BusinessRule,\n context: RuleEngineContext,\n options: RuleEngineExecutionOptions = {}\n): Promise<RuleExecutionResult> {\n const startTime = Date.now()\n\n try {\n // Wrap execution in timeout\n const executeWithTimeout = async () => {\n // Build evaluation context\n const evalContext: RuleEvaluationContext = {\n entityType: context.entityType,\n entityId: context.entityId,\n eventType: context.eventType,\n user: context.user,\n tenant: context.tenant,\n organization: context.organization,\n }\n\n // Evaluate rule conditions\n const result = await ruleEvaluator.evaluateSingleRule(rule, context.data, evalContext)\n\n // Check if evaluation completed (not just if conditions passed)\n if (!result.evaluationCompleted) {\n const executionTime = Date.now() - startTime\n\n let logId: string | undefined\n\n // Log failure if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: result.error,\n })\n\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage: result.error ?? null,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n return {\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: result.error,\n logId,\n }\n }\n\n // Evaluation completed successfully - determine which actions to execute\n const actions = result.conditionsPassed ? rule.successActions : rule.failureActions\n\n let actionsExecuted: ActionExecutionOutcome | null = null\n\n if (actions && Array.isArray(actions) && actions.length > 0) {\n // Build action context\n const actionContext: ActionContext = {\n ...evalContext,\n data: context.data,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n }\n\n // Execute actions\n actionsExecuted = await actionExecutor.executeActions(actions, actionContext)\n }\n\n const executionTime = Date.now() - startTime\n\n let logId: string | undefined\n\n // Log execution if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: result.conditionsPassed,\n actionsExecuted,\n executionTime,\n })\n }\n\n return {\n rule,\n conditionResult: result.conditionsPassed,\n actionsExecuted,\n executionTime,\n logId,\n }\n }\n\n // Execute with single rule timeout\n return await withTimeout(\n executeWithTimeout(),\n MAX_SINGLE_RULE_TIMEOUT_MS,\n `Single rule execution timeout [ruleId=${rule.ruleId}]`\n )\n } catch (error) {\n const executionTime = Date.now() - startTime\n const errorMessage = error instanceof Error ? error.message : String(error)\n const enhancedError = `Failed to execute rule [ruleId=${rule.ruleId}, entityType=${context.entityType}]: ${errorMessage}`\n\n let logId: string | undefined\n\n // Log error if not dry run\n if (!context.dryRun) {\n logId = await logRuleExecution(em, {\n rule,\n context,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: enhancedError,\n })\n\n await emitRuleExecutionFailed(options.eventBus, {\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n entityType: context.entityType ?? null,\n errorMessage: enhancedError,\n tenantId: context.tenantId,\n organizationId: context.organizationId ?? null,\n })\n }\n\n return {\n rule,\n conditionResult: false,\n actionsExecuted: null,\n executionTime,\n error: enhancedError,\n logId,\n }\n }\n}\n\n/**\n * Find all applicable rules for the given criteria\n */\nexport async function findApplicableRules(\n em: EntityManager,\n options: RuleDiscoveryOptions\n): Promise<BusinessRule[]> {\n // Validate input\n ruleDiscoveryOptionsSchema.parse(options)\n\n const { entityType, eventType, tenantId, organizationId, ruleType } = options\n\n const where: Partial<BusinessRule> = {\n entityType,\n tenantId,\n organizationId,\n enabled: true,\n deletedAt: null,\n }\n\n if (eventType) {\n where.eventType = eventType\n }\n\n if (ruleType) {\n where.ruleType = ruleType\n }\n\n const rules = await em.find(BusinessRule, where, {\n orderBy: { priority: 'DESC', ruleId: 'ASC' },\n })\n\n // Filter by effective date range\n const now = new Date()\n return rules.filter((rule) => {\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return false\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return false\n }\n return true\n })\n}\n\n/**\n * Execute a specific rule by its database UUID\n * This bypasses the entityType/eventType discovery mechanism and directly executes the rule\n */\nexport async function executeRuleById(\n em: EntityManager,\n context: DirectRuleExecutionContext\n): Promise<DirectRuleExecutionResult> {\n const startTime = Date.now()\n\n // Validate input\n const validation = directRuleExecutionContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Validation failed: ${validationErrors.join(', ')}`,\n }\n }\n\n // Fetch rule by ID with tenant/org validation\n const rule = await em.findOne(BusinessRule, {\n id: context.ruleId,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n deletedAt: null,\n })\n\n if (!rule) {\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule not found',\n }\n }\n\n if (!rule.enabled) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule is disabled',\n }\n }\n\n // Check effective date range\n const now = new Date()\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule is not yet effective (starts ${rule.effectiveFrom.toISOString()})`,\n }\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule has expired (ended ${rule.effectiveTo.toISOString()})`,\n }\n }\n\n // Build RuleEngineContext (use provided entityType or fall back to rule's)\n const engineContext: RuleEngineContext = {\n entityType: context.entityType || rule.entityType,\n entityId: context.entityId,\n eventType: context.eventType || rule.eventType || undefined,\n data: context.data,\n user: context.user,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy,\n dryRun: context.dryRun,\n }\n\n // Execute via existing executeSingleRule\n const result = await executeSingleRule(em, rule, engineContext)\n\n return {\n success: !result.error,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId,\n }\n}\n\n/**\n * Execute a rule by its string rule_id identifier\n * Looks up rule by rule_id (string column) + tenant_id (unique constraint)\n * This is useful for workflow conditions that reference rules by their string identifiers\n */\nexport async function executeRuleByRuleId(\n em: EntityManager,\n context: RuleIdExecutionContext\n): Promise<DirectRuleExecutionResult> {\n const startTime = Date.now()\n\n // Validate input\n const validation = ruleIdExecutionContextSchema.safeParse(context)\n if (!validation.success) {\n const validationErrors = validation.error.issues.map(e => `${e.path.join('.')}: ${e.message}`)\n return {\n success: false,\n ruleId: context.ruleId || 'unknown',\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Validation failed: ${validationErrors.join(', ')}`,\n }\n }\n\n // Fetch rule by rule_id (string identifier) + tenant/org\n const rule = await em.findOne(BusinessRule, {\n ruleId: context.ruleId, // String identifier column\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n deletedAt: null,\n })\n\n if (!rule) {\n return {\n success: false,\n ruleId: context.ruleId,\n ruleName: 'Unknown',\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule not found',\n }\n }\n\n if (!rule.enabled) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: 'Rule is disabled',\n }\n }\n\n // Check effective date range\n const now = new Date()\n if (rule.effectiveFrom && rule.effectiveFrom > now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule is not yet effective (starts ${rule.effectiveFrom.toISOString()})`,\n }\n }\n if (rule.effectiveTo && rule.effectiveTo < now) {\n return {\n success: false,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: false,\n actionsExecuted: null,\n executionTime: Date.now() - startTime,\n error: `Rule has expired (ended ${rule.effectiveTo.toISOString()})`,\n }\n }\n\n // Build RuleEngineContext (use provided entityType or fall back to rule's)\n const engineContext: RuleEngineContext = {\n entityType: context.entityType || rule.entityType,\n entityId: context.entityId,\n eventType: context.eventType || rule.eventType || undefined,\n data: context.data,\n user: context.user,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy,\n dryRun: context.dryRun,\n }\n\n // Execute via existing executeSingleRule\n const result = await executeSingleRule(em, rule, engineContext)\n\n return {\n success: !result.error,\n ruleId: rule.ruleId,\n ruleName: rule.ruleName,\n conditionResult: result.conditionResult,\n actionsExecuted: result.actionsExecuted,\n executionTime: result.executionTime,\n error: result.error,\n logId: result.logId,\n }\n}\n\n/**\n * Sensitive field patterns to exclude from logs\n */\nconst SENSITIVE_FIELD_PATTERNS = [\n /password/i,\n /passwd/i,\n /pwd/i,\n /secret/i,\n /token/i,\n /api[_-]?key/i,\n /auth/i,\n /credit[_-]?card/i,\n /card[_-]?number/i,\n /cvv/i,\n /ssn/i,\n /social[_-]?security/i,\n /tax[_-]?id/i,\n /driver[_-]?license/i,\n /passport/i,\n]\n\n/**\n * Maximum depth for nested object sanitization\n */\nconst MAX_SANITIZATION_DEPTH = 5\n\n/**\n * Sanitize data for logging by removing sensitive fields\n */\nfunction sanitizeForLogging(data: any, depth: number = 0): any {\n // Prevent infinite recursion\n if (depth > MAX_SANITIZATION_DEPTH) {\n return '[Max depth exceeded]'\n }\n\n // Handle null/undefined\n if (data === null || data === undefined) {\n return data\n }\n\n // Handle primitives\n if (typeof data !== 'object') {\n return data\n }\n\n // Handle arrays\n if (Array.isArray(data)) {\n return data.map(item => sanitizeForLogging(item, depth + 1))\n }\n\n // Handle objects\n const sanitized: Record<string, any> = {}\n\n for (const [key, value] of Object.entries(data)) {\n // Check if field name matches sensitive patterns\n const isSensitive = SENSITIVE_FIELD_PATTERNS.some(pattern => pattern.test(key))\n\n if (isSensitive) {\n sanitized[key] = '[REDACTED]'\n } else if (typeof value === 'object' && value !== null) {\n sanitized[key] = sanitizeForLogging(value, depth + 1)\n } else {\n sanitized[key] = value\n }\n }\n\n return sanitized\n}\n\n/**\n * Sanitize user object for logging (keep only safe fields)\n */\nfunction sanitizeUser(user: any): any {\n if (!user) {\n return undefined\n }\n\n // Only log safe user fields\n return {\n id: user.id,\n role: user.role,\n // Don't log: email, name, phone, address, etc.\n }\n}\n\n/**\n * Log rule execution to database\n */\ninterface LogExecutionOptions {\n rule: BusinessRule\n context: RuleEngineContext\n conditionResult: boolean\n actionsExecuted: ActionExecutionOutcome | null\n executionTime: number\n error?: string\n}\n\nexport async function logRuleExecution(\n em: EntityManager,\n options: LogExecutionOptions\n): Promise<string> {\n const { rule, context, conditionResult, actionsExecuted, executionTime, error } = options\n\n const executionResult: 'SUCCESS' | 'FAILURE' | 'ERROR' = error\n ? EXECUTION_RESULT_ERROR\n : conditionResult\n ? EXECUTION_RESULT_SUCCESS\n : EXECUTION_RESULT_FAILURE\n\n const log = em.create(RuleExecutionLog, {\n rule,\n entityId: context.entityId || DEFAULT_ENTITY_ID,\n entityType: context.entityType,\n executionResult,\n inputContext: {\n data: sanitizeForLogging(context.data),\n eventType: context.eventType,\n user: sanitizeUser(context.user),\n },\n outputContext: actionsExecuted\n ? {\n conditionResult,\n actionsExecuted: actionsExecuted.results.map((r) => ({\n type: r.action.type,\n success: r.success,\n error: r.error,\n })),\n }\n : null,\n errorMessage: error || null,\n executionTimeMs: executionTime,\n tenantId: context.tenantId,\n organizationId: context.organizationId,\n executedBy: context.executedBy || null,\n })\n\n await em.persistAndFlush(log)\n\n return String(log.id)\n}\n\nasync function emitRuleExecutionFailed(\n eventBus: Pick<EventBus, 'emitEvent'> | null | undefined,\n payload: RuleExecutionFailedPayload\n): Promise<void> {\n if (!eventBus?.emitEvent) return\n\n await eventBus.emitEvent('business_rules.rule.execution_failed', payload).catch(() => undefined)\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["executionTime", "logId"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@open-mercato/core",
3
- "version": "0.4.8-develop-bc5be31b5c",
3
+ "version": "0.4.8-develop-409bb4a065",
4
4
  "type": "module",
5
5
  "main": "./dist/index.js",
6
6
  "scripts": {
@@ -217,10 +217,10 @@
217
217
  "semver": "^7.6.3"
218
218
  },
219
219
  "peerDependencies": {
220
- "@open-mercato/shared": "0.4.8-develop-bc5be31b5c"
220
+ "@open-mercato/shared": "0.4.8-develop-409bb4a065"
221
221
  },
222
222
  "devDependencies": {
223
- "@open-mercato/shared": "0.4.8-develop-bc5be31b5c",
223
+ "@open-mercato/shared": "0.4.8-develop-409bb4a065",
224
224
  "@testing-library/dom": "^10.4.1",
225
225
  "@testing-library/jest-dom": "^6.9.1",
226
226
  "@testing-library/react": "^16.3.1",
@@ -310,7 +310,6 @@ function AttachmentFilesField({
310
310
  value,
311
311
  setValue,
312
312
  disabled,
313
- error,
314
313
  labels,
315
314
  uploading,
316
315
  }: AttachmentFilesFieldProps) {
@@ -428,7 +427,6 @@ function AttachmentFilesField({
428
427
  />
429
428
  </div>
430
429
  {renderFileList()}
431
- {error ? <p className="text-xs font-medium text-red-600">{error}</p> : null}
432
430
  </div>
433
431
  )
434
432
  }
@@ -112,7 +112,7 @@ export async function POST(req: Request, context: RouteContext) {
112
112
  } : null,
113
113
  executionTime: result.executionTime,
114
114
  error: result.error,
115
- logId: result.logId,
115
+ logId: result.logId ? String(result.logId) : undefined,
116
116
  }
117
117
 
118
118
  // Return appropriate status based on result
@@ -111,11 +111,11 @@ export async function POST(req: Request) {
111
111
  } : null,
112
112
  executionTime: r.executionTime,
113
113
  error: r.error,
114
- logId: r.logId,
114
+ logId: r.logId ? String(r.logId) : undefined,
115
115
  })),
116
116
  totalExecutionTime: result.totalExecutionTime,
117
117
  errors: result.errors,
118
- logIds: result.logIds,
118
+ logIds: result.logIds?.map((entry) => String(entry)),
119
119
  }
120
120
 
121
121
  return NextResponse.json(response, { status: 200 })
@@ -83,7 +83,7 @@ export async function GET(_req: Request, ctx: { params?: { id?: string } }) {
83
83
  }
84
84
 
85
85
  const response = {
86
- id: log.id,
86
+ id: String(log.id),
87
87
  rule: {
88
88
  id: log.rule.id,
89
89
  ruleId: log.rule.ruleId,
@@ -137,7 +137,7 @@ export async function GET(req: Request) {
137
137
  )
138
138
 
139
139
  const items = rows.map((log) => ({
140
- id: log.id,
140
+ id: String(log.id),
141
141
  ruleId: log.rule.id,
142
142
  ruleName: log.rule.ruleName,
143
143
  ruleType: log.rule.ruleType,
@@ -867,7 +867,7 @@ export async function logRuleExecution(
867
867
 
868
868
  await em.persistAndFlush(log)
869
869
 
870
- return log.id
870
+ return String(log.id)
871
871
  }
872
872
 
873
873
  async function emitRuleExecutionFailed(