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,490 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module middleware
|
|
3
|
+
* @audience both
|
|
4
|
+
* @layer shared-core
|
|
5
|
+
* @stability stable
|
|
6
|
+
*
|
|
7
|
+
* HTTP handler factory and request context types for all Spine Netlify functions.
|
|
8
|
+
* This module owns the boundary between raw HTTP events and the typed execution
|
|
9
|
+
* context (`RequestContext`) used by every handler. It also provides guard
|
|
10
|
+
* utilities (`requireUserContext`, `requireSystemContextWithAudit`) used to
|
|
11
|
+
* enforce authentication at the top of handlers.
|
|
12
|
+
*
|
|
13
|
+
* The key invariant: `createHandler` always resolves a `Principal` via
|
|
14
|
+
* `resolvePrincipal` before calling the inner handler. Handlers never receive
|
|
15
|
+
* an unauthenticated context — anonymous requests are rejected at the wrapper.
|
|
16
|
+
*
|
|
17
|
+
* IMPORTANT: `result.data` is never unwrapped in `createHandler`. Handlers
|
|
18
|
+
* return records directly. Unwrapping would collide with records that have a
|
|
19
|
+
* `.data` JSONB column.
|
|
20
|
+
*
|
|
21
|
+
* @seeAlso principal.ts (resolvePrincipal, getPrincipalDb, isSystemAdmin)
|
|
22
|
+
* @seeAlso audit.ts (emitAudit — called after every successful handler)
|
|
23
|
+
* @seeAlso db.ts (adminDb, getUserDb — selected by getPrincipalDb)
|
|
24
|
+
* @seeAlso index.ts (re-exports CoreContext, createHandler, requireUserContext)
|
|
25
|
+
*/
|
|
26
|
+
|
|
27
|
+
import {
|
|
28
|
+
Principal,
|
|
29
|
+
resolvePrincipal,
|
|
30
|
+
isSystemAdmin,
|
|
31
|
+
getPrincipalDb
|
|
32
|
+
} from './principal'
|
|
33
|
+
import { emitAudit } from './audit'
|
|
34
|
+
|
|
35
|
+
// ─── CONTEXT TYPES ───────────────────────────────────────────────────────────
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Minimal execution context passed to all Spine core functions.
|
|
39
|
+
*
|
|
40
|
+
* This is the canonical context for `pipeline-runner`, `trigger-engine`,
|
|
41
|
+
* `agent-runner`, `audit`, and any custom code in `v2-custom/`. It contains
|
|
42
|
+
* only what core logic needs: identity, account scope, and a DB client.
|
|
43
|
+
*
|
|
44
|
+
* `RequestContext` (used inside HTTP handlers) extends this with HTTP-specific
|
|
45
|
+
* fields (`appId`, `query`). Direct importers and CLI callers construct
|
|
46
|
+
* `CoreContext` directly without going through `createHandler`.
|
|
47
|
+
*
|
|
48
|
+
* @inputSpec principal: Principal — must be a resolved principal (not null)
|
|
49
|
+
* @inputSpec accountId: string | null — null is allowed for system-level ops only
|
|
50
|
+
* @inputSpec db: SupabaseClient — use adminDb for system ops, getUserDb for RLS
|
|
51
|
+
* @inputSpec requestId: string — UUID; ties execution to audit log entries
|
|
52
|
+
* @calledBy pipeline-runner.ts, trigger-engine.ts, agent-runner.ts, audit.ts,
|
|
53
|
+
* tests/integration/helpers.ts (makeTestCtx), cli/context.ts
|
|
54
|
+
*
|
|
55
|
+
* @example Import usage (v2-custom/)
|
|
56
|
+
* ```ts
|
|
57
|
+
* import { CoreContext, adminDb, SYSTEM_PRINCIPAL } from '../_shared/index'
|
|
58
|
+
* const ctx: CoreContext = {
|
|
59
|
+
* principal: SYSTEM_PRINCIPAL,
|
|
60
|
+
* accountId: 'uuid-of-account',
|
|
61
|
+
* db: adminDb,
|
|
62
|
+
* requestId: crypto.randomUUID()
|
|
63
|
+
* }
|
|
64
|
+
* await runPipeline(pipelineId, data, ctx)
|
|
65
|
+
* ```
|
|
66
|
+
*
|
|
67
|
+
* @example CLI usage
|
|
68
|
+
* ```bash
|
|
69
|
+
* # CLI constructs CoreContext from .xenv credentials automatically
|
|
70
|
+
* spine pipelines run <pipeline-id>
|
|
71
|
+
* ```
|
|
72
|
+
*/
|
|
73
|
+
export interface CoreContext {
|
|
74
|
+
/** Resolved principal for this execution */
|
|
75
|
+
principal: Principal
|
|
76
|
+
/** Primary account scope — null for system-level operations */
|
|
77
|
+
accountId: string | null
|
|
78
|
+
/** Database client — use adminDb for system ops, getUserDb for RLS-scoped */
|
|
79
|
+
db: any
|
|
80
|
+
/** Unique ID for this execution (used in audit logs) */
|
|
81
|
+
requestId: string
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* HTTP-layer execution context — extends `CoreContext` with request-specific fields.
|
|
86
|
+
*
|
|
87
|
+
* Constructed inside `createHandler` after principal resolution. Not used by
|
|
88
|
+
* core logic directly — core functions accept `CoreContext`. The extra fields
|
|
89
|
+
* are available to handlers that need them (e.g., `query.action`, `appId`).
|
|
90
|
+
*
|
|
91
|
+
* @inputSpec appId: string | null — from `x-app-id` header; null if absent
|
|
92
|
+
* @inputSpec query: Record<string, string> — parsed queryStringParameters from event
|
|
93
|
+
* @calledBy All 19 API handler functions (as the first argument)
|
|
94
|
+
* @seeAlso CoreContext (parent interface)
|
|
95
|
+
*/
|
|
96
|
+
export interface RequestContext extends CoreContext {
|
|
97
|
+
/** App ID from `x-app-id` header — used for app-scoped operations */
|
|
98
|
+
appId: string | null
|
|
99
|
+
/** Parsed query string parameters from the Netlify event */
|
|
100
|
+
query: Record<string, string>
|
|
101
|
+
/** Request path from the Netlify event */
|
|
102
|
+
requestPath: string
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Signature for all Spine HTTP handler functions.
|
|
107
|
+
*
|
|
108
|
+
* Every handler file exports a default that calls `createHandler(myHandler)`.
|
|
109
|
+
* The `body` parameter is the parsed JSON body, or null for GET requests.
|
|
110
|
+
*
|
|
111
|
+
* @inputSpec ctx: RequestContext — fully resolved context; never null
|
|
112
|
+
* @inputSpec body: any — parsed JSON body or null; undefined for GET requests
|
|
113
|
+
* @outputSpec Promise<any> — return value is wrapped in `{ data: result }` by createHandler
|
|
114
|
+
*/
|
|
115
|
+
export type HandlerFunction = (ctx: RequestContext, body?: any) => Promise<any>
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Standard envelope shape returned by `createHandler` to the HTTP client.
|
|
119
|
+
*
|
|
120
|
+
* On success: `{ data: <handler result>, error: null, meta: { requestId, duration } }`
|
|
121
|
+
* On error: `{ data: null, error: <message> }` with appropriate HTTP status code.
|
|
122
|
+
*
|
|
123
|
+
* @outputSpec data: any — handler return value, never unwrapped
|
|
124
|
+
* @outputSpec error: string | undefined — error message; present only on failure
|
|
125
|
+
* @outputSpec meta: object | undefined — requestId + duration on success
|
|
126
|
+
*/
|
|
127
|
+
export interface HandlerResult {
|
|
128
|
+
data?: any
|
|
129
|
+
error?: string
|
|
130
|
+
meta?: any
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
// ─── HANDLER FACTORY ─────────────────────────────────────────────────────────
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* Wraps a handler function with principal resolution, request parsing, audit
|
|
137
|
+
* logging, and error handling. This is the entry point for every Netlify function.
|
|
138
|
+
*
|
|
139
|
+
* Execution flow:
|
|
140
|
+
* 1. Detect nested calls (event already has requestId + principal) → pass through
|
|
141
|
+
* 2. Generate `requestId`, parse headers, query params
|
|
142
|
+
* 3. Call `resolvePrincipal(event)` → reject anonymous with 401
|
|
143
|
+
* 4. Call `getPrincipalDb(principal)` → select correct DB client
|
|
144
|
+
* 5. Build `RequestContext`, parse + merge body
|
|
145
|
+
* 6. Call inner handler, measure duration
|
|
146
|
+
* 7. Emit `request.<method>` audit log (account-scoped requests only)
|
|
147
|
+
* 8. Return `json({ data: result, error: null, meta })` envelope
|
|
148
|
+
* 9. On any thrown error → return `error(message, 500)`
|
|
149
|
+
*
|
|
150
|
+
* @param handler - The inner handler function implementing the route logic
|
|
151
|
+
* @returns Netlify-compatible async function `(event, context) => Response`
|
|
152
|
+
* @throws never — all errors are caught and returned as HTTP 500
|
|
153
|
+
* @inputSpec handler: HandlerFunction — must return a Promise
|
|
154
|
+
* @outputSpec Netlify Lambda response object with statusCode, headers, body
|
|
155
|
+
* @sideEffects DB read: principal resolution (people, api_keys tables)
|
|
156
|
+
* @sideEffects DB write: emitAudit to logs table (account-scoped requests only)
|
|
157
|
+
* @calledBy Every function in functions/*.ts as the default export wrapper
|
|
158
|
+
* @calls resolvePrincipal, getPrincipalDb, emitAudit, json, error
|
|
159
|
+
* @testIntegration tests/integration/admin-data-accounts.test.ts
|
|
160
|
+
*
|
|
161
|
+
* @example API handler file pattern
|
|
162
|
+
* ```ts
|
|
163
|
+
* import { createHandler, RequestContext } from './_shared/middleware'
|
|
164
|
+
* export const handler = createHandler(async (ctx: RequestContext, body) => {
|
|
165
|
+
* const action = ctx.query.action || 'list'
|
|
166
|
+
* if (action === 'list') return listItems(ctx)
|
|
167
|
+
* throw new Error(`Unknown action: ${action}`)
|
|
168
|
+
* })
|
|
169
|
+
* ```
|
|
170
|
+
*/
|
|
171
|
+
export function createHandler(handler: HandlerFunction) {
|
|
172
|
+
return async (event: any, context: any) => {
|
|
173
|
+
// Detect nested call: if event is already a RequestContext with a principal,
|
|
174
|
+
// skip event parsing and call the raw handler directly
|
|
175
|
+
if (event && event.requestId && event.principal) {
|
|
176
|
+
return handler(event, context)
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
const requestId = crypto.randomUUID()
|
|
180
|
+
const startTime = Date.now()
|
|
181
|
+
|
|
182
|
+
try {
|
|
183
|
+
// Parse headers
|
|
184
|
+
const appId = event.headers?.['x-app-id'] || event.headers?.['X-App-Id']
|
|
185
|
+
|
|
186
|
+
// Parse query string parameters
|
|
187
|
+
const queryParams: Record<string, string> = {}
|
|
188
|
+
if (event.queryStringParameters) {
|
|
189
|
+
Object.assign(queryParams, event.queryStringParameters)
|
|
190
|
+
}
|
|
191
|
+
if (!queryParams.method && event.httpMethod) {
|
|
192
|
+
queryParams.method = event.httpMethod
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
const principal = await resolvePrincipal(event)
|
|
196
|
+
|
|
197
|
+
if (!principal || principal.id === 'anonymous') {
|
|
198
|
+
return error('Authentication required', 401)
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// Get RLS-scoped database client based on principal type
|
|
202
|
+
const ctxDb = getPrincipalDb(principal)
|
|
203
|
+
|
|
204
|
+
// Build request context
|
|
205
|
+
const ctx: RequestContext = {
|
|
206
|
+
requestId,
|
|
207
|
+
principal,
|
|
208
|
+
db: ctxDb,
|
|
209
|
+
accountId: principal.accountId,
|
|
210
|
+
appId: appId || null,
|
|
211
|
+
query: queryParams,
|
|
212
|
+
requestPath: event.path || '/',
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
// Parse body
|
|
216
|
+
let body = null
|
|
217
|
+
if (event.body) {
|
|
218
|
+
try {
|
|
219
|
+
body = JSON.parse(event.body)
|
|
220
|
+
} catch (e) {
|
|
221
|
+
return error('Invalid JSON in request body', 400)
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
if (ctx.query.id) {
|
|
225
|
+
if (!body || typeof body !== 'object' || Array.isArray(body)) {
|
|
226
|
+
body = {}
|
|
227
|
+
}
|
|
228
|
+
if (!('id' in body)) {
|
|
229
|
+
body.id = ctx.query.id
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
// Capture request details for audit
|
|
234
|
+
const httpMethod = event.httpMethod || queryParams.method || 'GET'
|
|
235
|
+
const requestPath = event.path || '/'
|
|
236
|
+
|
|
237
|
+
// Execute handler
|
|
238
|
+
const result = await handler(ctx, body)
|
|
239
|
+
const durationMs = Date.now() - startTime
|
|
240
|
+
|
|
241
|
+
// Emit audit log for successful request
|
|
242
|
+
if (ctx.accountId) {
|
|
243
|
+
await emitAudit(ctx, `request.${httpMethod.toLowerCase()}`, {
|
|
244
|
+
type: 'request',
|
|
245
|
+
id: requestId,
|
|
246
|
+
account_id: ctx.accountId
|
|
247
|
+
}, {
|
|
248
|
+
path: requestPath,
|
|
249
|
+
method: httpMethod,
|
|
250
|
+
duration_ms: durationMs,
|
|
251
|
+
principal_type: principal.type,
|
|
252
|
+
principal_id: principal.id,
|
|
253
|
+
app_id: ctx.appId
|
|
254
|
+
})
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
// Return success response.
|
|
258
|
+
// Never unwrap result.data here — handlers return the record directly.
|
|
259
|
+
// Using result.data would collide with records that have a .data JSONB column.
|
|
260
|
+
return json({
|
|
261
|
+
data: result,
|
|
262
|
+
error: null,
|
|
263
|
+
meta: {
|
|
264
|
+
requestId,
|
|
265
|
+
duration: durationMs
|
|
266
|
+
}
|
|
267
|
+
})
|
|
268
|
+
|
|
269
|
+
} catch (err: any) {
|
|
270
|
+
console.error(`[${requestId}] Handler error:`, err)
|
|
271
|
+
const status = typeof err.statusCode === 'number' ? err.statusCode : 500
|
|
272
|
+
return error(err.message || 'Internal server error', status)
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// ─── RESPONSE HELPERS ────────────────────────────────────────────────────────
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* Builds a JSON HTTP response object compatible with Netlify Functions.
|
|
281
|
+
*
|
|
282
|
+
* Always includes CORS headers permitting requests from any origin. Used
|
|
283
|
+
* internally by `createHandler` and directly by handlers that need a custom
|
|
284
|
+
* status code (e.g., 201 Created).
|
|
285
|
+
*
|
|
286
|
+
* @param data - Any JSON-serializable value to include as the response body
|
|
287
|
+
* @param status - HTTP status code (default: 200)
|
|
288
|
+
* @returns Netlify Lambda response object
|
|
289
|
+
* @throws never
|
|
290
|
+
* @inputSpec data: any — must be JSON-serializable; circular refs will throw at stringify
|
|
291
|
+
* @inputSpec status: number — valid HTTP status code (default 200)
|
|
292
|
+
* @outputSpec { statusCode, headers, body: string } — body is JSON.stringify(data)
|
|
293
|
+
* @sideEffects none
|
|
294
|
+
* @calledBy createHandler, error, cors, and directly by some handlers
|
|
295
|
+
* @testUnit none — trivial; verified by integration tests on every request
|
|
296
|
+
*/
|
|
297
|
+
export function json(data: any, status: number = 200) {
|
|
298
|
+
return {
|
|
299
|
+
statusCode: status,
|
|
300
|
+
headers: {
|
|
301
|
+
'Content-Type': 'application/json',
|
|
302
|
+
'Access-Control-Allow-Origin': '*',
|
|
303
|
+
'Access-Control-Allow-Headers': 'Content-Type, Authorization, X-Account-Id, X-App-Id',
|
|
304
|
+
'Access-Control-Allow-Methods': 'GET, POST, PATCH, DELETE, OPTIONS'
|
|
305
|
+
},
|
|
306
|
+
body: JSON.stringify(data)
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
/**
|
|
311
|
+
* Builds a JSON error response with `{ data: null, error: message }` shape.
|
|
312
|
+
*
|
|
313
|
+
* Use this to return structured error responses from handlers. The message
|
|
314
|
+
* is safe to surface to clients — do not pass internal error details.
|
|
315
|
+
*
|
|
316
|
+
* @param message - Human-readable error message safe to return to client
|
|
317
|
+
* @param status - HTTP status code (default: 400)
|
|
318
|
+
* @returns Netlify Lambda response object
|
|
319
|
+
* @throws never
|
|
320
|
+
* @inputSpec message: string — client-safe error description
|
|
321
|
+
* @inputSpec status: number — HTTP status code (400, 401, 403, 404, 500)
|
|
322
|
+
* @outputSpec { statusCode: status, body: '{"data":null,"error":"<message>"}' }
|
|
323
|
+
* @sideEffects none
|
|
324
|
+
* @calledBy createHandler (on caught errors), requireUserContext,
|
|
325
|
+
* requireSystemContextWithAudit, and many individual handlers
|
|
326
|
+
*/
|
|
327
|
+
export function error(message: string, status: number = 400) {
|
|
328
|
+
return json({
|
|
329
|
+
data: null,
|
|
330
|
+
error: message
|
|
331
|
+
}, status)
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
/**
|
|
335
|
+
* Parses the JSON body from a Netlify event object.
|
|
336
|
+
*
|
|
337
|
+
* Returns `null` if there is no body. Throws a descriptive error on malformed
|
|
338
|
+
* JSON so the error surfaces cleanly from `createHandler`'s catch block.
|
|
339
|
+
*
|
|
340
|
+
* @param event - Raw Netlify event object
|
|
341
|
+
* @returns Parsed body object or null
|
|
342
|
+
* @throws Error('Invalid JSON in request body') — when body is present but not valid JSON
|
|
343
|
+
* @inputSpec event.body: string | null | undefined — raw JSON string from HTTP request
|
|
344
|
+
* @outputSpec any — parsed JSON object, or null if no body
|
|
345
|
+
* @sideEffects none
|
|
346
|
+
* @calledBy Handlers that need body outside of createHandler's automatic parsing
|
|
347
|
+
*/
|
|
348
|
+
export function parseBody(event: any): any {
|
|
349
|
+
if (!event.body) return null
|
|
350
|
+
|
|
351
|
+
try {
|
|
352
|
+
return JSON.parse(event.body)
|
|
353
|
+
} catch (e) {
|
|
354
|
+
throw new Error('Invalid JSON in request body')
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
// ─── AUTH GUARDS ─────────────────────────────────────────────────────────────
|
|
359
|
+
|
|
360
|
+
/**
|
|
361
|
+
* Overloaded guard that requires a resolved human principal with an account scope.
|
|
362
|
+
*
|
|
363
|
+
* **Overload 1 — wrapper:** wrap a handler to enforce auth before it runs.
|
|
364
|
+
* **Overload 2 — inline:** call with `ctx` directly; returns an error response
|
|
365
|
+
* object if auth is missing, or `null` if auth is present (allowing the
|
|
366
|
+
* caller to do `const authErr = requireUserContext(ctx); if (authErr) return authErr`).
|
|
367
|
+
*
|
|
368
|
+
* Rejects requests where:
|
|
369
|
+
* - `ctx.principal` is absent
|
|
370
|
+
* - `ctx.principal.id === 'anonymous'`
|
|
371
|
+
* - `ctx.accountId` is null or empty (machine principals without an account)
|
|
372
|
+
*
|
|
373
|
+
* @inputSpec ctx or handler: RequestContext or HandlerFunction
|
|
374
|
+
* @outputSpec HandlerFunction (overload 1) or HandlerResult | null (overload 2)
|
|
375
|
+
* @throws Error('User context required') — in wrapper mode if not authenticated
|
|
376
|
+
* @sideEffects none
|
|
377
|
+
* @calledBy API handlers that require an authenticated human with account scope
|
|
378
|
+
* @testIntegration tests/integration/isolation.test.ts
|
|
379
|
+
*
|
|
380
|
+
* @example Inline guard pattern (preferred)
|
|
381
|
+
* ```ts
|
|
382
|
+
* const authErr = requireUserContext(ctx)
|
|
383
|
+
* if (authErr) return authErr
|
|
384
|
+
* // ctx.principal and ctx.accountId are guaranteed non-null below here
|
|
385
|
+
* ```
|
|
386
|
+
*
|
|
387
|
+
* @example Wrapper pattern
|
|
388
|
+
* ```ts
|
|
389
|
+
* export const handler = createHandler(requireUserContext(async (ctx, body) => {
|
|
390
|
+
* return listItems(ctx)
|
|
391
|
+
* }))
|
|
392
|
+
* ```
|
|
393
|
+
*/
|
|
394
|
+
export function requireUserContext(handler: HandlerFunction): HandlerFunction
|
|
395
|
+
export function requireUserContext(ctx: RequestContext): HandlerResult | null
|
|
396
|
+
export function requireUserContext(arg: HandlerFunction | RequestContext): HandlerFunction | HandlerResult | null {
|
|
397
|
+
if (typeof arg === 'function') {
|
|
398
|
+
const handler = arg
|
|
399
|
+
return async (ctx: RequestContext, body?: any) => {
|
|
400
|
+
if (!ctx.principal || ctx.principal.id === 'anonymous' || !ctx.accountId) {
|
|
401
|
+
throw new Error('User context (person and account) required')
|
|
402
|
+
}
|
|
403
|
+
return handler(ctx, body)
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
const ctx = arg
|
|
408
|
+
if (!ctx.principal || ctx.principal.id === 'anonymous' || !ctx.accountId) {
|
|
409
|
+
return error('User context (person and account) required', 403) as any
|
|
410
|
+
}
|
|
411
|
+
return null
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
/**
|
|
415
|
+
* Overloaded guard that requires a `system_admin` principal.
|
|
416
|
+
*
|
|
417
|
+
* **Overload 1 — wrapper:** wrap a handler; throws if not system_admin.
|
|
418
|
+
* **Overload 2 — inline:** call with `ctx`; returns error response or null.
|
|
419
|
+
* Also accepts an optional `triggeredBy` string to set on the context for
|
|
420
|
+
* audit trail chaining (e.g. pipeline execution ID).
|
|
421
|
+
*
|
|
422
|
+
* Rejects requests where:
|
|
423
|
+
* - `ctx.principal` is absent or anonymous
|
|
424
|
+
* - `ctx.principal` does not have the `system_admin` role
|
|
425
|
+
*
|
|
426
|
+
* @param arg - Handler to wrap, or RequestContext to validate inline
|
|
427
|
+
* @param triggeredBy - Optional: ID of the triggering entity (set on ctx)
|
|
428
|
+
* @returns HandlerFunction (overload 1) or HandlerResult | null (overload 2)
|
|
429
|
+
* @throws Error('System context required') — in wrapper mode if not system_admin
|
|
430
|
+
* @inputSpec ctx.principal.roles: string[] — must include 'system_admin'
|
|
431
|
+
* @outputSpec HandlerFunction | HandlerResult | null
|
|
432
|
+
* @sideEffects sets `ctx.triggeredBy` when validation passes (inline mode)
|
|
433
|
+
* @calledBy system-cron.ts, pipeline-executions.ts, and admin-only handlers
|
|
434
|
+
* @testIntegration tests/integration/isolation.test.ts
|
|
435
|
+
*
|
|
436
|
+
* @example Inline guard pattern
|
|
437
|
+
* ```ts
|
|
438
|
+
* const authErr = requireSystemContextWithAudit(ctx, 'cron-job-id')
|
|
439
|
+
* if (authErr) return authErr
|
|
440
|
+
* ```
|
|
441
|
+
*/
|
|
442
|
+
export function requireSystemContextWithAudit(handler: HandlerFunction): HandlerFunction
|
|
443
|
+
export function requireSystemContextWithAudit(ctx: RequestContext, triggeredBy?: string): HandlerResult | null
|
|
444
|
+
export function requireSystemContextWithAudit(
|
|
445
|
+
arg: HandlerFunction | RequestContext,
|
|
446
|
+
triggeredBy?: string,
|
|
447
|
+
): HandlerFunction | HandlerResult | null {
|
|
448
|
+
if (typeof arg === 'function') {
|
|
449
|
+
const handler = arg
|
|
450
|
+
return async (ctx: RequestContext, body?: any) => {
|
|
451
|
+
if (!ctx.principal || ctx.principal.id === 'anonymous') {
|
|
452
|
+
throw new Error('Authentication required')
|
|
453
|
+
}
|
|
454
|
+
if (!isSystemAdmin(ctx.principal)) {
|
|
455
|
+
throw new Error('System context required')
|
|
456
|
+
}
|
|
457
|
+
;(ctx as any).triggeredBy = ctx.principal.id
|
|
458
|
+
return handler(ctx, body)
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
const ctx = arg
|
|
463
|
+
if (!ctx.principal || ctx.principal.id === 'anonymous') {
|
|
464
|
+
return error('Authentication required', 401) as any
|
|
465
|
+
}
|
|
466
|
+
if (!isSystemAdmin(ctx.principal)) {
|
|
467
|
+
return error('System context required', 403) as any
|
|
468
|
+
}
|
|
469
|
+
;(ctx as any).triggeredBy = triggeredBy || ctx.principal.id
|
|
470
|
+
return null
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
// ─── UTILITY ─────────────────────────────────────────────────────────────────
|
|
474
|
+
|
|
475
|
+
/**
|
|
476
|
+
* Returns a 200 JSON response for CORS preflight requests.
|
|
477
|
+
*
|
|
478
|
+
* Netlify automatically handles OPTIONS at the CDN level for most routes, but
|
|
479
|
+
* handlers that need to explicitly handle OPTIONS can call this.
|
|
480
|
+
*
|
|
481
|
+
* @returns json({ message: 'CORS enabled' }, 200) with CORS headers
|
|
482
|
+
* @throws never
|
|
483
|
+
* @inputSpec none
|
|
484
|
+
* @outputSpec Netlify Lambda response with CORS headers
|
|
485
|
+
* @sideEffects none
|
|
486
|
+
* @calledBy Handler files that explicitly handle OPTIONS method
|
|
487
|
+
*/
|
|
488
|
+
export function cors() {
|
|
489
|
+
return json({ message: 'CORS enabled' }, 200)
|
|
490
|
+
}
|