spine-framework 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.framework/README.md +129 -0
- package/.framework/cli/bin.cjs +14 -0
- package/.framework/cli/commands/agents.ts +153 -0
- package/.framework/cli/commands/auth.ts +94 -0
- package/.framework/cli/commands/create-app.ts +185 -0
- package/.framework/cli/commands/dev.ts +295 -0
- package/.framework/cli/commands/doctor.ts +442 -0
- package/.framework/cli/commands/generate.ts +332 -0
- package/.framework/cli/commands/init.ts +272 -0
- package/.framework/cli/commands/install-app.ts +391 -0
- package/.framework/cli/commands/items.ts +253 -0
- package/.framework/cli/commands/migrations.ts +141 -0
- package/.framework/cli/commands/pipelines.ts +166 -0
- package/.framework/cli/commands/status.ts +197 -0
- package/.framework/cli/commands/system.ts +184 -0
- package/.framework/cli/commands/test.ts +227 -0
- package/.framework/cli/commands/uninstall-app.ts +166 -0
- package/.framework/cli/context.ts +268 -0
- package/.framework/cli/env-loader.ts +36 -0
- package/.framework/cli/index.ts +106 -0
- package/.framework/cli/welcome.cjs +45 -0
- package/.framework/docs/API.md +384 -0
- package/.framework/docs/STABILITY.md +52 -0
- package/.framework/docs/admin-routes.md +76 -0
- package/.framework/docs/api-docs-progress.md +38 -0
- package/.framework/docs/api-governance.md +146 -0
- package/.framework/docs/api-testing-results.md +212 -0
- package/.framework/docs/apis/admin-configs.md +567 -0
- package/.framework/docs/apis/admin-data.md +272 -0
- package/.framework/docs/apis/index.md +231 -0
- package/.framework/docs/apis/internal.md +295 -0
- package/.framework/docs/apis/runtime.md +537 -0
- package/.framework/docs/assembly-launch-guide.md +138 -0
- package/.framework/docs/audit-results.md +590 -0
- package/.framework/docs/authorization-model.md +170 -0
- package/.framework/docs/db-api-inventory.md +95 -0
- package/.framework/docs/examples/custom-app/README.md +77 -0
- package/.framework/docs/examples/custom-function/README.md +27 -0
- package/.framework/docs/examples/custom-function/handler.ts +48 -0
- package/.framework/docs/examples/custom-webhook/README.md +68 -0
- package/.framework/docs/gap-remediation-backlog.md +103 -0
- package/.framework/docs/guides/cli-guide.md +224 -0
- package/.framework/docs/guides/getting-started.md +103 -0
- package/.framework/docs/guides/import-guide.md +193 -0
- package/.framework/docs/guides/testing-guide.md +229 -0
- package/.framework/docs/permission-examples.md +326 -0
- package/.framework/docs/ui-adoption-verification.md +111 -0
- package/.framework/docs/ui-api-coverage.md +84 -0
- package/.framework/docs/v2-compatibility-audit.md +228 -0
- package/.framework/functions/.gitkeep +1 -0
- package/.framework/functions/_shared/agent-runner.ts +1097 -0
- package/.framework/functions/_shared/app-manifest.ts +184 -0
- package/.framework/functions/_shared/audit.ts +150 -0
- package/.framework/functions/_shared/db.ts +174 -0
- package/.framework/functions/_shared/index.ts +382 -0
- package/.framework/functions/_shared/middleware.ts +490 -0
- package/.framework/functions/_shared/permissions.ts +1325 -0
- package/.framework/functions/_shared/pipeline-runner.ts +731 -0
- package/.framework/functions/_shared/principal.ts +760 -0
- package/.framework/functions/_shared/schema-utils.ts +967 -0
- package/.framework/functions/_shared/testing.ts +258 -0
- package/.framework/functions/_shared/trigger-engine.ts +425 -0
- package/.framework/functions/_shared/webhook-registration.ts +168 -0
- package/.framework/functions/_shared/webhook-registry.ts +129 -0
- package/.framework/functions/account-nodes.ts +111 -0
- package/.framework/functions/admin-data.ts +606 -0
- package/.framework/functions/ai-agents.ts +323 -0
- package/.framework/functions/api-keys.ts +376 -0
- package/.framework/functions/apps.ts +483 -0
- package/.framework/functions/auth.ts +196 -0
- package/.framework/functions/debug-auth.ts +107 -0
- package/.framework/functions/embeddings.ts +556 -0
- package/.framework/functions/integration-routes.ts +523 -0
- package/.framework/functions/integrations.ts +319 -0
- package/.framework/functions/item-progress.ts +272 -0
- package/.framework/functions/logs.ts +438 -0
- package/.framework/functions/observability.ts +275 -0
- package/.framework/functions/pipeline-executions.ts +494 -0
- package/.framework/functions/pipelines.ts +485 -0
- package/.framework/functions/prompt-configs.ts +339 -0
- package/.framework/functions/roles.ts +387 -0
- package/.framework/functions/system-cron.ts +742 -0
- package/.framework/functions/system.ts +323 -0
- package/.framework/functions/tests.ts +119 -0
- package/.framework/functions/timers.ts +357 -0
- package/.framework/functions/triggers.ts +563 -0
- package/.framework/functions/types.ts +604 -0
- package/.framework/migrations/000_foundation.sql +1256 -0
- package/.framework/migrations/001_seed.sql +92 -0
- package/.framework/migrations/002_seed_constraints.sql +13 -0
- package/.framework/migrations/003_auth_user_trigger.sql +59 -0
- package/.framework/src/App.tsx +126 -0
- package/.framework/src/apps/admin/index.tsx +173 -0
- package/.framework/src/components/AppWrapper.tsx +56 -0
- package/.framework/src/components/CustomAppLoader.tsx +116 -0
- package/.framework/src/components/admin/AdminListPage.tsx +151 -0
- package/.framework/src/components/admin/AdminSidebar.tsx +166 -0
- package/.framework/src/components/admin/AdminStatsCard.tsx +62 -0
- package/.framework/src/components/admin/SortableTableHeader.tsx +42 -0
- package/.framework/src/components/app-shell/GenericAppShell.tsx +181 -0
- package/.framework/src/components/app-shell/GenericDetailPage.tsx +200 -0
- package/.framework/src/components/app-shell/GenericListPage.tsx +116 -0
- package/.framework/src/components/app-sidebar.tsx +228 -0
- package/.framework/src/components/auth/ProtectedRoute.tsx +88 -0
- package/.framework/src/components/layout/AppShell.tsx +91 -0
- package/.framework/src/components/layout/Header.tsx +88 -0
- package/.framework/src/components/layout/Layout.tsx +95 -0
- package/.framework/src/components/layout/Sidebar.tsx +329 -0
- package/.framework/src/components/runtime/DataDetailHeader.tsx +77 -0
- package/.framework/src/components/runtime/DataDetailPage.tsx +171 -0
- package/.framework/src/components/runtime/DataFilters.tsx +91 -0
- package/.framework/src/components/runtime/DataHeader.tsx +68 -0
- package/.framework/src/components/runtime/DataListPage.tsx +124 -0
- package/.framework/src/components/runtime/DataStats.tsx +70 -0
- package/.framework/src/components/runtime/DataTable.tsx +174 -0
- package/.framework/src/components/runtime/SchemaDetailForm.tsx +134 -0
- package/.framework/src/components/runtime/index.ts +18 -0
- package/.framework/src/components/search-form.tsx +29 -0
- package/.framework/src/components/shared/AgentView.tsx +213 -0
- package/.framework/src/components/shared/FieldRenderer.tsx +478 -0
- package/.framework/src/components/shared/SchemaFields.tsx +226 -0
- package/.framework/src/components/ui/DataTable.tsx +343 -0
- package/.framework/src/components/ui/Form.tsx +281 -0
- package/.framework/src/components/ui/ItemCard.tsx +296 -0
- package/.framework/src/components/ui/ItemListView.tsx +308 -0
- package/.framework/src/components/ui/LoadingSpinner.tsx +52 -0
- package/.framework/src/components/ui/Modal.tsx +61 -0
- package/.framework/src/components/ui/RichTextEditor.tsx +210 -0
- package/.framework/src/components/ui/accordion.tsx +82 -0
- package/.framework/src/components/ui/alert-dialog.tsx +197 -0
- package/.framework/src/components/ui/alert.tsx +76 -0
- package/.framework/src/components/ui/aspect-ratio.tsx +11 -0
- package/.framework/src/components/ui/avatar.tsx +110 -0
- package/.framework/src/components/ui/badge.tsx +49 -0
- package/.framework/src/components/ui/breadcrumb.tsx +122 -0
- package/.framework/src/components/ui/button-group.tsx +83 -0
- package/.framework/src/components/ui/button.tsx +65 -0
- package/.framework/src/components/ui/calendar.tsx +222 -0
- package/.framework/src/components/ui/card.tsx +100 -0
- package/.framework/src/components/ui/carousel.tsx +240 -0
- package/.framework/src/components/ui/chart.tsx +373 -0
- package/.framework/src/components/ui/checkbox.tsx +31 -0
- package/.framework/src/components/ui/collapsible.tsx +33 -0
- package/.framework/src/components/ui/combobox.tsx +299 -0
- package/.framework/src/components/ui/command.tsx +193 -0
- package/.framework/src/components/ui/context-menu.tsx +261 -0
- package/.framework/src/components/ui/dialog.tsx +165 -0
- package/.framework/src/components/ui/direction.tsx +22 -0
- package/.framework/src/components/ui/drawer.tsx +132 -0
- package/.framework/src/components/ui/dropdown-menu.tsx +269 -0
- package/.framework/src/components/ui/empty.tsx +104 -0
- package/.framework/src/components/ui/field.tsx +238 -0
- package/.framework/src/components/ui/hover-card.tsx +42 -0
- package/.framework/src/components/ui/input-group.tsx +153 -0
- package/.framework/src/components/ui/input-otp.tsx +87 -0
- package/.framework/src/components/ui/input.tsx +19 -0
- package/.framework/src/components/ui/item.tsx +196 -0
- package/.framework/src/components/ui/kbd.tsx +26 -0
- package/.framework/src/components/ui/label.tsx +22 -0
- package/.framework/src/components/ui/menubar.tsx +277 -0
- package/.framework/src/components/ui/native-select.tsx +61 -0
- package/.framework/src/components/ui/navigation-menu.tsx +164 -0
- package/.framework/src/components/ui/pagination.tsx +129 -0
- package/.framework/src/components/ui/popover.tsx +87 -0
- package/.framework/src/components/ui/progress.tsx +31 -0
- package/.framework/src/components/ui/radio-group.tsx +42 -0
- package/.framework/src/components/ui/resizable.tsx +50 -0
- package/.framework/src/components/ui/scroll-area.tsx +53 -0
- package/.framework/src/components/ui/select.tsx +195 -0
- package/.framework/src/components/ui/separator.tsx +26 -0
- package/.framework/src/components/ui/sheet.tsx +145 -0
- package/.framework/src/components/ui/sidebar.tsx +706 -0
- package/.framework/src/components/ui/skeleton.tsx +13 -0
- package/.framework/src/components/ui/slider.tsx +59 -0
- package/.framework/src/components/ui/sonner.tsx +47 -0
- package/.framework/src/components/ui/spinner.tsx +10 -0
- package/.framework/src/components/ui/switch.tsx +33 -0
- package/.framework/src/components/ui/table-primitives.tsx +141 -0
- package/.framework/src/components/ui/table.tsx +114 -0
- package/.framework/src/components/ui/tabs.tsx +90 -0
- package/.framework/src/components/ui/textarea.tsx +18 -0
- package/.framework/src/components/ui/toggle-group.tsx +89 -0
- package/.framework/src/components/ui/toggle.tsx +45 -0
- package/.framework/src/components/ui/tooltip.tsx +57 -0
- package/.framework/src/contexts/AppContext.tsx +133 -0
- package/.framework/src/contexts/AuthContext.tsx +371 -0
- package/.framework/src/hooks/use-mobile.ts +19 -0
- package/.framework/src/hooks/useApi.ts +526 -0
- package/.framework/src/hooks/useApps.ts +114 -0
- package/.framework/src/hooks/useEntityList.ts +190 -0
- package/.framework/src/hooks/useEntityRecord.ts +308 -0
- package/.framework/src/hooks/useForm.ts +307 -0
- package/.framework/src/hooks/useListSchema.ts +264 -0
- package/.framework/src/hooks/useSchemaRecord.ts +223 -0
- package/.framework/src/index.css +128 -0
- package/.framework/src/lib/api.ts +156 -0
- package/.framework/src/lib/supabase.ts +94 -0
- package/.framework/src/lib/utils.ts +317 -0
- package/.framework/src/main.tsx +27 -0
- package/.framework/src/pages/DashboardPage.tsx +181 -0
- package/.framework/src/pages/NotFoundPage.tsx +39 -0
- package/.framework/src/pages/admin/AIAgentDetailPage.tsx +161 -0
- package/.framework/src/pages/admin/AIAgentsPage.tsx +318 -0
- package/.framework/src/pages/admin/APIKeyDetailPage.tsx +199 -0
- package/.framework/src/pages/admin/APIKeysPage.tsx +303 -0
- package/.framework/src/pages/admin/AlertsConfigPage.tsx +523 -0
- package/.framework/src/pages/admin/AppDetailPage.tsx +493 -0
- package/.framework/src/pages/admin/AppsPage.tsx +355 -0
- package/.framework/src/pages/admin/DesignedPage.tsx +491 -0
- package/.framework/src/pages/admin/EmbeddingDetailPage.tsx +534 -0
- package/.framework/src/pages/admin/EmbeddingsPage.tsx +424 -0
- package/.framework/src/pages/admin/ExtendedShadcnTestPage.tsx +176 -0
- package/.framework/src/pages/admin/IncrementalShadcnTestPage.tsx +109 -0
- package/.framework/src/pages/admin/IntegratedDashboard.tsx +402 -0
- package/.framework/src/pages/admin/IntegrationDetailPage.tsx +187 -0
- package/.framework/src/pages/admin/IntegrationsPage.tsx +301 -0
- package/.framework/src/pages/admin/LogsPage.tsx +283 -0
- package/.framework/src/pages/admin/MinimalShadcnTestPage.tsx +85 -0
- package/.framework/src/pages/admin/ObservabilityDashboard.tsx +470 -0
- package/.framework/src/pages/admin/PipelineDetailPage.tsx +183 -0
- package/.framework/src/pages/admin/PipelineExecutionsPage.tsx +279 -0
- package/.framework/src/pages/admin/PipelinesPage.tsx +390 -0
- package/.framework/src/pages/admin/PromptConfigDetailPage.tsx +299 -0
- package/.framework/src/pages/admin/PromptConfigsPage.tsx +292 -0
- package/.framework/src/pages/admin/ProperlyDesignedPage.tsx +434 -0
- package/.framework/src/pages/admin/RoleDetailPage.tsx +273 -0
- package/.framework/src/pages/admin/RolesPage.tsx +292 -0
- package/.framework/src/pages/admin/SelectTestPage.tsx +61 -0
- package/.framework/src/pages/admin/ShadcnTestPage.tsx +588 -0
- package/.framework/src/pages/admin/SimpleDashboard.tsx +387 -0
- package/.framework/src/pages/admin/TestRunDetailPage.tsx +172 -0
- package/.framework/src/pages/admin/TestingDashboard.tsx +257 -0
- package/.framework/src/pages/admin/TimerDetailPage.tsx +151 -0
- package/.framework/src/pages/admin/TimersPage.tsx +376 -0
- package/.framework/src/pages/admin/TriggerDetailPage.tsx +149 -0
- package/.framework/src/pages/admin/TriggersPage.tsx +381 -0
- package/.framework/src/pages/admin/TypeDetailPage.tsx +694 -0
- package/.framework/src/pages/admin/TypesPage.tsx +295 -0
- package/.framework/src/pages/auth/LoginPage.tsx +188 -0
- package/.framework/src/pages/auth/RegisterPage.tsx +163 -0
- package/.framework/src/pages/spine-framework/APIPage.tsx +17 -0
- package/.framework/src/pages/spine-framework/CLIPage.tsx +25 -0
- package/.framework/src/types/auth.ts +125 -0
- package/.framework/src/types/types.ts +407 -0
- package/STRUCTURE.md +150 -0
- package/config/components.json +25 -0
- package/config/deno.lock +108 -0
- package/config/package-lock.json +17183 -0
- package/config/postcss.config.cjs +10 -0
- package/config/tailwind.config.cjs +78 -0
- package/config/tsconfig.build.json +32 -0
- package/config/tsconfig.cli.json +18 -0
- package/config/tsconfig.json +41 -0
- package/config/tsconfig.node.json +17 -0
- package/config/tsconfig.node.tsbuildinfo +1 -0
- package/config/tsconfig.tsbuildinfo +1 -0
- package/config/typedoc.json +16 -0
- package/config/vite.config.d.ts +2 -0
- package/config/vite.config.ts +72 -0
- package/dist/cli/commands/agents.d.ts +39 -0
- package/dist/cli/commands/agents.d.ts.map +1 -0
- package/dist/cli/commands/auth.d.ts +36 -0
- package/dist/cli/commands/auth.d.ts.map +1 -0
- package/dist/cli/commands/create-app.d.ts +23 -0
- package/dist/cli/commands/create-app.d.ts.map +1 -0
- package/dist/cli/commands/dev.d.ts +39 -0
- package/dist/cli/commands/dev.d.ts.map +1 -0
- package/dist/cli/commands/doctor.d.ts +42 -0
- package/dist/cli/commands/doctor.d.ts.map +1 -0
- package/dist/cli/commands/generate.d.ts +36 -0
- package/dist/cli/commands/generate.d.ts.map +1 -0
- package/dist/cli/commands/init.d.ts +30 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/install-app.d.ts +30 -0
- package/dist/cli/commands/install-app.d.ts.map +1 -0
- package/dist/cli/commands/items.d.ts +45 -0
- package/dist/cli/commands/items.d.ts.map +1 -0
- package/dist/cli/commands/migrations.d.ts +41 -0
- package/dist/cli/commands/migrations.d.ts.map +1 -0
- package/dist/cli/commands/pipelines.d.ts +40 -0
- package/dist/cli/commands/pipelines.d.ts.map +1 -0
- package/dist/cli/commands/status.d.ts +23 -0
- package/dist/cli/commands/status.d.ts.map +1 -0
- package/dist/cli/commands/system.d.ts +29 -0
- package/dist/cli/commands/system.d.ts.map +1 -0
- package/dist/cli/commands/test.d.ts +46 -0
- package/dist/cli/commands/test.d.ts.map +1 -0
- package/dist/cli/commands/uninstall-app.d.ts +23 -0
- package/dist/cli/commands/uninstall-app.d.ts.map +1 -0
- package/dist/cli/context.d.ts +88 -0
- package/dist/cli/context.d.ts.map +1 -0
- package/dist/cli/env-loader.d.ts +14 -0
- package/dist/cli/env-loader.d.ts.map +1 -0
- package/dist/cli/index.d.ts +41 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/functions/_shared/agent-runner.d.ts +156 -0
- package/dist/functions/_shared/agent-runner.d.ts.map +1 -0
- package/dist/functions/_shared/app-manifest.d.ts +68 -0
- package/dist/functions/_shared/app-manifest.d.ts.map +1 -0
- package/dist/functions/_shared/audit.d.ts +91 -0
- package/dist/functions/_shared/audit.d.ts.map +1 -0
- package/dist/functions/_shared/db.d.ts +125 -0
- package/dist/functions/_shared/db.d.ts.map +1 -0
- package/dist/functions/_shared/index.d.ts +298 -0
- package/dist/functions/_shared/index.d.ts.map +1 -0
- package/dist/functions/_shared/middleware.d.ts +315 -0
- package/dist/functions/_shared/middleware.d.ts.map +1 -0
- package/dist/functions/_shared/permissions.d.ts +626 -0
- package/dist/functions/_shared/permissions.d.ts.map +1 -0
- package/dist/functions/_shared/pipeline-runner.d.ts +124 -0
- package/dist/functions/_shared/pipeline-runner.d.ts.map +1 -0
- package/dist/functions/_shared/principal.d.ts +284 -0
- package/dist/functions/_shared/principal.d.ts.map +1 -0
- package/dist/functions/_shared/schema-utils.d.ts +181 -0
- package/dist/functions/_shared/schema-utils.d.ts.map +1 -0
- package/dist/functions/_shared/testing.d.ts +172 -0
- package/dist/functions/_shared/testing.d.ts.map +1 -0
- package/dist/functions/_shared/trigger-engine.d.ts +140 -0
- package/dist/functions/_shared/trigger-engine.d.ts.map +1 -0
- package/dist/functions/_shared/webhook-registration.d.ts +81 -0
- package/dist/functions/_shared/webhook-registration.d.ts.map +1 -0
- package/dist/functions/_shared/webhook-registry.d.ts +57 -0
- package/dist/functions/_shared/webhook-registry.d.ts.map +1 -0
- package/dist/functions/account-nodes.d.ts +48 -0
- package/dist/functions/account-nodes.d.ts.map +1 -0
- package/dist/functions/admin-data.d.ts +178 -0
- package/dist/functions/admin-data.d.ts.map +1 -0
- package/dist/functions/ai-agents.d.ts +125 -0
- package/dist/functions/ai-agents.d.ts.map +1 -0
- package/dist/functions/api-keys.d.ts +140 -0
- package/dist/functions/api-keys.d.ts.map +1 -0
- package/dist/functions/apps.d.ts +163 -0
- package/dist/functions/apps.d.ts.map +1 -0
- package/dist/functions/auth.d.ts +74 -0
- package/dist/functions/auth.d.ts.map +1 -0
- package/dist/functions/debug-auth.d.ts +33 -0
- package/dist/functions/debug-auth.d.ts.map +1 -0
- package/dist/functions/embeddings.d.ts +205 -0
- package/dist/functions/embeddings.d.ts.map +1 -0
- package/dist/functions/integration-routes.d.ts +45 -0
- package/dist/functions/integration-routes.d.ts.map +1 -0
- package/dist/functions/integrations.d.ts +124 -0
- package/dist/functions/integrations.d.ts.map +1 -0
- package/dist/functions/item-progress.d.ts +41 -0
- package/dist/functions/item-progress.d.ts.map +1 -0
- package/dist/functions/logs.d.ts +162 -0
- package/dist/functions/logs.d.ts.map +1 -0
- package/dist/functions/observability.d.ts +123 -0
- package/dist/functions/observability.d.ts.map +1 -0
- package/dist/functions/pipeline-executions.d.ts +190 -0
- package/dist/functions/pipeline-executions.d.ts.map +1 -0
- package/dist/functions/pipelines.d.ts +171 -0
- package/dist/functions/pipelines.d.ts.map +1 -0
- package/dist/functions/prompt-configs.d.ts +125 -0
- package/dist/functions/prompt-configs.d.ts.map +1 -0
- package/dist/functions/roles.d.ts +118 -0
- package/dist/functions/roles.d.ts.map +1 -0
- package/dist/functions/system-cron.d.ts +65 -0
- package/dist/functions/system-cron.d.ts.map +1 -0
- package/dist/functions/system.d.ts +29 -0
- package/dist/functions/system.d.ts.map +1 -0
- package/dist/functions/tests.d.ts +28 -0
- package/dist/functions/tests.d.ts.map +1 -0
- package/dist/functions/timers.d.ts +139 -0
- package/dist/functions/timers.d.ts.map +1 -0
- package/dist/functions/triggers.d.ts +203 -0
- package/dist/functions/triggers.d.ts.map +1 -0
- package/dist/functions/types.d.ts +151 -0
- package/dist/functions/types.d.ts.map +1 -0
- package/dist/src/types/types.d.ts +364 -0
- package/dist/src/types/types.d.ts.map +1 -0
- package/package.json +192 -0
- package/scripts/app-install-cli.ts +286 -0
- package/scripts/assemble-frontend.sh +79 -0
- package/scripts/assemble-functions.sh +62 -0
- package/scripts/assemble.sh +35 -0
- package/scripts/boundary-check.sh +106 -0
- package/scripts/build-manifest.sh +80 -0
- package/scripts/check-core-integrity.sh +82 -0
- package/scripts/ingest-chunks.cjs +202 -0
- package/scripts/kb-chunk-parser.cjs +312 -0
- package/scripts/kb-chunk-parser.ts +330 -0
- package/scripts/load-test-app-install.ts +484 -0
- package/scripts/netlify-dev-wrapper.sh +22 -0
- package/scripts/verify-integrity.sh +69 -0
|
@@ -0,0 +1,470 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module src/pages/admin/ObservabilityDashboard
|
|
3
|
+
* @audience installer
|
|
4
|
+
* @layer frontend-page
|
|
5
|
+
* @stability stable
|
|
6
|
+
*
|
|
7
|
+
* System observability dashboard. Fetches aggregated metrics from
|
|
8
|
+
* `/api/admin-data?action=metrics` and renders time-series line charts,
|
|
9
|
+
* bar charts, and pie charts using `recharts`. Covers request rates,
|
|
10
|
+
* error rates, pipeline execution counts, and resource utilisation.
|
|
11
|
+
* Data is re-fetched on a configurable auto-refresh interval.
|
|
12
|
+
*
|
|
13
|
+
* @seeAlso src/pages/admin/LogsPage.tsx
|
|
14
|
+
* @seeAlso src/pages/admin/AlertsConfigPage.tsx
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import React, { useState, useEffect, useMemo } from 'react'
|
|
18
|
+
import {
|
|
19
|
+
LineChart,
|
|
20
|
+
Line,
|
|
21
|
+
BarChart,
|
|
22
|
+
Bar,
|
|
23
|
+
XAxis,
|
|
24
|
+
YAxis,
|
|
25
|
+
CartesianGrid,
|
|
26
|
+
Tooltip,
|
|
27
|
+
ResponsiveContainer,
|
|
28
|
+
PieChart,
|
|
29
|
+
Pie,
|
|
30
|
+
Cell,
|
|
31
|
+
} from 'recharts'
|
|
32
|
+
import { BarChart3, AlertTriangle, Clock, RefreshCw, Bell } from 'lucide-react';
|
|
33
|
+
import { LoadingSpinner } from '../../components/ui/LoadingSpinner'
|
|
34
|
+
import { Button } from '../../components/ui/button'
|
|
35
|
+
import { useApi } from '../../hooks/useApi'
|
|
36
|
+
import { apiFetch } from '../../lib/api'
|
|
37
|
+
import { formatDateTime } from '../../lib/utils'
|
|
38
|
+
import { Link } from 'react-router-dom'
|
|
39
|
+
|
|
40
|
+
interface MetricData {
|
|
41
|
+
bucket: string
|
|
42
|
+
count: number
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
interface ErrorRateData {
|
|
46
|
+
total: number
|
|
47
|
+
errors: number
|
|
48
|
+
rate: number
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
interface LatencyData {
|
|
52
|
+
p50: number
|
|
53
|
+
p90: number
|
|
54
|
+
p95: number
|
|
55
|
+
p99: number
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
interface PipelineStats {
|
|
59
|
+
pipeline_id: string
|
|
60
|
+
success_count: number
|
|
61
|
+
failure_count: number
|
|
62
|
+
avg_duration_ms: number
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
interface TopActor {
|
|
66
|
+
principal_id: string
|
|
67
|
+
event_count: number
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
interface AlertEvent {
|
|
71
|
+
id: string
|
|
72
|
+
metric: string
|
|
73
|
+
threshold_value: number
|
|
74
|
+
actual_value: number
|
|
75
|
+
created_at: string
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const COLORS = ['#3b82f6', '#10b981', '#f59e0b', '#ef4444', '#8b5cf6']
|
|
79
|
+
|
|
80
|
+
export function ObservabilityDashboard() {
|
|
81
|
+
const [timeRange, setTimeRange] = useState<'1h' | '24h' | '7d'>('24h')
|
|
82
|
+
const [refreshInterval, setRefreshInterval] = useState(30)
|
|
83
|
+
|
|
84
|
+
const timeParams = useMemo(() => {
|
|
85
|
+
const now = new Date()
|
|
86
|
+
let from = new Date()
|
|
87
|
+
switch (timeRange) {
|
|
88
|
+
case '1h':
|
|
89
|
+
from = new Date(now.getTime() - 60 * 60 * 1000)
|
|
90
|
+
break
|
|
91
|
+
case '24h':
|
|
92
|
+
from = new Date(now.getTime() - 24 * 60 * 60 * 1000)
|
|
93
|
+
break
|
|
94
|
+
case '7d':
|
|
95
|
+
from = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000)
|
|
96
|
+
break
|
|
97
|
+
}
|
|
98
|
+
return {
|
|
99
|
+
from: from.toISOString(),
|
|
100
|
+
to: now.toISOString(),
|
|
101
|
+
}
|
|
102
|
+
}, [timeRange])
|
|
103
|
+
|
|
104
|
+
// Event volume data
|
|
105
|
+
const { data: eventVolume, loading: loadingVolume, refetch: refetchVolume } = useApi<MetricData[]>(
|
|
106
|
+
async () => {
|
|
107
|
+
const bucket = timeRange === '1h' ? 'minute' : timeRange === '24h' ? 'hour' : 'day'
|
|
108
|
+
const response = await apiFetch(
|
|
109
|
+
`/api/observability?action=event_volume&from=${encodeURIComponent(timeParams.from)}&to=${encodeURIComponent(timeParams.to)}&bucket=${bucket}`
|
|
110
|
+
)
|
|
111
|
+
if (!response.ok) return []
|
|
112
|
+
const result = await response.json()
|
|
113
|
+
return (result.data || result || []).map((d: any) => ({
|
|
114
|
+
bucket: new Date(d.bucket).toLocaleString('en-US', {
|
|
115
|
+
month: 'short',
|
|
116
|
+
day: 'numeric',
|
|
117
|
+
hour: 'numeric',
|
|
118
|
+
minute: timeRange === '1h' ? 'numeric' : undefined,
|
|
119
|
+
}),
|
|
120
|
+
count: parseInt(d.count),
|
|
121
|
+
}))
|
|
122
|
+
},
|
|
123
|
+
{ immediate: true, deps: [timeRange] }
|
|
124
|
+
)
|
|
125
|
+
|
|
126
|
+
// Error rate data
|
|
127
|
+
const { data: errorRate, loading: loadingError, refetch: refetchError } = useApi<ErrorRateData>(
|
|
128
|
+
async () => {
|
|
129
|
+
const response = await apiFetch(
|
|
130
|
+
`/api/observability?action=error_rate&from=${encodeURIComponent(timeParams.from)}&to=${encodeURIComponent(timeParams.to)}`
|
|
131
|
+
)
|
|
132
|
+
if (!response.ok) return { total: 0, errors: 0, rate: 0 }
|
|
133
|
+
const result = await response.json()
|
|
134
|
+
return result.data || result || { total: 0, errors: 0, rate: 0 }
|
|
135
|
+
},
|
|
136
|
+
{ immediate: true, deps: [timeRange] }
|
|
137
|
+
)
|
|
138
|
+
|
|
139
|
+
// Latency percentiles
|
|
140
|
+
const { data: latency, loading: loadingLatency, refetch: refetchLatency } = useApi<LatencyData>(
|
|
141
|
+
async () => {
|
|
142
|
+
const response = await apiFetch(
|
|
143
|
+
`/api/observability?action=latency_percentiles&from=${encodeURIComponent(timeParams.from)}&to=${encodeURIComponent(timeParams.to)}`
|
|
144
|
+
)
|
|
145
|
+
if (!response.ok) return { p50: 0, p90: 0, p95: 0, p99: 0 }
|
|
146
|
+
const result = await response.json()
|
|
147
|
+
return result.data || result || { p50: 0, p90: 0, p95: 0, p99: 0 }
|
|
148
|
+
},
|
|
149
|
+
{ immediate: true, deps: [timeRange] }
|
|
150
|
+
)
|
|
151
|
+
|
|
152
|
+
// Pipeline stats
|
|
153
|
+
const { data: pipelineStats, loading: loadingPipelines, refetch: refetchPipelines } = useApi<PipelineStats[]>(
|
|
154
|
+
async () => {
|
|
155
|
+
const response = await apiFetch(
|
|
156
|
+
`/api/observability?action=pipeline_stats&from=${encodeURIComponent(timeParams.from)}&to=${encodeURIComponent(timeParams.to)}`
|
|
157
|
+
)
|
|
158
|
+
if (!response.ok) return []
|
|
159
|
+
const result = await response.json()
|
|
160
|
+
return result.data || result || []
|
|
161
|
+
},
|
|
162
|
+
{ immediate: true, deps: [timeRange] }
|
|
163
|
+
)
|
|
164
|
+
|
|
165
|
+
// Top actors
|
|
166
|
+
const { data: topActors, loading: loadingActors, refetch: refetchActors } = useApi<TopActor[]>(
|
|
167
|
+
async () => {
|
|
168
|
+
const response = await apiFetch(
|
|
169
|
+
`/api/observability?action=top_actors&from=${encodeURIComponent(timeParams.from)}&to=${encodeURIComponent(timeParams.to)}&limit=5`
|
|
170
|
+
)
|
|
171
|
+
if (!response.ok) return []
|
|
172
|
+
const result = await response.json()
|
|
173
|
+
return result.data || result || []
|
|
174
|
+
},
|
|
175
|
+
{ immediate: true, deps: [timeRange] }
|
|
176
|
+
)
|
|
177
|
+
|
|
178
|
+
// Recent alerts
|
|
179
|
+
const { data: recentAlerts } = useApi<AlertEvent[]>(
|
|
180
|
+
async () => {
|
|
181
|
+
const response = await apiFetch('/api/logs?action=account&event=threshold.breached&limit=5')
|
|
182
|
+
if (!response.ok) return []
|
|
183
|
+
const result = await response.json()
|
|
184
|
+
return (result.data || result || []).slice(0, 5)
|
|
185
|
+
},
|
|
186
|
+
{ immediate: true }
|
|
187
|
+
)
|
|
188
|
+
|
|
189
|
+
// Auto refresh
|
|
190
|
+
useEffect(() => {
|
|
191
|
+
const interval = setInterval(() => {
|
|
192
|
+
refetchVolume()
|
|
193
|
+
refetchError()
|
|
194
|
+
refetchLatency()
|
|
195
|
+
refetchPipelines()
|
|
196
|
+
refetchActors()
|
|
197
|
+
}, refreshInterval * 1000)
|
|
198
|
+
return () => clearInterval(interval)
|
|
199
|
+
}, [refreshInterval])
|
|
200
|
+
|
|
201
|
+
const pipelineChartData = useMemo(() => {
|
|
202
|
+
return (pipelineStats || []).map((stat) => ({
|
|
203
|
+
name: stat.pipeline_id.slice(0, 8),
|
|
204
|
+
success: parseInt(stat.success_count as any),
|
|
205
|
+
failure: parseInt(stat.failure_count as any),
|
|
206
|
+
avgDuration: Math.round(parseFloat(stat.avg_duration_ms as any)),
|
|
207
|
+
}))
|
|
208
|
+
}, [pipelineStats])
|
|
209
|
+
|
|
210
|
+
const statsCards = [
|
|
211
|
+
{
|
|
212
|
+
title: 'Total Events',
|
|
213
|
+
value: eventVolume?.reduce((sum, d) => sum + d.count, 0) || 0,
|
|
214
|
+
icon: BarChart3,
|
|
215
|
+
iconColor: 'text-blue-500',
|
|
216
|
+
bgColor: 'bg-blue-50',
|
|
217
|
+
},
|
|
218
|
+
{
|
|
219
|
+
title: 'Error Rate',
|
|
220
|
+
value: `${(errorRate?.rate || 0).toFixed(2)}%`,
|
|
221
|
+
icon: AlertTriangle,
|
|
222
|
+
iconColor: errorRate && errorRate.rate > 5 ? 'text-red-500' : 'text-amber-500',
|
|
223
|
+
bgColor: errorRate && errorRate.rate > 5 ? 'bg-red-50' : 'bg-amber-50',
|
|
224
|
+
},
|
|
225
|
+
{
|
|
226
|
+
title: 'P95 Latency',
|
|
227
|
+
value: `${Math.round(latency?.p95 || 0)}ms`,
|
|
228
|
+
icon: Clock,
|
|
229
|
+
iconColor: latency && latency.p95 > 2000 ? 'text-red-500' : 'text-green-500',
|
|
230
|
+
bgColor: latency && latency.p95 > 2000 ? 'bg-red-50' : 'bg-green-50',
|
|
231
|
+
},
|
|
232
|
+
{
|
|
233
|
+
title: 'Active Pipelines',
|
|
234
|
+
value: pipelineStats?.length || 0,
|
|
235
|
+
icon: RefreshCw,
|
|
236
|
+
iconColor: 'text-purple-500',
|
|
237
|
+
bgColor: 'bg-purple-50',
|
|
238
|
+
},
|
|
239
|
+
]
|
|
240
|
+
|
|
241
|
+
const isLoading = loadingVolume || loadingError || loadingLatency || loadingPipelines || loadingActors
|
|
242
|
+
|
|
243
|
+
const handleRefresh = () => {
|
|
244
|
+
refetchVolume()
|
|
245
|
+
refetchError()
|
|
246
|
+
refetchLatency()
|
|
247
|
+
refetchPipelines()
|
|
248
|
+
refetchActors()
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
return (
|
|
252
|
+
<div className="space-y-6">
|
|
253
|
+
{/* Header */}
|
|
254
|
+
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
|
|
255
|
+
<div>
|
|
256
|
+
<h1 className="text-2xl font-bold text-slate-900">Observability Dashboard</h1>
|
|
257
|
+
<p className="text-slate-500 mt-1">System health and performance metrics</p>
|
|
258
|
+
</div>
|
|
259
|
+
<div className="flex items-center gap-3">
|
|
260
|
+
<select
|
|
261
|
+
value={timeRange}
|
|
262
|
+
onChange={(e) => setTimeRange(e.target.value as any)}
|
|
263
|
+
className="px-3 py-2 text-sm border border-slate-300 rounded-lg focus:ring-2 focus:ring-navy focus:border-navy"
|
|
264
|
+
>
|
|
265
|
+
<option value="1h">Last Hour</option>
|
|
266
|
+
<option value="24h">Last 24 Hours</option>
|
|
267
|
+
<option value="7d">Last 7 Days</option>
|
|
268
|
+
</select>
|
|
269
|
+
<select
|
|
270
|
+
value={refreshInterval}
|
|
271
|
+
onChange={(e) => setRefreshInterval(parseInt(e.target.value))}
|
|
272
|
+
className="px-3 py-2 text-sm border border-slate-300 rounded-lg focus:ring-2 focus:ring-navy focus:border-navy"
|
|
273
|
+
>
|
|
274
|
+
<option value={10}>Refresh: 10s</option>
|
|
275
|
+
<option value={30}>Refresh: 30s</option>
|
|
276
|
+
<option value={60}>Refresh: 1m</option>
|
|
277
|
+
</select>
|
|
278
|
+
<Button variant="secondary" onClick={handleRefresh} className="text-sm">
|
|
279
|
+
Refresh Now
|
|
280
|
+
</Button>
|
|
281
|
+
</div>
|
|
282
|
+
</div>
|
|
283
|
+
|
|
284
|
+
{/* Stats Cards */}
|
|
285
|
+
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-4 gap-4">
|
|
286
|
+
{statsCards.map((card) => (
|
|
287
|
+
<div key={card.title} className="bg-white rounded-lg border border-slate-200 p-4">
|
|
288
|
+
<div className="flex items-center justify-between">
|
|
289
|
+
<div>
|
|
290
|
+
<p className="text-sm font-medium text-slate-500">{card.title}</p>
|
|
291
|
+
<p className="text-2xl font-bold text-slate-900 mt-1">{card.value}</p>
|
|
292
|
+
</div>
|
|
293
|
+
<div className={`p-3 rounded-lg ${card.bgColor}`}>
|
|
294
|
+
<card.icon className={`h-6 w-6 ${card.iconColor}`} />
|
|
295
|
+
</div>
|
|
296
|
+
</div>
|
|
297
|
+
</div>
|
|
298
|
+
))}
|
|
299
|
+
</div>
|
|
300
|
+
|
|
301
|
+
{isLoading && !eventVolume ? (
|
|
302
|
+
<div className="flex items-center justify-center h-64">
|
|
303
|
+
<LoadingSpinner className="w-8 h-8" />
|
|
304
|
+
</div>
|
|
305
|
+
) : (
|
|
306
|
+
<>
|
|
307
|
+
{/* Charts Row 1 */}
|
|
308
|
+
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
|
309
|
+
{/* Event Volume Chart */}
|
|
310
|
+
<div className="bg-white rounded-lg border border-slate-200 p-6">
|
|
311
|
+
<h3 className="text-lg font-semibold text-slate-900 mb-4">Event Volume</h3>
|
|
312
|
+
<div className="h-64">
|
|
313
|
+
<ResponsiveContainer width="100%" height="100%">
|
|
314
|
+
<LineChart data={eventVolume || []}>
|
|
315
|
+
<CartesianGrid strokeDasharray="3 3" stroke="#e2e8f0" />
|
|
316
|
+
<XAxis dataKey="bucket" tick={{ fontSize: 12 }} />
|
|
317
|
+
<YAxis tick={{ fontSize: 12 }} />
|
|
318
|
+
<Tooltip
|
|
319
|
+
contentStyle={{
|
|
320
|
+
backgroundColor: '#fff',
|
|
321
|
+
border: '1px solid #e2e8f0',
|
|
322
|
+
borderRadius: '6px',
|
|
323
|
+
}}
|
|
324
|
+
/>
|
|
325
|
+
<Line
|
|
326
|
+
type="monotone"
|
|
327
|
+
dataKey="count"
|
|
328
|
+
stroke="#3b82f6"
|
|
329
|
+
strokeWidth={2}
|
|
330
|
+
dot={false}
|
|
331
|
+
/>
|
|
332
|
+
</LineChart>
|
|
333
|
+
</ResponsiveContainer>
|
|
334
|
+
</div>
|
|
335
|
+
</div>
|
|
336
|
+
|
|
337
|
+
{/* Latency Percentiles */}
|
|
338
|
+
<div className="bg-white rounded-lg border border-slate-200 p-6">
|
|
339
|
+
<h3 className="text-lg font-semibold text-slate-900 mb-4">Latency Percentiles</h3>
|
|
340
|
+
<div className="h-64">
|
|
341
|
+
{latency ? (
|
|
342
|
+
<div className="flex items-center justify-center h-full">
|
|
343
|
+
<div className="grid grid-cols-3 gap-8">
|
|
344
|
+
<div className="text-center">
|
|
345
|
+
<div className="text-3xl font-bold text-blue-500">{Math.round(latency.p50)}ms</div>
|
|
346
|
+
<div className="text-sm text-slate-500 mt-1">P50 (Median)</div>
|
|
347
|
+
</div>
|
|
348
|
+
<div className="text-center">
|
|
349
|
+
<div className="text-3xl font-bold text-amber-500">{Math.round(latency.p90)}ms</div>
|
|
350
|
+
<div className="text-sm text-slate-500 mt-1">P90</div>
|
|
351
|
+
</div>
|
|
352
|
+
<div className="text-center">
|
|
353
|
+
<div className="text-3xl font-bold text-red-500">{Math.round(latency.p99)}ms</div>
|
|
354
|
+
<div className="text-sm text-slate-500 mt-1">P99</div>
|
|
355
|
+
</div>
|
|
356
|
+
</div>
|
|
357
|
+
</div>
|
|
358
|
+
) : (
|
|
359
|
+
<div className="flex items-center justify-center h-full text-slate-400">
|
|
360
|
+
No latency data available
|
|
361
|
+
</div>
|
|
362
|
+
)}
|
|
363
|
+
</div>
|
|
364
|
+
</div>
|
|
365
|
+
</div>
|
|
366
|
+
|
|
367
|
+
{/* Charts Row 2 */}
|
|
368
|
+
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
|
369
|
+
{/* Pipeline Health */}
|
|
370
|
+
<div className="bg-white rounded-lg border border-slate-200 p-6">
|
|
371
|
+
<h3 className="text-lg font-semibold text-slate-900 mb-4">Pipeline Health</h3>
|
|
372
|
+
<div className="h-64">
|
|
373
|
+
{pipelineChartData.length > 0 ? (
|
|
374
|
+
<ResponsiveContainer width="100%" height="100%">
|
|
375
|
+
<BarChart data={pipelineChartData}>
|
|
376
|
+
<CartesianGrid strokeDasharray="3 3" stroke="#e2e8f0" />
|
|
377
|
+
<XAxis dataKey="name" tick={{ fontSize: 12 }} />
|
|
378
|
+
<YAxis tick={{ fontSize: 12 }} />
|
|
379
|
+
<Tooltip
|
|
380
|
+
contentStyle={{
|
|
381
|
+
backgroundColor: '#fff',
|
|
382
|
+
border: '1px solid #e2e8f0',
|
|
383
|
+
borderRadius: '6px',
|
|
384
|
+
}}
|
|
385
|
+
/>
|
|
386
|
+
<Bar dataKey="success" fill="#10b981" />
|
|
387
|
+
<Bar dataKey="failure" fill="#ef4444" />
|
|
388
|
+
</BarChart>
|
|
389
|
+
</ResponsiveContainer>
|
|
390
|
+
) : (
|
|
391
|
+
<div className="flex items-center justify-center h-full text-slate-400">
|
|
392
|
+
No pipeline executions in this time range
|
|
393
|
+
</div>
|
|
394
|
+
)}
|
|
395
|
+
</div>
|
|
396
|
+
</div>
|
|
397
|
+
|
|
398
|
+
{/* Top Actors */}
|
|
399
|
+
<div className="bg-white rounded-lg border border-slate-200 p-6">
|
|
400
|
+
<h3 className="text-lg font-semibold text-slate-900 mb-4">Top Actors</h3>
|
|
401
|
+
<div className="space-y-3">
|
|
402
|
+
{topActors && topActors.length > 0 ? (
|
|
403
|
+
topActors.map((actor, index) => (
|
|
404
|
+
<div key={actor.principal_id} className="flex items-center justify-between py-2">
|
|
405
|
+
<div className="flex items-center gap-3">
|
|
406
|
+
<div
|
|
407
|
+
className="w-8 h-8 rounded-full flex items-center justify-center text-sm font-medium"
|
|
408
|
+
style={{ backgroundColor: COLORS[index % COLORS.length] + '20', color: COLORS[index % COLORS.length] }}
|
|
409
|
+
>
|
|
410
|
+
{index + 1}
|
|
411
|
+
</div>
|
|
412
|
+
<span className="text-sm font-medium text-slate-700">
|
|
413
|
+
{actor.principal_id.slice(0, 8)}...
|
|
414
|
+
</span>
|
|
415
|
+
</div>
|
|
416
|
+
<span className="text-sm text-slate-500">{actor.event_count} events</span>
|
|
417
|
+
</div>
|
|
418
|
+
))
|
|
419
|
+
) : (
|
|
420
|
+
<div className="text-center text-slate-400 py-8">No activity data</div>
|
|
421
|
+
)}
|
|
422
|
+
</div>
|
|
423
|
+
</div>
|
|
424
|
+
</div>
|
|
425
|
+
|
|
426
|
+
{/* Recent Alerts */}
|
|
427
|
+
<div className="bg-white rounded-lg border border-slate-200 p-6">
|
|
428
|
+
<div className="flex items-center justify-between mb-4">
|
|
429
|
+
<h3 className="text-lg font-semibold text-slate-900">Recent Alerts</h3>
|
|
430
|
+
<Link
|
|
431
|
+
to="/spine-framework/admin/observability/alerts"
|
|
432
|
+
className="text-sm text-navy hover:underline flex items-center gap-1"
|
|
433
|
+
>
|
|
434
|
+
<Bell className="h-4 w-4" />
|
|
435
|
+
Manage Alerts
|
|
436
|
+
</Link>
|
|
437
|
+
</div>
|
|
438
|
+
<div className="space-y-2">
|
|
439
|
+
{recentAlerts && recentAlerts.length > 0 ? (
|
|
440
|
+
recentAlerts.map((alert) => (
|
|
441
|
+
<div
|
|
442
|
+
key={alert.id}
|
|
443
|
+
className="flex items-center justify-between py-3 px-4 bg-red-50 rounded-lg border border-red-100"
|
|
444
|
+
>
|
|
445
|
+
<div className="flex items-center gap-3">
|
|
446
|
+
<AlertTriangle className="h-5 w-5 text-red-500" />
|
|
447
|
+
<div>
|
|
448
|
+
<span className="font-medium text-slate-900">
|
|
449
|
+
{alert.metric} threshold breached
|
|
450
|
+
</span>
|
|
451
|
+
<span className="text-sm text-slate-500 ml-2">
|
|
452
|
+
Expected: {alert.threshold_value}, Got: {alert.actual_value?.toFixed(2) ?? 'N/A'}
|
|
453
|
+
</span>
|
|
454
|
+
</div>
|
|
455
|
+
</div>
|
|
456
|
+
<span className="text-sm text-slate-500">
|
|
457
|
+
{formatDateTime(alert.created_at)}
|
|
458
|
+
</span>
|
|
459
|
+
</div>
|
|
460
|
+
))
|
|
461
|
+
) : (
|
|
462
|
+
<div className="text-center text-slate-400 py-8">No recent alerts</div>
|
|
463
|
+
)}
|
|
464
|
+
</div>
|
|
465
|
+
</div>
|
|
466
|
+
</>
|
|
467
|
+
)}
|
|
468
|
+
</div>
|
|
469
|
+
)
|
|
470
|
+
}
|
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module src/pages/admin/PipelineDetailPage
|
|
3
|
+
*/
|
|
4
|
+
import { useState, useEffect } from 'react'
|
|
5
|
+
import { useParams, useNavigate } from 'react-router-dom'
|
|
6
|
+
import { apiFetch } from '../../lib/api'
|
|
7
|
+
import { Button } from '../../components/ui/button'
|
|
8
|
+
import { Input } from '../../components/ui/input'
|
|
9
|
+
import { Textarea } from '../../components/ui/textarea'
|
|
10
|
+
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '../../components/ui/select'
|
|
11
|
+
import { Checkbox } from '../../components/ui/checkbox'
|
|
12
|
+
import { Label } from '../../components/ui/label'
|
|
13
|
+
import { Card, CardContent, CardHeader, CardTitle } from '../../components/ui/card'
|
|
14
|
+
import { Skeleton } from '../../components/ui/skeleton'
|
|
15
|
+
import { Alert, AlertDescription, AlertTitle } from '../../components/ui/alert'
|
|
16
|
+
import { Badge } from '../../components/ui/badge'
|
|
17
|
+
import { ArrowLeft, Play, Pause, AlertCircle } from 'lucide-react'
|
|
18
|
+
import { formatDateTime } from '../../lib/utils'
|
|
19
|
+
|
|
20
|
+
interface Pipeline {
|
|
21
|
+
id: string
|
|
22
|
+
name: string
|
|
23
|
+
description?: string
|
|
24
|
+
status: 'draft' | 'active' | 'paused' | 'archived'
|
|
25
|
+
steps: Array<{ id: string; name: string; type: string; config: Record<string, any> }>
|
|
26
|
+
trigger_type: 'manual' | 'scheduled' | 'webhook' | 'event'
|
|
27
|
+
schedule?: string
|
|
28
|
+
is_active: boolean
|
|
29
|
+
last_run?: string
|
|
30
|
+
last_run_status?: 'success' | 'failed' | 'running'
|
|
31
|
+
run_count: number
|
|
32
|
+
created_at: string
|
|
33
|
+
updated_at: string
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const STATUSES = [
|
|
37
|
+
{ value: 'draft', label: 'Draft' },
|
|
38
|
+
{ value: 'active', label: 'Active' },
|
|
39
|
+
{ value: 'paused', label: 'Paused' },
|
|
40
|
+
{ value: 'archived', label: 'Archived' }
|
|
41
|
+
]
|
|
42
|
+
|
|
43
|
+
const TRIGGER_TYPES = [
|
|
44
|
+
{ value: 'manual', label: 'Manual' },
|
|
45
|
+
{ value: 'scheduled', label: 'Scheduled' },
|
|
46
|
+
{ value: 'webhook', label: 'Webhook' },
|
|
47
|
+
{ value: 'event', label: 'Event' }
|
|
48
|
+
]
|
|
49
|
+
|
|
50
|
+
export function PipelineDetailPage() {
|
|
51
|
+
const { id } = useParams<{ id: string }>()
|
|
52
|
+
const navigate = useNavigate()
|
|
53
|
+
const isCreateMode = !id || id === 'new'
|
|
54
|
+
const [pipeline, setPipeline] = useState<Pipeline | null>(null)
|
|
55
|
+
const [loading, setLoading] = useState(true)
|
|
56
|
+
const [error, setError] = useState<string | null>(null)
|
|
57
|
+
const [isEditing, setIsEditing] = useState(isCreateMode)
|
|
58
|
+
const [formData, setFormData] = useState({ name: '', description: '', status: 'draft' as const, trigger_type: 'manual' as const, schedule: '', is_active: true, steps: [] as any[] })
|
|
59
|
+
|
|
60
|
+
useEffect(() => {
|
|
61
|
+
if (isCreateMode) {
|
|
62
|
+
setLoading(false)
|
|
63
|
+
return
|
|
64
|
+
}
|
|
65
|
+
setLoading(true)
|
|
66
|
+
const fetchPipeline = async () => {
|
|
67
|
+
try {
|
|
68
|
+
const response = await apiFetch(`/api/pipelines?action=get&id=${id}`)
|
|
69
|
+
if (!response.ok) throw new Error('Failed to fetch pipeline')
|
|
70
|
+
const result = await response.json()
|
|
71
|
+
const data = result.data
|
|
72
|
+
setPipeline(data)
|
|
73
|
+
setFormData({
|
|
74
|
+
name: data.name, description: data.description || '', status: data.status, trigger_type: data.trigger_type,
|
|
75
|
+
schedule: data.schedule || '', is_active: data.is_active, steps: data.steps || []
|
|
76
|
+
})
|
|
77
|
+
} catch (err) {
|
|
78
|
+
setError(err instanceof Error ? err.message : 'Failed to load pipeline')
|
|
79
|
+
} finally {
|
|
80
|
+
setLoading(false)
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
fetchPipeline()
|
|
84
|
+
}, [id, isCreateMode])
|
|
85
|
+
|
|
86
|
+
const handleSave = async () => {
|
|
87
|
+
try {
|
|
88
|
+
const url = isCreateMode ? '/api/pipelines?action=create' : `/api/pipelines?action=update&id=${id}`
|
|
89
|
+
const method = isCreateMode ? 'POST' : 'PATCH'
|
|
90
|
+
const response = await apiFetch(url, { method, headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(formData) })
|
|
91
|
+
if (!response.ok) throw new Error('Failed to save')
|
|
92
|
+
if (isCreateMode) {
|
|
93
|
+
const result = await response.json()
|
|
94
|
+
navigate(`/spine-framework/admin/configs/pipelines/${result.data?.id || result.id}`)
|
|
95
|
+
} else setIsEditing(false)
|
|
96
|
+
} catch (err) {
|
|
97
|
+
setError(err instanceof Error ? err.message : 'Failed to save')
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
const toggleStatus = async () => {
|
|
102
|
+
if (!pipeline) return
|
|
103
|
+
const newStatus = pipeline.status === 'active' ? 'paused' : 'active'
|
|
104
|
+
try {
|
|
105
|
+
const response = await apiFetch(`/api/pipelines?action=update&id=${id}`, { method: 'PATCH', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ status: newStatus }) })
|
|
106
|
+
if (!response.ok) throw new Error('Failed to update status')
|
|
107
|
+
setPipeline({ ...pipeline, status: newStatus })
|
|
108
|
+
} catch (err) {
|
|
109
|
+
setError(err instanceof Error ? err.message : 'Failed to update status')
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
if (loading) return (
|
|
114
|
+
<div className="space-y-6">
|
|
115
|
+
<Skeleton className="h-8 w-48" />
|
|
116
|
+
<Card><CardHeader><Skeleton className="h-6 w-32" /></CardHeader><CardContent className="space-y-4"><Skeleton className="h-4 w-full" /><Skeleton className="h-4 w-2/3" /></CardContent></Card>
|
|
117
|
+
</div>
|
|
118
|
+
)
|
|
119
|
+
|
|
120
|
+
if (error && !isCreateMode) return (
|
|
121
|
+
<Alert variant="destructive"><AlertCircle className="h-4 w-4" /><AlertTitle>Error</AlertTitle><AlertDescription>{error}</AlertDescription></Alert>
|
|
122
|
+
)
|
|
123
|
+
|
|
124
|
+
return (
|
|
125
|
+
<div className="space-y-6">
|
|
126
|
+
<div className="flex items-center justify-between">
|
|
127
|
+
<div className="flex items-center gap-4">
|
|
128
|
+
<Button variant="ghost" onClick={() => isCreateMode ? navigate('/spine-framework/admin/configs/pipelines') : navigate(-1)}><ArrowLeft className="h-5 w-5" /></Button>
|
|
129
|
+
<div>
|
|
130
|
+
<h1 className="text-2xl font-bold">{isCreateMode ? 'Create Pipeline' : pipeline?.name}</h1>
|
|
131
|
+
{pipeline && <Badge variant={pipeline.status === 'active' ? 'default' : pipeline.status === 'paused' ? 'secondary' : 'outline'}>{pipeline.status}</Badge>}
|
|
132
|
+
</div>
|
|
133
|
+
</div>
|
|
134
|
+
<div className="flex gap-2">
|
|
135
|
+
{!isCreateMode && pipeline?.status !== 'archived' && (
|
|
136
|
+
<Button variant="outline" onClick={toggleStatus}>{pipeline?.status === 'active' ? <><Pause className="h-4 w-4 mr-2" />Pause</> : <><Play className="h-4 w-4 mr-2" />Activate</>}</Button>
|
|
137
|
+
)}
|
|
138
|
+
{isEditing ? (
|
|
139
|
+
<><Button variant="outline" onClick={() => isCreateMode ? navigate(-1) : setIsEditing(false)}>Cancel</Button><Button onClick={handleSave}>{isCreateMode ? 'Create' : 'Save'}</Button></>
|
|
140
|
+
) : !isCreateMode && <Button onClick={() => setIsEditing(true)}>Edit</Button>}
|
|
141
|
+
</div>
|
|
142
|
+
</div>
|
|
143
|
+
|
|
144
|
+
<Card><CardContent className="p-6">
|
|
145
|
+
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
|
|
146
|
+
<div className="space-y-4">
|
|
147
|
+
<h3 className="text-sm font-medium text-muted-foreground">Configuration</h3>
|
|
148
|
+
<div className="space-y-2"><Label>Name</Label>{isEditing ? <Input value={formData.name} onChange={e => setFormData({...formData, name: e.target.value})} /> : <div className="text-sm">{pipeline?.name}</div>}</div>
|
|
149
|
+
<div className="space-y-2"><Label>Status</Label>{isEditing ? <Select value={formData.status} onValueChange={v => setFormData({...formData, status: v as any})}><SelectTrigger><SelectValue /></SelectTrigger><SelectContent>{STATUSES.map(s => <SelectItem key={s.value} value={s.value}>{s.label}</SelectItem>)}</SelectContent></Select> : <Badge>{pipeline?.status}</Badge>}</div>
|
|
150
|
+
<div className="space-y-2"><Label>Trigger Type</Label>{isEditing ? <Select value={formData.trigger_type} onValueChange={v => setFormData({...formData, trigger_type: v as any})}><SelectTrigger><SelectValue /></SelectTrigger><SelectContent>{TRIGGER_TYPES.map(t => <SelectItem key={t.value} value={t.value}>{t.label}</SelectItem>)}</SelectContent></Select> : <div className="text-sm">{pipeline?.trigger_type}</div>}</div>
|
|
151
|
+
{formData.trigger_type === 'scheduled' && <div className="space-y-2"><Label>Schedule (cron)</Label>{isEditing ? <Input value={formData.schedule} onChange={e => setFormData({...formData, schedule: e.target.value})} placeholder="0 0 * * *" /> : <div className="text-sm font-mono">{pipeline?.schedule || '—'}</div>}</div>}
|
|
152
|
+
<div className="space-y-2"><Label>Active</Label>{isEditing ? <div className="flex items-center gap-2"><Checkbox id="active" checked={formData.is_active} onCheckedChange={c => setFormData({...formData, is_active: c === true})} /><Label htmlFor="active">Enabled</Label></div> : <div className="text-sm">{pipeline?.is_active ? 'Yes' : 'No'}</div>}</div>
|
|
153
|
+
</div>
|
|
154
|
+
<div className="space-y-4">
|
|
155
|
+
<h3 className="text-sm font-medium text-muted-foreground">Execution Info</h3>
|
|
156
|
+
{!isCreateMode && (
|
|
157
|
+
<>
|
|
158
|
+
<div className="flex justify-between"><span className="text-sm text-muted-foreground">Last Run</span><span className="text-sm">{pipeline?.last_run ? formatDateTime(pipeline.last_run) : '—'}</span></div>
|
|
159
|
+
<div className="flex justify-between"><span className="text-sm text-muted-foreground">Last Status</span><Badge variant={pipeline?.last_run_status === 'success' ? 'default' : pipeline?.last_run_status === 'failed' ? 'destructive' : 'outline'}>{pipeline?.last_run_status || '—'}</Badge></div>
|
|
160
|
+
<div className="flex justify-between"><span className="text-sm text-muted-foreground">Total Runs</span><span className="text-sm">{pipeline?.run_count || 0}</span></div>
|
|
161
|
+
<div className="flex justify-between"><span className="text-sm text-muted-foreground">Steps</span><span className="text-sm">{pipeline?.steps?.length || 0}</span></div>
|
|
162
|
+
</>
|
|
163
|
+
)}
|
|
164
|
+
</div>
|
|
165
|
+
</div>
|
|
166
|
+
<div className="mt-6 space-y-2"><Label>Description</Label>{isEditing ? <Textarea value={formData.description} onChange={e => setFormData({...formData, description: e.target.value})} rows={3} /> : <div className="text-sm">{pipeline?.description || '—'}</div>}</div>
|
|
167
|
+
</CardContent></Card>
|
|
168
|
+
|
|
169
|
+
{!isCreateMode && pipeline?.steps && (
|
|
170
|
+
<Card><CardHeader><CardTitle>Steps ({pipeline.steps.length})</CardTitle></CardHeader><CardContent>
|
|
171
|
+
<div className="space-y-2">
|
|
172
|
+
{pipeline.steps.map((step, idx) => (
|
|
173
|
+
<div key={step.id} className="flex items-center gap-4 p-3 bg-muted rounded">
|
|
174
|
+
<div className="flex h-8 w-8 items-center justify-center rounded-full bg-primary text-primary-foreground text-sm font-medium">{idx + 1}</div>
|
|
175
|
+
<div className="flex-1"><div className="font-medium">{step.name}</div><div className="text-sm text-muted-foreground">{step.type}</div></div>
|
|
176
|
+
</div>
|
|
177
|
+
))}
|
|
178
|
+
</div>
|
|
179
|
+
</CardContent></Card>
|
|
180
|
+
)}
|
|
181
|
+
</div>
|
|
182
|
+
)
|
|
183
|
+
}
|