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,319 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module integrations
|
|
3
|
+
* @audience core-contributor
|
|
4
|
+
* @layer api-handler
|
|
5
|
+
* @stability stable
|
|
6
|
+
*
|
|
7
|
+
* CRUD API for the `integrations` table. Integration records describe
|
|
8
|
+
* third-party service connections (API credentials, provider, version,
|
|
9
|
+
* configuration). Each integration is scoped to an account and optionally
|
|
10
|
+
* to an app. `is_configured` tracks whether credentials have been set.
|
|
11
|
+
*
|
|
12
|
+
* **Routed by:** `GET/POST/PATCH/DELETE /.netlify/functions/integrations`
|
|
13
|
+
*
|
|
14
|
+
* **Standard CRUD — routes directly by HTTP method (no ?action switch):**
|
|
15
|
+
* | method | condition | handler |
|
|
16
|
+
* |--------|-----------|---------|
|
|
17
|
+
* | GET | ?id | get |
|
|
18
|
+
* | GET | (default) | list |
|
|
19
|
+
* | POST | — | create |
|
|
20
|
+
* | PATCH | — | update |
|
|
21
|
+
* | DELETE | — | remove (soft) |
|
|
22
|
+
*
|
|
23
|
+
* **Authorization:** All operations use `ctx.db` (RLS-scoped). Authenticated
|
|
24
|
+
* principal required for writes.
|
|
25
|
+
*
|
|
26
|
+
* INVARIANT: `remove` is a soft delete (sets `is_active = false`). Hard deletes
|
|
27
|
+
* are not supported to preserve audit trails on integration-linked data.
|
|
28
|
+
* INVARIANT: `update` only patches the explicit allowlist of fields.
|
|
29
|
+
*
|
|
30
|
+
* @seeAlso api-keys.ts (api_keys belong to integrations)
|
|
31
|
+
* @seeAlso trigger-engine.ts (integration webhooks trigger pipelines)
|
|
32
|
+
* @seeAlso audit.ts (emitLog for integration.* events)
|
|
33
|
+
*/
|
|
34
|
+
|
|
35
|
+
import { createHandler } from './_shared/middleware'
|
|
36
|
+
import { joins } from './_shared/db'
|
|
37
|
+
import { emitLog } from './_shared/audit'
|
|
38
|
+
import { sanitizeRecordData } from './_shared/permissions'
|
|
39
|
+
|
|
40
|
+
// ─── HANDLERS ─────────────────────────────────────────────────────────────────
|
|
41
|
+
|
|
42
|
+
// ─── CHUNK_START: INTEGRATIONS_LIST ──────────────────────────────────────────────
|
|
43
|
+
/**
|
|
44
|
+
* @chunk-id INTEGRATIONS_LIST_1_0_0
|
|
45
|
+
* @version 1.0.0
|
|
46
|
+
* @hash 5faf0b7052d9aac5f764d1fc30ddde1485a0296ad2b209cd2e3a73c09061a1fc
|
|
47
|
+
* @macro Integrations List Handler
|
|
48
|
+
* @micro Lists integrations with filtering, pagination, and joins
|
|
49
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
50
|
+
* @inputs _body: any — Request body (unused for GET)
|
|
51
|
+
* @outputs Array of sanitized integration records with app/createdBy joins
|
|
52
|
+
* @depends-on [createHandler, joins, sanitizeRecordData]
|
|
53
|
+
* @depended-by [Netlify function routing]
|
|
54
|
+
* @side-effects [DB queries, permission sanitization]
|
|
55
|
+
* @tags integrations, list, crud, pagination
|
|
56
|
+
*/
|
|
57
|
+
export const list = createHandler(async (ctx, _body) => {
|
|
58
|
+
const { integration_type, provider, is_active, is_configured, limit = 100, offset = 0 } = ctx.query || {}
|
|
59
|
+
|
|
60
|
+
if (!ctx.accountId) {
|
|
61
|
+
throw new Error('Account context required')
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
let query = ctx.db
|
|
65
|
+
.from('integrations')
|
|
66
|
+
.select(`*, ${joins.app}, ${joins.createdBy}`)
|
|
67
|
+
.order('name')
|
|
68
|
+
|
|
69
|
+
if (integration_type) {
|
|
70
|
+
query = query.eq('integration_type', integration_type)
|
|
71
|
+
}
|
|
72
|
+
if (provider) {
|
|
73
|
+
query = query.eq('provider', provider)
|
|
74
|
+
}
|
|
75
|
+
if (is_active !== undefined) {
|
|
76
|
+
query = query.eq('is_active', is_active === 'true')
|
|
77
|
+
}
|
|
78
|
+
if (is_configured !== undefined) {
|
|
79
|
+
query = query.eq('is_configured', is_configured === 'true')
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
const { data, error: err } = await query.range(
|
|
83
|
+
parseInt(offset.toString()),
|
|
84
|
+
parseInt(offset.toString()) + parseInt(limit.toString()) - 1
|
|
85
|
+
)
|
|
86
|
+
|
|
87
|
+
if (err) throw err
|
|
88
|
+
|
|
89
|
+
const sanitized = []
|
|
90
|
+
for (const integration of data || []) {
|
|
91
|
+
sanitized.push(await sanitizeRecordData(ctx, integration, 'integration'))
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
return sanitized
|
|
95
|
+
})
|
|
96
|
+
// ─── CHUNK_END: INTEGRATIONS_LIST ────────────────────────────────────────────────
|
|
97
|
+
|
|
98
|
+
// ─── CHUNK_START: INTEGRATIONS_GET ──────────────────────────────────────────────
|
|
99
|
+
/**
|
|
100
|
+
* @chunk-id INTEGRATIONS_GET_1_0_0
|
|
101
|
+
* @version 1.0.0
|
|
102
|
+
* @hash 975126629cff24c75d2b74328d1ae08dd033ed451d3820d828c62b1f7a27413e
|
|
103
|
+
* @macro Integration Get Handler
|
|
104
|
+
* @micro Returns single integration record with joins and sanitization
|
|
105
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
106
|
+
* @inputs _body: any — Request body (unused for GET)
|
|
107
|
+
* @outputs Sanitized integration record with app/createdBy joins
|
|
108
|
+
* @depends-on [createHandler, joins, sanitizeRecordData]
|
|
109
|
+
* @depended-by [Netlify function routing]
|
|
110
|
+
* @side-effects [DB single row query, permission sanitization]
|
|
111
|
+
* @tags integrations, get, crud, single-record
|
|
112
|
+
*/
|
|
113
|
+
export const get = createHandler(async (ctx, _body) => {
|
|
114
|
+
const { id } = ctx.query || {}
|
|
115
|
+
|
|
116
|
+
if (!id) {
|
|
117
|
+
throw new Error('Integration ID is required')
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
const { data, error: err } = await ctx.db
|
|
121
|
+
.from('integrations')
|
|
122
|
+
.select(`*, ${joins.app}, ${joins.createdBy}`)
|
|
123
|
+
.eq('id', id)
|
|
124
|
+
.single()
|
|
125
|
+
|
|
126
|
+
if (err) throw err
|
|
127
|
+
|
|
128
|
+
return await sanitizeRecordData(ctx, data, 'integration')
|
|
129
|
+
})
|
|
130
|
+
// ─── CHUNK_END: INTEGRATIONS_GET ────────────────────────────────────────────────
|
|
131
|
+
|
|
132
|
+
// ─── CHUNK_START: INTEGRATIONS_CREATE ──────────────────────────────────────────────
|
|
133
|
+
/**
|
|
134
|
+
* @chunk-id INTEGRATIONS_CREATE_1_0_0
|
|
135
|
+
* @version 1.0.0
|
|
136
|
+
* @hash 27d2242703930d7eeb33b49bbce4c22681ab891750c4c7a298c1c717a39870d7
|
|
137
|
+
* @macro Integration Create Handler
|
|
138
|
+
* @micro Creates integration record with validation and audit logging
|
|
139
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
140
|
+
* @inputs body: object — Integration data including name, integration_type, provider
|
|
141
|
+
* @outputs Inserted integration record
|
|
142
|
+
* @depends-on [createHandler, emitLog]
|
|
143
|
+
* @depended-by [Netlify function routing]
|
|
144
|
+
* @side-effects [DB insert, audit logging]
|
|
145
|
+
* @tags integrations, create, crud, audit
|
|
146
|
+
*/
|
|
147
|
+
export const create = createHandler(async (ctx, body) => {
|
|
148
|
+
const { app_id, name, description, integration_type, provider, version, config, credentials, metadata } = body
|
|
149
|
+
|
|
150
|
+
if (!name || !integration_type || !provider) {
|
|
151
|
+
throw new Error('name, integration_type, and provider are required')
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
if (!ctx.principal || ctx.principal.id === 'anonymous' || !ctx.accountId) {
|
|
155
|
+
throw new Error('User context (person and account) required')
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
const { data, error: err } = await ctx.db
|
|
159
|
+
.from('integrations')
|
|
160
|
+
.insert({
|
|
161
|
+
app_id: app_id || null,
|
|
162
|
+
account_id: ctx.accountId,
|
|
163
|
+
name,
|
|
164
|
+
description: description || null,
|
|
165
|
+
integration_type,
|
|
166
|
+
provider,
|
|
167
|
+
version: version || '1.0.0',
|
|
168
|
+
config: config || {},
|
|
169
|
+
credentials: credentials || {},
|
|
170
|
+
metadata: metadata || {},
|
|
171
|
+
created_by: ctx.principal.id
|
|
172
|
+
})
|
|
173
|
+
.select()
|
|
174
|
+
.single()
|
|
175
|
+
|
|
176
|
+
if (err) throw err
|
|
177
|
+
|
|
178
|
+
await emitLog(ctx, 'integration.created',
|
|
179
|
+
{ type: 'integration', id: data.id },
|
|
180
|
+
{ after: { name, integration_type, provider } }
|
|
181
|
+
)
|
|
182
|
+
|
|
183
|
+
return data
|
|
184
|
+
})
|
|
185
|
+
// ─── CHUNK_END: INTEGRATIONS_CREATE ────────────────────────────────────────────────
|
|
186
|
+
|
|
187
|
+
// ─── CHUNK_START: INTEGRATIONS_UPDATE ──────────────────────────────────────────────
|
|
188
|
+
/**
|
|
189
|
+
* @chunk-id INTEGRATIONS_UPDATE_1_0_0
|
|
190
|
+
* @version 1.0.0
|
|
191
|
+
* @hash ab5c8404ddec24ff912c55e7e52cd0ecdd220646b0e07f56a49a02ef3288f3e9
|
|
192
|
+
* @macro Integration Update Handler
|
|
193
|
+
* @micro Updates integration with field allowlist and audit logging
|
|
194
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
195
|
+
* @inputs body: object — Integration updates including id
|
|
196
|
+
* @outputs Updated integration record
|
|
197
|
+
* @depends-on [createHandler, emitLog]
|
|
198
|
+
* @depended-by [Netlify function routing]
|
|
199
|
+
* @side-effects [DB update, audit logging]
|
|
200
|
+
* @tags integrations, update, crud, audit
|
|
201
|
+
*/
|
|
202
|
+
export const update = createHandler(async (ctx, body) => {
|
|
203
|
+
const id = body?.id || ctx.query?.id
|
|
204
|
+
const { id: _bodyId, ...updates } = body || {}
|
|
205
|
+
|
|
206
|
+
if (!id) {
|
|
207
|
+
throw new Error('Integration ID is required')
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
const allowed = ['name', 'description', 'integration_type', 'provider', 'version', 'config', 'credentials', 'metadata', 'is_active', 'is_configured']
|
|
211
|
+
const updateData: Record<string, any> = { updated_at: new Date().toISOString() }
|
|
212
|
+
for (const key of allowed) {
|
|
213
|
+
if (updates[key] !== undefined) updateData[key] = updates[key]
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const { data, error: err } = await ctx.db
|
|
217
|
+
.from('integrations')
|
|
218
|
+
.update(updateData)
|
|
219
|
+
.eq('id', id)
|
|
220
|
+
.select()
|
|
221
|
+
.single()
|
|
222
|
+
|
|
223
|
+
if (err) throw err
|
|
224
|
+
|
|
225
|
+
await emitLog(ctx, 'integration.updated',
|
|
226
|
+
{ type: 'integration', id },
|
|
227
|
+
{ after: updateData }
|
|
228
|
+
)
|
|
229
|
+
|
|
230
|
+
return data
|
|
231
|
+
})
|
|
232
|
+
// ─── CHUNK_END: INTEGRATIONS_UPDATE ────────────────────────────────────────────────
|
|
233
|
+
|
|
234
|
+
// ─── CHUNK_START: INTEGRATIONS_REMOVE ──────────────────────────────────────────────
|
|
235
|
+
/**
|
|
236
|
+
* @chunk-id INTEGRATIONS_REMOVE_1_0_0
|
|
237
|
+
* @version 1.0.0
|
|
238
|
+
* @hash a8b5f084d745d2ce69a1e29d74903832ed39647ccb779d0f651a99157777a08a
|
|
239
|
+
* @macro Integration Remove Handler
|
|
240
|
+
* @micro Soft-deletes integration with validation and audit logging
|
|
241
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
242
|
+
* @inputs _body: any — Request body (unused for DELETE)
|
|
243
|
+
* @outputs Updated integration record with is_active: false
|
|
244
|
+
* @depends-on [createHandler, emitLog]
|
|
245
|
+
* @depended-by [Netlify function routing]
|
|
246
|
+
* @side-effects [DB soft delete, audit logging]
|
|
247
|
+
* @tags integrations, remove, crud, audit
|
|
248
|
+
*/
|
|
249
|
+
export const remove = createHandler(async (ctx, _body) => {
|
|
250
|
+
const id = ctx.query?.id
|
|
251
|
+
|
|
252
|
+
if (!id) {
|
|
253
|
+
throw new Error('Integration ID is required')
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
const { data: current } = await ctx.db
|
|
257
|
+
.from('integrations')
|
|
258
|
+
.select('id, name, provider')
|
|
259
|
+
.eq('id', id)
|
|
260
|
+
.single()
|
|
261
|
+
|
|
262
|
+
if (!current) throw new Error('Integration not found')
|
|
263
|
+
|
|
264
|
+
const { data, error: err } = await ctx.db
|
|
265
|
+
.from('integrations')
|
|
266
|
+
.update({ is_active: false, updated_at: new Date().toISOString() })
|
|
267
|
+
.eq('id', id)
|
|
268
|
+
.select()
|
|
269
|
+
.single()
|
|
270
|
+
|
|
271
|
+
if (err) throw err
|
|
272
|
+
|
|
273
|
+
await emitLog(ctx, 'integration.deleted',
|
|
274
|
+
{ type: 'integration', id },
|
|
275
|
+
{ before: current, after: { is_active: false } }
|
|
276
|
+
)
|
|
277
|
+
|
|
278
|
+
return data
|
|
279
|
+
})
|
|
280
|
+
// ─── CHUNK_END: INTEGRATIONS_REMOVE ────────────────────────────────────────────────
|
|
281
|
+
|
|
282
|
+
// ─── MAIN HANDLER ────────────────────────────────────────────────────────────
|
|
283
|
+
|
|
284
|
+
// ─── CHUNK_START: INTEGRATIONS_HANDLER ──────────────────────────────────────────────
|
|
285
|
+
/**
|
|
286
|
+
* @chunk-id INTEGRATIONS_HANDLER_1_0_0
|
|
287
|
+
* @version 1.0.0
|
|
288
|
+
* @hash 14c00f276a22287ed65c019efe327cbb513f860a408191e299864b05c641833c
|
|
289
|
+
* @macro Integrations Router
|
|
290
|
+
* @micro Routes HTTP methods to appropriate handlers (no action switch)
|
|
291
|
+
* @inputs ctx: CoreContext — Request context with principal and database
|
|
292
|
+
* @inputs body: any — Request body for POST/PATCH operations
|
|
293
|
+
* @outputs Varies — Depends on routed handler (list/get/create/update/remove)
|
|
294
|
+
* @depends-on [createHandler, list, get, create, update, remove]
|
|
295
|
+
* @depended-by [Netlify function routing]
|
|
296
|
+
* @side-effects [Delegates to appropriate handler]
|
|
297
|
+
* @tags integrations, router, crud, netlify-function
|
|
298
|
+
*/
|
|
299
|
+
export const handler = createHandler(async (ctx, body) => {
|
|
300
|
+
const method = ctx.query?.method || 'GET'
|
|
301
|
+
|
|
302
|
+
switch (method) {
|
|
303
|
+
case 'GET':
|
|
304
|
+
if (ctx.query?.id) {
|
|
305
|
+
return await get(ctx, body)
|
|
306
|
+
} else {
|
|
307
|
+
return await list(ctx, body)
|
|
308
|
+
}
|
|
309
|
+
case 'POST':
|
|
310
|
+
return await create(ctx, body)
|
|
311
|
+
case 'PATCH':
|
|
312
|
+
return await update(ctx, body)
|
|
313
|
+
case 'DELETE':
|
|
314
|
+
return await remove(ctx, body)
|
|
315
|
+
default:
|
|
316
|
+
throw new Error(`Unsupported method: ${method}`)
|
|
317
|
+
}
|
|
318
|
+
})
|
|
319
|
+
// ─── CHUNK_END: INTEGRATIONS_HANDLER ────────────────────────────────────────────────
|
|
@@ -0,0 +1,272 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module item-progress
|
|
3
|
+
* @audience both
|
|
4
|
+
* @layer api-handler
|
|
5
|
+
* @stability stable
|
|
6
|
+
*
|
|
7
|
+
* CRUD + upsert API for the `item_progress` table. Tracks per-person, per-item
|
|
8
|
+
* progress state for courses, tasks, onboarding, and any item-based pipeline.
|
|
9
|
+
*
|
|
10
|
+
* Routed by: GET/POST/PATCH /.netlify/functions/item-progress
|
|
11
|
+
*
|
|
12
|
+
* INVARIANT: (person_id, item_id) is unique — state, not a log.
|
|
13
|
+
* INVARIANT: status transitions are forward-only unless force: true.
|
|
14
|
+
* INVARIANT: score must be 0–100 or null.
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import { createHandler } from './_shared/middleware'
|
|
18
|
+
import { adminDb } from './_shared/db'
|
|
19
|
+
import { emitLog } from './_shared/audit'
|
|
20
|
+
|
|
21
|
+
// ─── HELPERS ──────────────────────────────────────────────────────────────────
|
|
22
|
+
|
|
23
|
+
const PIPELINE_DEFAULT = ['not_started', 'in_progress', 'completed']
|
|
24
|
+
|
|
25
|
+
function getPipeline(typeRecord: any): string[] {
|
|
26
|
+
return typeRecord?.design_schema?.pipeline ?? PIPELINE_DEFAULT
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function isValidTransition(pipeline: string[], from: string, to: string): boolean {
|
|
30
|
+
const fromIdx = pipeline.indexOf(from)
|
|
31
|
+
const toIdx = pipeline.indexOf(to)
|
|
32
|
+
if (fromIdx === -1 || toIdx === -1) return true
|
|
33
|
+
return toIdx >= fromIdx
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function composeTitle(itemTitle: string | null, personName: string | null): string | null {
|
|
37
|
+
if (!itemTitle && !personName) return null
|
|
38
|
+
if (!personName) return itemTitle
|
|
39
|
+
if (!itemTitle) return personName
|
|
40
|
+
return `${itemTitle} — ${personName}`
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function composeDescription(status: string, score: number | null, attempts: number | null): string {
|
|
44
|
+
const parts: string[] = []
|
|
45
|
+
const label = status === 'completed' ? 'Completed' : status === 'in_progress' ? 'In Progress' : 'Not Started'
|
|
46
|
+
parts.push(label)
|
|
47
|
+
if (score !== null && score !== undefined) parts.push(`score ${score}`)
|
|
48
|
+
if (attempts !== null && attempts !== undefined && attempts > 0) parts.push(`${attempts} attempt${attempts === 1 ? '' : 's'}`)
|
|
49
|
+
return parts.join(' · ')
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// ─── HANDLERS ─────────────────────────────────────────────────────────────────
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Lists progress records for the authenticated principal.
|
|
56
|
+
* Query params: person_id, item_id, item_ids (comma-sep), status, limit, offset
|
|
57
|
+
*/
|
|
58
|
+
export const list = createHandler(async (ctx) => {
|
|
59
|
+
const { person_id, item_id, item_ids, status, limit = '100', offset = '0' } = ctx.query || {}
|
|
60
|
+
|
|
61
|
+
let query = ctx.db
|
|
62
|
+
.from('item_progress')
|
|
63
|
+
.select('*')
|
|
64
|
+
.eq('is_active', true)
|
|
65
|
+
.order('updated_at', { ascending: false })
|
|
66
|
+
.range(Number(offset), Number(offset) + Number(limit) - 1)
|
|
67
|
+
|
|
68
|
+
if (person_id) query = query.eq('person_id', person_id)
|
|
69
|
+
if (item_id) query = query.eq('item_id', item_id)
|
|
70
|
+
if (item_ids) query = query.in('item_id', item_ids.split(',').map((s: string) => s.trim()))
|
|
71
|
+
if (status) query = query.eq('status', status)
|
|
72
|
+
|
|
73
|
+
const { data, error } = await query
|
|
74
|
+
if (error) throw error
|
|
75
|
+
return data
|
|
76
|
+
})
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Gets a single item_progress record by id.
|
|
80
|
+
*/
|
|
81
|
+
export const get = createHandler(async (ctx) => {
|
|
82
|
+
const id = ctx.query?.id
|
|
83
|
+
if (!id) throw new Error('Progress record ID is required')
|
|
84
|
+
|
|
85
|
+
const { data, error } = await ctx.db
|
|
86
|
+
.from('item_progress')
|
|
87
|
+
.select('*')
|
|
88
|
+
.eq('id', id)
|
|
89
|
+
.single()
|
|
90
|
+
|
|
91
|
+
if (error || !data) throw new Error('Progress record not found')
|
|
92
|
+
return data
|
|
93
|
+
})
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Upserts a progress record for (person_id, item_id).
|
|
97
|
+
* Auto-composes title and description. Enforces pipeline transitions.
|
|
98
|
+
* Sets started_at / completed_at timestamps in data. Increments attempts.
|
|
99
|
+
*
|
|
100
|
+
* Body: person_id*, item_id*, type_id*, account_id*, status, score, data,
|
|
101
|
+
* title, description, app_id, force (bypass direction check)
|
|
102
|
+
*/
|
|
103
|
+
export const upsert = createHandler(async (ctx, body) => {
|
|
104
|
+
const { person_id, item_id, type_id, account_id, app_id, force } = body || {}
|
|
105
|
+
let { status, score, data: dataPayload, title, description } = body || {}
|
|
106
|
+
|
|
107
|
+
if (!person_id || !item_id || !type_id || !account_id) {
|
|
108
|
+
throw new Error('person_id, item_id, type_id, and account_id are required')
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// Fetch type record for pipeline
|
|
112
|
+
const { data: typeRecord } = await adminDb
|
|
113
|
+
.from('types')
|
|
114
|
+
.select('design_schema')
|
|
115
|
+
.eq('id', type_id)
|
|
116
|
+
.single()
|
|
117
|
+
|
|
118
|
+
const pipeline = getPipeline(typeRecord)
|
|
119
|
+
|
|
120
|
+
// Fetch existing record (if any)
|
|
121
|
+
const { data: existing } = await ctx.db
|
|
122
|
+
.from('item_progress')
|
|
123
|
+
.select('*')
|
|
124
|
+
.eq('person_id', person_id)
|
|
125
|
+
.eq('item_id', item_id)
|
|
126
|
+
.maybeSingle()
|
|
127
|
+
|
|
128
|
+
const currentStatus = existing?.status ?? 'not_started'
|
|
129
|
+
const targetStatus = status ?? currentStatus
|
|
130
|
+
|
|
131
|
+
// Validate pipeline transition
|
|
132
|
+
if (status && existing && !force && !isValidTransition(pipeline, currentStatus, targetStatus)) {
|
|
133
|
+
throw new Error(`Invalid status transition: ${currentStatus} → ${targetStatus}. Use force: true to override.`)
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Merge data payload with timestamps and attempts
|
|
137
|
+
const now = new Date().toISOString()
|
|
138
|
+
const existingData = existing?.data ?? {}
|
|
139
|
+
const attempts = (existingData.attempts ?? 0) + (status && status !== currentStatus ? 1 : 0)
|
|
140
|
+
const mergedData: any = {
|
|
141
|
+
...existingData,
|
|
142
|
+
...(dataPayload || {}),
|
|
143
|
+
attempts,
|
|
144
|
+
}
|
|
145
|
+
if (targetStatus === 'in_progress' && !existingData.started_at) mergedData.started_at = now
|
|
146
|
+
if (targetStatus === 'completed' && !existingData.completed_at) mergedData.completed_at = now
|
|
147
|
+
|
|
148
|
+
// Auto-compose title/description if not provided
|
|
149
|
+
if (!title) {
|
|
150
|
+
const [{ data: itemRow }, { data: personRow }] = await Promise.all([
|
|
151
|
+
adminDb.from('items').select('title').eq('id', item_id).single(),
|
|
152
|
+
adminDb.from('people').select('full_name').eq('id', person_id).single(),
|
|
153
|
+
])
|
|
154
|
+
title = composeTitle(itemRow?.title ?? null, personRow?.full_name ?? null) ?? undefined
|
|
155
|
+
}
|
|
156
|
+
if (!description) {
|
|
157
|
+
description = composeDescription(targetStatus, score ?? existing?.score ?? null, mergedData.attempts)
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
const payload: any = {
|
|
161
|
+
type_id,
|
|
162
|
+
account_id,
|
|
163
|
+
app_id: app_id ?? existing?.app_id ?? null,
|
|
164
|
+
person_id,
|
|
165
|
+
item_id,
|
|
166
|
+
title: title ?? existing?.title ?? null,
|
|
167
|
+
description,
|
|
168
|
+
status: targetStatus,
|
|
169
|
+
score: score !== undefined ? score : (existing?.score ?? null),
|
|
170
|
+
data: mergedData,
|
|
171
|
+
is_active: true,
|
|
172
|
+
updated_at: now,
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
if (!existing) {
|
|
176
|
+
payload.created_by = ctx.principal?.id ?? null
|
|
177
|
+
payload.created_at = now
|
|
178
|
+
} else {
|
|
179
|
+
payload.updated_by = ctx.principal?.id ?? null
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
const { data: result, error } = await ctx.db
|
|
183
|
+
.from('item_progress')
|
|
184
|
+
.upsert(payload, { onConflict: 'person_id,item_id' })
|
|
185
|
+
.select()
|
|
186
|
+
.single()
|
|
187
|
+
|
|
188
|
+
if (error) throw error
|
|
189
|
+
|
|
190
|
+
await emitLog(ctx, 'item_progress.upserted',
|
|
191
|
+
{ type: 'item_progress', id: result.id },
|
|
192
|
+
{ before: existing ?? null, after: result }
|
|
193
|
+
)
|
|
194
|
+
|
|
195
|
+
return result
|
|
196
|
+
})
|
|
197
|
+
|
|
198
|
+
/**
|
|
199
|
+
* Partially updates an existing item_progress record.
|
|
200
|
+
* Body: id (required), plus any updatable fields.
|
|
201
|
+
* Enforces forward-only status transitions unless force: true.
|
|
202
|
+
*/
|
|
203
|
+
export const update = createHandler(async (ctx, body) => {
|
|
204
|
+
const id = body?.id || ctx.query?.id
|
|
205
|
+
if (!id) throw new Error('Progress record ID is required')
|
|
206
|
+
|
|
207
|
+
const { id: _id, force, ...updates } = body || {}
|
|
208
|
+
|
|
209
|
+
const { data: existing, error: fetchErr } = await ctx.db
|
|
210
|
+
.from('item_progress')
|
|
211
|
+
.select('*')
|
|
212
|
+
.eq('id', id)
|
|
213
|
+
.single()
|
|
214
|
+
|
|
215
|
+
if (fetchErr || !existing) throw new Error('Progress record not found')
|
|
216
|
+
|
|
217
|
+
if (updates.status && updates.status !== existing.status) {
|
|
218
|
+
const { data: typeRecord } = await adminDb
|
|
219
|
+
.from('types')
|
|
220
|
+
.select('design_schema')
|
|
221
|
+
.eq('id', existing.type_id)
|
|
222
|
+
.single()
|
|
223
|
+
|
|
224
|
+
const pipeline = getPipeline(typeRecord)
|
|
225
|
+
if (!force && !isValidTransition(pipeline, existing.status, updates.status)) {
|
|
226
|
+
throw new Error(`Invalid status transition: ${existing.status} → ${updates.status}. Use force: true to override.`)
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
const now = new Date().toISOString()
|
|
230
|
+
const mergedData = { ...existing.data, ...(updates.data || {}) }
|
|
231
|
+
if (updates.status === 'in_progress' && !existing.data?.started_at) mergedData.started_at = now
|
|
232
|
+
if (updates.status === 'completed' && !existing.data?.completed_at) mergedData.completed_at = now
|
|
233
|
+
updates.data = mergedData
|
|
234
|
+
|
|
235
|
+
if (!updates.description) {
|
|
236
|
+
updates.description = composeDescription(
|
|
237
|
+
updates.status,
|
|
238
|
+
updates.score ?? existing.score,
|
|
239
|
+
mergedData.attempts ?? null
|
|
240
|
+
)
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
const { data: result, error } = await ctx.db
|
|
245
|
+
.from('item_progress')
|
|
246
|
+
.update({ ...updates, updated_by: ctx.principal?.id ?? null, updated_at: new Date().toISOString() })
|
|
247
|
+
.eq('id', id)
|
|
248
|
+
.select()
|
|
249
|
+
.single()
|
|
250
|
+
|
|
251
|
+
if (error) throw error
|
|
252
|
+
|
|
253
|
+
await emitLog(ctx, 'item_progress.updated',
|
|
254
|
+
{ type: 'item_progress', id },
|
|
255
|
+
{ before: existing, after: result }
|
|
256
|
+
)
|
|
257
|
+
|
|
258
|
+
return result
|
|
259
|
+
})
|
|
260
|
+
|
|
261
|
+
// ─── ROUTER ──────────────────────────────────────────────────────────────────
|
|
262
|
+
|
|
263
|
+
export const handler = createHandler(async (ctx, body) => {
|
|
264
|
+
const method = ctx.query?.method || 'GET'
|
|
265
|
+
const id = ctx.query?.id
|
|
266
|
+
|
|
267
|
+
if (method === 'GET') return id ? get(ctx, body) : list(ctx, body)
|
|
268
|
+
if (method === 'POST') return upsert(ctx, body)
|
|
269
|
+
if (method === 'PATCH') return update(ctx, body)
|
|
270
|
+
|
|
271
|
+
throw new Error(`Method ${method} not supported`)
|
|
272
|
+
})
|