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,523 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module integration-routes
|
|
3
|
+
* @audience core-contributor
|
|
4
|
+
* @layer api-handler
|
|
5
|
+
* @stability stable
|
|
6
|
+
*
|
|
7
|
+
* Unified integration routing system for all integration types.
|
|
8
|
+
* Handles webhook, API, database, file, and custom integrations
|
|
9
|
+
* through a single entry point with type-specific handlers.
|
|
10
|
+
*
|
|
11
|
+
** Routed by:** `POST /api/integration-routes?slug=:slug`
|
|
12
|
+
*
|
|
13
|
+
* Route patterns:
|
|
14
|
+
* | method | path | handler |
|
|
15
|
+
* |--------|-------------------------------------------|------------------------|
|
|
16
|
+
* | POST | /api/integration-routes?slug=:slug | handleWebhook |
|
|
17
|
+
*
|
|
18
|
+
* **Authorization:** API key authentication with integration permissions
|
|
19
|
+
* - API key validated via resolvePrincipal
|
|
20
|
+
* - Integration permissions checked against principal.permissions.integrations[slug]
|
|
21
|
+
*
|
|
22
|
+
* **Security Flow:**
|
|
23
|
+
* 1. API key validation (resolvePrincipal)
|
|
24
|
+
* 2. Integration permission check
|
|
25
|
+
* 3. Schema validation (if configured)
|
|
26
|
+
* 4. Data sanitization
|
|
27
|
+
* 5. Custom script execution
|
|
28
|
+
*
|
|
29
|
+
* INVARIANT: All validation failures return generic { status: "rejected" }
|
|
30
|
+
* INVARIANT: Detailed errors logged server-side only
|
|
31
|
+
* INVARIANT: Unknown fields in payload = HARD FAIL (fishing detection)
|
|
32
|
+
*
|
|
33
|
+
* @seeAlso integrations.ts (integration configuration)
|
|
34
|
+
* @seeAlso principal.ts (machine principal resolution)
|
|
35
|
+
* @seeAlso api-keys.ts (API key permissions)
|
|
36
|
+
*/
|
|
37
|
+
|
|
38
|
+
import { adminDb } from './_shared/db'
|
|
39
|
+
import { resolvePrincipal } from './_shared/principal'
|
|
40
|
+
import { resolveHandler, listRegisteredHandlers } from './_shared/webhook-registry'
|
|
41
|
+
|
|
42
|
+
// ─── TYPES ────────────────────────────────────────────────────────────────
|
|
43
|
+
|
|
44
|
+
interface IntegrationContext {
|
|
45
|
+
integration: any
|
|
46
|
+
principal: any
|
|
47
|
+
requestPath: string
|
|
48
|
+
method: string
|
|
49
|
+
headers: Record<string, string>
|
|
50
|
+
body: any
|
|
51
|
+
query: Record<string, string>
|
|
52
|
+
requestId: string
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
interface ValidationResult {
|
|
56
|
+
valid: boolean
|
|
57
|
+
errors: string[]
|
|
58
|
+
unknownFields: string[]
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
interface IntegrationHandler {
|
|
62
|
+
(ctx: IntegrationContext): Promise<any>
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// ─── INTEGRATION RESOLUTION ────────────────────────────────────────────────
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Resolves integration by slug and validates it's active and configured
|
|
69
|
+
*/
|
|
70
|
+
async function resolveIntegration(slug: string, type: string): Promise<any> {
|
|
71
|
+
const { data, error } = await adminDb
|
|
72
|
+
.from('integrations')
|
|
73
|
+
.select('*')
|
|
74
|
+
.eq('name', slug)
|
|
75
|
+
.eq('integration_type', type)
|
|
76
|
+
.eq('is_active', true)
|
|
77
|
+
.single()
|
|
78
|
+
|
|
79
|
+
if (error || !data) {
|
|
80
|
+
throw new Error(`Integration not found: ${type}/${slug}`)
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
if (!data.is_configured) {
|
|
84
|
+
throw new Error(`Integration not configured: ${type}/${slug}`)
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return data
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// ─── VALIDATION & SANITIZATION ───────────────────────────────────────────
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Validates data against schema definition
|
|
94
|
+
* Returns validation result with any errors and unknown fields
|
|
95
|
+
*/
|
|
96
|
+
function validateSchema(data: any, schema: any, requestId: string): ValidationResult {
|
|
97
|
+
const errors: string[] = []
|
|
98
|
+
const unknownFields: string[] = []
|
|
99
|
+
|
|
100
|
+
if (!schema || typeof schema !== 'object') {
|
|
101
|
+
return { valid: true, errors: [], unknownFields: [] }
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Check for unknown fields
|
|
105
|
+
const allowedFields = Object.keys(schema)
|
|
106
|
+
for (const field of Object.keys(data)) {
|
|
107
|
+
if (!allowedFields.includes(field)) {
|
|
108
|
+
unknownFields.push(field)
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// Validate each field against schema
|
|
113
|
+
for (const [fieldName, fieldConfig] of Object.entries(schema)) {
|
|
114
|
+
const config = fieldConfig as any
|
|
115
|
+
const value = data[fieldName]
|
|
116
|
+
|
|
117
|
+
// Check required fields
|
|
118
|
+
if (config.required && (value === undefined || value === null || value === '')) {
|
|
119
|
+
errors.push(`Field '${fieldName}' is required but missing`)
|
|
120
|
+
continue
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Skip further validation if field is not required and not present
|
|
124
|
+
if (!config.required && (value === undefined || value === null)) {
|
|
125
|
+
continue
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Validate data type
|
|
129
|
+
if (config.type) {
|
|
130
|
+
const typeValid = validateType(value, config.type, fieldName)
|
|
131
|
+
if (!typeValid.valid) {
|
|
132
|
+
errors.push(typeValid.error)
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Validate constraints
|
|
137
|
+
if (config.min !== undefined && typeof value === 'number' && value < config.min) {
|
|
138
|
+
errors.push(`Field '${fieldName}' must be at least ${config.min}`)
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
if (config.max !== undefined && typeof value === 'number' && value > config.max) {
|
|
142
|
+
errors.push(`Field '${fieldName}' must be at most ${config.max}`)
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
if (config.maxLength !== undefined && typeof value === 'string' && value.length > config.maxLength) {
|
|
146
|
+
errors.push(`Field '${fieldName}' must be at most ${config.maxLength} characters`)
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
if (config.pattern && typeof value === 'string') {
|
|
150
|
+
const regex = new RegExp(config.pattern)
|
|
151
|
+
if (!regex.test(value)) {
|
|
152
|
+
errors.push(`Field '${fieldName}' does not match required pattern`)
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
return {
|
|
158
|
+
valid: errors.length === 0 && unknownFields.length === 0,
|
|
159
|
+
errors,
|
|
160
|
+
unknownFields
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* Validates a value against a type definition
|
|
166
|
+
*/
|
|
167
|
+
function validateType(value: any, type: string, fieldName: string): { valid: boolean; error?: string } {
|
|
168
|
+
switch (type) {
|
|
169
|
+
case 'string':
|
|
170
|
+
if (typeof value !== 'string') {
|
|
171
|
+
return { valid: false, error: `Field '${fieldName}' must be a string` }
|
|
172
|
+
}
|
|
173
|
+
break
|
|
174
|
+
case 'number':
|
|
175
|
+
if (typeof value !== 'number' || isNaN(value)) {
|
|
176
|
+
return { valid: false, error: `Field '${fieldName}' must be a number` }
|
|
177
|
+
}
|
|
178
|
+
break
|
|
179
|
+
case 'boolean':
|
|
180
|
+
if (typeof value !== 'boolean') {
|
|
181
|
+
return { valid: false, error: `Field '${fieldName}' must be a boolean` }
|
|
182
|
+
}
|
|
183
|
+
break
|
|
184
|
+
case 'email':
|
|
185
|
+
if (typeof value !== 'string' || !value.includes('@')) {
|
|
186
|
+
return { valid: false, error: `Field '${fieldName}' must be a valid email` }
|
|
187
|
+
}
|
|
188
|
+
break
|
|
189
|
+
case 'array':
|
|
190
|
+
if (!Array.isArray(value)) {
|
|
191
|
+
return { valid: false, error: `Field '${fieldName}' must be an array` }
|
|
192
|
+
}
|
|
193
|
+
break
|
|
194
|
+
case 'object':
|
|
195
|
+
if (typeof value !== 'object' || value === null || Array.isArray(value)) {
|
|
196
|
+
return { valid: false, error: `Field '${fieldName}' must be an object` }
|
|
197
|
+
}
|
|
198
|
+
break
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
return { valid: true }
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
/**
|
|
205
|
+
* Sanitizes data to prevent injection attacks
|
|
206
|
+
*/
|
|
207
|
+
function sanitizeData(data: any, rules: string = 'strict'): any {
|
|
208
|
+
if (data === null || data === undefined) {
|
|
209
|
+
return data
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
if (typeof data === 'string') {
|
|
213
|
+
// Remove HTML tags
|
|
214
|
+
let sanitized = data.replace(/<[^>]*>/g, '')
|
|
215
|
+
// Escape special characters
|
|
216
|
+
sanitized = sanitized
|
|
217
|
+
.replace(/&/g, '&')
|
|
218
|
+
.replace(/</g, '<')
|
|
219
|
+
.replace(/>/g, '>')
|
|
220
|
+
.replace(/"/g, '"')
|
|
221
|
+
.replace(/'/g, ''')
|
|
222
|
+
// Normalize whitespace
|
|
223
|
+
sanitized = sanitized.trim().replace(/\s+/g, ' ')
|
|
224
|
+
return sanitized
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
if (Array.isArray(data)) {
|
|
228
|
+
return data.map(item => sanitizeData(item, rules))
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
if (typeof data === 'object') {
|
|
232
|
+
const result: any = {}
|
|
233
|
+
const BLOCKED_KEYS = ['__proto__', 'constructor', 'prototype']
|
|
234
|
+
for (const [key, value] of Object.entries(data)) {
|
|
235
|
+
if (BLOCKED_KEYS.includes(key)) continue
|
|
236
|
+
// Sanitize keys (prevent prototype pollution, allow hyphens)
|
|
237
|
+
const sanitizedKey = key.replace(/[^a-zA-Z0-9_-]/g, '')
|
|
238
|
+
if (!sanitizedKey) continue
|
|
239
|
+
result[sanitizedKey] = sanitizeData(value, rules)
|
|
240
|
+
}
|
|
241
|
+
return result
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
return data
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// ─── CUSTOM SCRIPT LOADER ────────────────────────────────────────────────
|
|
248
|
+
|
|
249
|
+
/**
|
|
250
|
+
* Loads a custom handler from the dynamic webhook registry.
|
|
251
|
+
*
|
|
252
|
+
* Handlers are registered in the `webhook_handlers` table at runtime,
|
|
253
|
+
* allowing custom functions to self-register without core code changes.
|
|
254
|
+
*
|
|
255
|
+
* @param handlerName - The handler key from integration config (e.g. "webhook-handler")
|
|
256
|
+
*/
|
|
257
|
+
async function loadCustomScript(handlerName: string): Promise<Function | null> {
|
|
258
|
+
return resolveHandler(handlerName)
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
// ─── DIAGNOSTIC HANDLER (temporary — full trace in response) ──────────────
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* Integration router with full diagnostic tracing.
|
|
265
|
+
* Every step logs to a trace[] array that is returned in the response body.
|
|
266
|
+
* REMOVE THIS DIAGNOSTIC MODE before production.
|
|
267
|
+
*/
|
|
268
|
+
const integrationHandler = async (event: any, _context: any) => {
|
|
269
|
+
const requestId = crypto.randomUUID()
|
|
270
|
+
const trace: { step: string; status: string; detail?: any }[] = []
|
|
271
|
+
|
|
272
|
+
// ── Step 0: Can an external service hit the endpoint? ───────────────
|
|
273
|
+
trace.push({
|
|
274
|
+
step: '0_endpoint_reached',
|
|
275
|
+
status: 'PASS',
|
|
276
|
+
detail: {
|
|
277
|
+
method: event.httpMethod,
|
|
278
|
+
path: event.path,
|
|
279
|
+
queryParams: event.queryStringParameters,
|
|
280
|
+
hasBody: !!event.body,
|
|
281
|
+
timestamp: new Date().toISOString()
|
|
282
|
+
}
|
|
283
|
+
})
|
|
284
|
+
|
|
285
|
+
// Method check
|
|
286
|
+
if ((event.httpMethod || 'GET') !== 'POST') {
|
|
287
|
+
trace.push({ step: '0_method_check', status: 'FAIL', detail: `Expected POST, got ${event.httpMethod}` })
|
|
288
|
+
return respond(405, { status: 'rejected', trace })
|
|
289
|
+
}
|
|
290
|
+
trace.push({ step: '0_method_check', status: 'PASS' })
|
|
291
|
+
|
|
292
|
+
// Slug check
|
|
293
|
+
const integrationSlug = event.queryStringParameters?.slug
|
|
294
|
+
if (!integrationSlug) {
|
|
295
|
+
trace.push({ step: '0_slug_check', status: 'FAIL', detail: 'No ?slug= query param' })
|
|
296
|
+
return respond(400, { status: 'rejected', trace })
|
|
297
|
+
}
|
|
298
|
+
trace.push({ step: '0_slug_check', status: 'PASS', detail: { slug: integrationSlug } })
|
|
299
|
+
|
|
300
|
+
// ── Step 1: Does the endpoint see the API key header? ───────────────
|
|
301
|
+
const apiKeyRaw = event.headers?.['x-api-key'] || event.headers?.['X-Api-Key'] || null
|
|
302
|
+
trace.push({
|
|
303
|
+
step: '1_header_extraction',
|
|
304
|
+
status: apiKeyRaw ? 'PASS' : 'FAIL',
|
|
305
|
+
detail: {
|
|
306
|
+
'x-api-key_present': !!apiKeyRaw,
|
|
307
|
+
'x-api-key_prefix': apiKeyRaw ? apiKeyRaw.substring(0, 6) + '...' : null,
|
|
308
|
+
all_header_keys: Object.keys(event.headers || {})
|
|
309
|
+
}
|
|
310
|
+
})
|
|
311
|
+
if (!apiKeyRaw) {
|
|
312
|
+
return respond(401, { status: 'rejected', trace })
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
// ── Step 2: Does resolvePrincipal validate the key? ─────────────────
|
|
316
|
+
let principal: any
|
|
317
|
+
try {
|
|
318
|
+
principal = await resolvePrincipal(event)
|
|
319
|
+
const isAnon = !principal || principal.id === 'anonymous'
|
|
320
|
+
trace.push({
|
|
321
|
+
step: '2_api_key_validation',
|
|
322
|
+
status: isAnon ? 'FAIL' : 'PASS',
|
|
323
|
+
detail: {
|
|
324
|
+
principal_id: principal?.id,
|
|
325
|
+
principal_type: principal?.type,
|
|
326
|
+
account_id: principal?.accountId,
|
|
327
|
+
scopes: principal?.scopes,
|
|
328
|
+
machine_type: principal?.machineType,
|
|
329
|
+
is_internal: principal?.isInternal
|
|
330
|
+
}
|
|
331
|
+
})
|
|
332
|
+
if (isAnon) {
|
|
333
|
+
return respond(401, { status: 'rejected', trace })
|
|
334
|
+
}
|
|
335
|
+
} catch (err: any) {
|
|
336
|
+
trace.push({
|
|
337
|
+
step: '2_api_key_validation',
|
|
338
|
+
status: 'ERROR',
|
|
339
|
+
detail: { message: err.message, stack: err.stack?.split('\n').slice(0, 3) }
|
|
340
|
+
})
|
|
341
|
+
return respond(401, { status: 'rejected', trace })
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
// ── Step 3: Permission check ────────────────────────────────────────
|
|
345
|
+
const permViaIntegrations = (principal as any).permissions?.integrations?.[integrationSlug]
|
|
346
|
+
const permViaScopes = principal.scopes?.includes('webhook:execute')
|
|
347
|
+
const hasPermission = !!permViaIntegrations?.includes?.('execute') || !!permViaScopes
|
|
348
|
+
trace.push({
|
|
349
|
+
step: '3_permission_check',
|
|
350
|
+
status: hasPermission ? 'PASS' : 'FAIL',
|
|
351
|
+
detail: {
|
|
352
|
+
checked_permissions_path: `principal.permissions.integrations.${integrationSlug}`,
|
|
353
|
+
permissions_value: permViaIntegrations ?? 'undefined (property does not exist)',
|
|
354
|
+
checked_scopes: 'webhook:execute',
|
|
355
|
+
scopes_value: principal.scopes,
|
|
356
|
+
scopes_match: permViaScopes,
|
|
357
|
+
final_result: hasPermission
|
|
358
|
+
}
|
|
359
|
+
})
|
|
360
|
+
if (!hasPermission) {
|
|
361
|
+
return respond(403, { status: 'rejected', trace })
|
|
362
|
+
}
|
|
363
|
+
|
|
364
|
+
// ── Step 4: Resolve integration record ──────────────────────────────
|
|
365
|
+
let integration: any
|
|
366
|
+
try {
|
|
367
|
+
integration = await resolveIntegration(integrationSlug, 'webhook')
|
|
368
|
+
trace.push({
|
|
369
|
+
step: '4_integration_resolution',
|
|
370
|
+
status: 'PASS',
|
|
371
|
+
detail: {
|
|
372
|
+
id: integration.id,
|
|
373
|
+
name: integration.name,
|
|
374
|
+
type: integration.integration_type,
|
|
375
|
+
is_active: integration.is_active,
|
|
376
|
+
is_configured: integration.is_configured,
|
|
377
|
+
has_config: !!integration.config,
|
|
378
|
+
config_keys: integration.config ? Object.keys(integration.config) : []
|
|
379
|
+
}
|
|
380
|
+
})
|
|
381
|
+
} catch (err: any) {
|
|
382
|
+
trace.push({
|
|
383
|
+
step: '4_integration_resolution',
|
|
384
|
+
status: 'ERROR',
|
|
385
|
+
detail: { message: err.message }
|
|
386
|
+
})
|
|
387
|
+
return respond(500, { status: 'rejected', trace })
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
// ── Step 5: Parse request body ──────────────────────────────────────
|
|
391
|
+
let body: any = {}
|
|
392
|
+
try {
|
|
393
|
+
body = event.body ? JSON.parse(event.body) : {}
|
|
394
|
+
trace.push({
|
|
395
|
+
step: '5_body_parse',
|
|
396
|
+
status: 'PASS',
|
|
397
|
+
detail: { parsed_keys: Object.keys(body), parsed_body: body }
|
|
398
|
+
})
|
|
399
|
+
} catch (e: any) {
|
|
400
|
+
body = { raw: event.body }
|
|
401
|
+
trace.push({
|
|
402
|
+
step: '5_body_parse',
|
|
403
|
+
status: 'WARN',
|
|
404
|
+
detail: { message: 'Not valid JSON, wrapped as { raw: ... }', raw_length: event.body?.length }
|
|
405
|
+
})
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
// ── Step 6: Schema validation ───────────────────────────────────────
|
|
409
|
+
const config = integration.config || {}
|
|
410
|
+
if (config.validation?.schema) {
|
|
411
|
+
const validation = validateSchema(body, config.validation.schema, requestId)
|
|
412
|
+
trace.push({
|
|
413
|
+
step: '6_schema_validation',
|
|
414
|
+
status: validation.valid ? 'PASS' : 'FAIL',
|
|
415
|
+
detail: {
|
|
416
|
+
schema_fields: Object.keys(config.validation.schema),
|
|
417
|
+
body_fields: Object.keys(body),
|
|
418
|
+
errors: validation.errors,
|
|
419
|
+
unknown_fields: validation.unknownFields,
|
|
420
|
+
reject_unknown: config.validation.rejectUnknownFields
|
|
421
|
+
}
|
|
422
|
+
})
|
|
423
|
+
if (!validation.valid) {
|
|
424
|
+
return respond(400, { status: 'rejected', trace })
|
|
425
|
+
}
|
|
426
|
+
} else {
|
|
427
|
+
trace.push({ step: '6_schema_validation', status: 'SKIP', detail: 'No validation.schema in config' })
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
// ── Step 7: Sanitize data ───────────────────────────────────────────
|
|
431
|
+
const sanitizedData = sanitizeData(body, config.validation?.sanitization || 'strict')
|
|
432
|
+
trace.push({
|
|
433
|
+
step: '7_sanitization',
|
|
434
|
+
status: 'PASS',
|
|
435
|
+
detail: {
|
|
436
|
+
input_keys: Object.keys(body),
|
|
437
|
+
output_keys: Object.keys(sanitizedData),
|
|
438
|
+
sanitized_data: sanitizedData,
|
|
439
|
+
note: 'Key sanitizer strips non-alphanumeric/underscore chars (hyphens removed)'
|
|
440
|
+
}
|
|
441
|
+
})
|
|
442
|
+
|
|
443
|
+
// ── Step 8: Locate custom handler ───────────────────────────────────
|
|
444
|
+
if (!config.handler?.path) {
|
|
445
|
+
trace.push({ step: '8_handler_lookup', status: 'SKIP', detail: 'No handler.path in config' })
|
|
446
|
+
return respond(200, { status: 'success', trace, result: sanitizedData })
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
const scriptHandler = await loadCustomScript(config.handler.path)
|
|
450
|
+
const registeredHandlers = await listRegisteredHandlers()
|
|
451
|
+
trace.push({
|
|
452
|
+
step: '8_handler_lookup',
|
|
453
|
+
status: scriptHandler ? 'PASS' : 'FAIL',
|
|
454
|
+
detail: {
|
|
455
|
+
handler_key: config.handler.path,
|
|
456
|
+
found_in_registry: !!scriptHandler,
|
|
457
|
+
registered_handlers: registeredHandlers,
|
|
458
|
+
note: 'Dynamic registry - handlers self-register via webhook_handlers table'
|
|
459
|
+
}
|
|
460
|
+
})
|
|
461
|
+
if (!scriptHandler) {
|
|
462
|
+
return respond(500, { status: 'rejected', trace })
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
// ── Step 9: Prepare handler arguments ───────────────────────────────
|
|
466
|
+
const scriptContext = {
|
|
467
|
+
integrationId: integration.id,
|
|
468
|
+
accountId: integration.account_id,
|
|
469
|
+
slug: config.slug || integration.name,
|
|
470
|
+
principal: { id: principal.id, type: principal.type, accountId: principal.accountId },
|
|
471
|
+
requestId,
|
|
472
|
+
headers: event.headers || {}
|
|
473
|
+
}
|
|
474
|
+
const scriptEvent = {
|
|
475
|
+
httpMethod: event.httpMethod,
|
|
476
|
+
headers: event.headers || {},
|
|
477
|
+
body: body,
|
|
478
|
+
path: event.path,
|
|
479
|
+
queryStringParameters: event.queryStringParameters || {}
|
|
480
|
+
}
|
|
481
|
+
trace.push({
|
|
482
|
+
step: '9_handler_args',
|
|
483
|
+
status: 'PASS',
|
|
484
|
+
detail: {
|
|
485
|
+
arg1_sanitized_data: sanitizedData,
|
|
486
|
+
arg2_context_keys: Object.keys(scriptContext),
|
|
487
|
+
arg3_event_keys: Object.keys(scriptEvent)
|
|
488
|
+
}
|
|
489
|
+
})
|
|
490
|
+
|
|
491
|
+
// ── Step 10: Execute handler ────────────────────────────────────────
|
|
492
|
+
try {
|
|
493
|
+
const result = await scriptHandler(sanitizedData, scriptContext, scriptEvent)
|
|
494
|
+
trace.push({
|
|
495
|
+
step: '10_handler_execution',
|
|
496
|
+
status: 'PASS',
|
|
497
|
+
detail: { result_type: typeof result, result }
|
|
498
|
+
})
|
|
499
|
+
return respond(200, { status: 'success', trace, result })
|
|
500
|
+
} catch (err: any) {
|
|
501
|
+
trace.push({
|
|
502
|
+
step: '10_handler_execution',
|
|
503
|
+
status: 'ERROR',
|
|
504
|
+
detail: { message: err.message, stack: err.stack?.split('\n').slice(0, 5) }
|
|
505
|
+
})
|
|
506
|
+
return respond(500, { status: 'rejected', trace })
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
// ─── HELPERS ──────────────────────────────────────────────────────────────
|
|
511
|
+
|
|
512
|
+
function respond(statusCode: number, body: any) {
|
|
513
|
+
return {
|
|
514
|
+
statusCode,
|
|
515
|
+
headers: { 'Content-Type': 'application/json' },
|
|
516
|
+
body: JSON.stringify(body, null, 2)
|
|
517
|
+
}
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
// ─── EXPORTS ─────────────────────────────────────────────────────────────
|
|
521
|
+
|
|
522
|
+
const handler = integrationHandler
|
|
523
|
+
export { handler }
|