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.
Files changed (385) hide show
  1. package/.framework/README.md +129 -0
  2. package/.framework/cli/bin.cjs +14 -0
  3. package/.framework/cli/commands/agents.ts +153 -0
  4. package/.framework/cli/commands/auth.ts +94 -0
  5. package/.framework/cli/commands/create-app.ts +185 -0
  6. package/.framework/cli/commands/dev.ts +295 -0
  7. package/.framework/cli/commands/doctor.ts +442 -0
  8. package/.framework/cli/commands/generate.ts +332 -0
  9. package/.framework/cli/commands/init.ts +272 -0
  10. package/.framework/cli/commands/install-app.ts +391 -0
  11. package/.framework/cli/commands/items.ts +253 -0
  12. package/.framework/cli/commands/migrations.ts +141 -0
  13. package/.framework/cli/commands/pipelines.ts +166 -0
  14. package/.framework/cli/commands/status.ts +197 -0
  15. package/.framework/cli/commands/system.ts +184 -0
  16. package/.framework/cli/commands/test.ts +227 -0
  17. package/.framework/cli/commands/uninstall-app.ts +166 -0
  18. package/.framework/cli/context.ts +268 -0
  19. package/.framework/cli/env-loader.ts +36 -0
  20. package/.framework/cli/index.ts +106 -0
  21. package/.framework/cli/welcome.cjs +45 -0
  22. package/.framework/docs/API.md +384 -0
  23. package/.framework/docs/STABILITY.md +52 -0
  24. package/.framework/docs/admin-routes.md +76 -0
  25. package/.framework/docs/api-docs-progress.md +38 -0
  26. package/.framework/docs/api-governance.md +146 -0
  27. package/.framework/docs/api-testing-results.md +212 -0
  28. package/.framework/docs/apis/admin-configs.md +567 -0
  29. package/.framework/docs/apis/admin-data.md +272 -0
  30. package/.framework/docs/apis/index.md +231 -0
  31. package/.framework/docs/apis/internal.md +295 -0
  32. package/.framework/docs/apis/runtime.md +537 -0
  33. package/.framework/docs/assembly-launch-guide.md +138 -0
  34. package/.framework/docs/audit-results.md +590 -0
  35. package/.framework/docs/authorization-model.md +170 -0
  36. package/.framework/docs/db-api-inventory.md +95 -0
  37. package/.framework/docs/examples/custom-app/README.md +77 -0
  38. package/.framework/docs/examples/custom-function/README.md +27 -0
  39. package/.framework/docs/examples/custom-function/handler.ts +48 -0
  40. package/.framework/docs/examples/custom-webhook/README.md +68 -0
  41. package/.framework/docs/gap-remediation-backlog.md +103 -0
  42. package/.framework/docs/guides/cli-guide.md +224 -0
  43. package/.framework/docs/guides/getting-started.md +103 -0
  44. package/.framework/docs/guides/import-guide.md +193 -0
  45. package/.framework/docs/guides/testing-guide.md +229 -0
  46. package/.framework/docs/permission-examples.md +326 -0
  47. package/.framework/docs/ui-adoption-verification.md +111 -0
  48. package/.framework/docs/ui-api-coverage.md +84 -0
  49. package/.framework/docs/v2-compatibility-audit.md +228 -0
  50. package/.framework/functions/.gitkeep +1 -0
  51. package/.framework/functions/_shared/agent-runner.ts +1097 -0
  52. package/.framework/functions/_shared/app-manifest.ts +184 -0
  53. package/.framework/functions/_shared/audit.ts +150 -0
  54. package/.framework/functions/_shared/db.ts +174 -0
  55. package/.framework/functions/_shared/index.ts +382 -0
  56. package/.framework/functions/_shared/middleware.ts +490 -0
  57. package/.framework/functions/_shared/permissions.ts +1325 -0
  58. package/.framework/functions/_shared/pipeline-runner.ts +731 -0
  59. package/.framework/functions/_shared/principal.ts +760 -0
  60. package/.framework/functions/_shared/schema-utils.ts +967 -0
  61. package/.framework/functions/_shared/testing.ts +258 -0
  62. package/.framework/functions/_shared/trigger-engine.ts +425 -0
  63. package/.framework/functions/_shared/webhook-registration.ts +168 -0
  64. package/.framework/functions/_shared/webhook-registry.ts +129 -0
  65. package/.framework/functions/account-nodes.ts +111 -0
  66. package/.framework/functions/admin-data.ts +606 -0
  67. package/.framework/functions/ai-agents.ts +323 -0
  68. package/.framework/functions/api-keys.ts +376 -0
  69. package/.framework/functions/apps.ts +483 -0
  70. package/.framework/functions/auth.ts +196 -0
  71. package/.framework/functions/debug-auth.ts +107 -0
  72. package/.framework/functions/embeddings.ts +556 -0
  73. package/.framework/functions/integration-routes.ts +523 -0
  74. package/.framework/functions/integrations.ts +319 -0
  75. package/.framework/functions/item-progress.ts +272 -0
  76. package/.framework/functions/logs.ts +438 -0
  77. package/.framework/functions/observability.ts +275 -0
  78. package/.framework/functions/pipeline-executions.ts +494 -0
  79. package/.framework/functions/pipelines.ts +485 -0
  80. package/.framework/functions/prompt-configs.ts +339 -0
  81. package/.framework/functions/roles.ts +387 -0
  82. package/.framework/functions/system-cron.ts +742 -0
  83. package/.framework/functions/system.ts +323 -0
  84. package/.framework/functions/tests.ts +119 -0
  85. package/.framework/functions/timers.ts +357 -0
  86. package/.framework/functions/triggers.ts +563 -0
  87. package/.framework/functions/types.ts +604 -0
  88. package/.framework/migrations/000_foundation.sql +1256 -0
  89. package/.framework/migrations/001_seed.sql +92 -0
  90. package/.framework/migrations/002_seed_constraints.sql +13 -0
  91. package/.framework/migrations/003_auth_user_trigger.sql +59 -0
  92. package/.framework/src/App.tsx +126 -0
  93. package/.framework/src/apps/admin/index.tsx +173 -0
  94. package/.framework/src/components/AppWrapper.tsx +56 -0
  95. package/.framework/src/components/CustomAppLoader.tsx +116 -0
  96. package/.framework/src/components/admin/AdminListPage.tsx +151 -0
  97. package/.framework/src/components/admin/AdminSidebar.tsx +166 -0
  98. package/.framework/src/components/admin/AdminStatsCard.tsx +62 -0
  99. package/.framework/src/components/admin/SortableTableHeader.tsx +42 -0
  100. package/.framework/src/components/app-shell/GenericAppShell.tsx +181 -0
  101. package/.framework/src/components/app-shell/GenericDetailPage.tsx +200 -0
  102. package/.framework/src/components/app-shell/GenericListPage.tsx +116 -0
  103. package/.framework/src/components/app-sidebar.tsx +228 -0
  104. package/.framework/src/components/auth/ProtectedRoute.tsx +88 -0
  105. package/.framework/src/components/layout/AppShell.tsx +91 -0
  106. package/.framework/src/components/layout/Header.tsx +88 -0
  107. package/.framework/src/components/layout/Layout.tsx +95 -0
  108. package/.framework/src/components/layout/Sidebar.tsx +329 -0
  109. package/.framework/src/components/runtime/DataDetailHeader.tsx +77 -0
  110. package/.framework/src/components/runtime/DataDetailPage.tsx +171 -0
  111. package/.framework/src/components/runtime/DataFilters.tsx +91 -0
  112. package/.framework/src/components/runtime/DataHeader.tsx +68 -0
  113. package/.framework/src/components/runtime/DataListPage.tsx +124 -0
  114. package/.framework/src/components/runtime/DataStats.tsx +70 -0
  115. package/.framework/src/components/runtime/DataTable.tsx +174 -0
  116. package/.framework/src/components/runtime/SchemaDetailForm.tsx +134 -0
  117. package/.framework/src/components/runtime/index.ts +18 -0
  118. package/.framework/src/components/search-form.tsx +29 -0
  119. package/.framework/src/components/shared/AgentView.tsx +213 -0
  120. package/.framework/src/components/shared/FieldRenderer.tsx +478 -0
  121. package/.framework/src/components/shared/SchemaFields.tsx +226 -0
  122. package/.framework/src/components/ui/DataTable.tsx +343 -0
  123. package/.framework/src/components/ui/Form.tsx +281 -0
  124. package/.framework/src/components/ui/ItemCard.tsx +296 -0
  125. package/.framework/src/components/ui/ItemListView.tsx +308 -0
  126. package/.framework/src/components/ui/LoadingSpinner.tsx +52 -0
  127. package/.framework/src/components/ui/Modal.tsx +61 -0
  128. package/.framework/src/components/ui/RichTextEditor.tsx +210 -0
  129. package/.framework/src/components/ui/accordion.tsx +82 -0
  130. package/.framework/src/components/ui/alert-dialog.tsx +197 -0
  131. package/.framework/src/components/ui/alert.tsx +76 -0
  132. package/.framework/src/components/ui/aspect-ratio.tsx +11 -0
  133. package/.framework/src/components/ui/avatar.tsx +110 -0
  134. package/.framework/src/components/ui/badge.tsx +49 -0
  135. package/.framework/src/components/ui/breadcrumb.tsx +122 -0
  136. package/.framework/src/components/ui/button-group.tsx +83 -0
  137. package/.framework/src/components/ui/button.tsx +65 -0
  138. package/.framework/src/components/ui/calendar.tsx +222 -0
  139. package/.framework/src/components/ui/card.tsx +100 -0
  140. package/.framework/src/components/ui/carousel.tsx +240 -0
  141. package/.framework/src/components/ui/chart.tsx +373 -0
  142. package/.framework/src/components/ui/checkbox.tsx +31 -0
  143. package/.framework/src/components/ui/collapsible.tsx +33 -0
  144. package/.framework/src/components/ui/combobox.tsx +299 -0
  145. package/.framework/src/components/ui/command.tsx +193 -0
  146. package/.framework/src/components/ui/context-menu.tsx +261 -0
  147. package/.framework/src/components/ui/dialog.tsx +165 -0
  148. package/.framework/src/components/ui/direction.tsx +22 -0
  149. package/.framework/src/components/ui/drawer.tsx +132 -0
  150. package/.framework/src/components/ui/dropdown-menu.tsx +269 -0
  151. package/.framework/src/components/ui/empty.tsx +104 -0
  152. package/.framework/src/components/ui/field.tsx +238 -0
  153. package/.framework/src/components/ui/hover-card.tsx +42 -0
  154. package/.framework/src/components/ui/input-group.tsx +153 -0
  155. package/.framework/src/components/ui/input-otp.tsx +87 -0
  156. package/.framework/src/components/ui/input.tsx +19 -0
  157. package/.framework/src/components/ui/item.tsx +196 -0
  158. package/.framework/src/components/ui/kbd.tsx +26 -0
  159. package/.framework/src/components/ui/label.tsx +22 -0
  160. package/.framework/src/components/ui/menubar.tsx +277 -0
  161. package/.framework/src/components/ui/native-select.tsx +61 -0
  162. package/.framework/src/components/ui/navigation-menu.tsx +164 -0
  163. package/.framework/src/components/ui/pagination.tsx +129 -0
  164. package/.framework/src/components/ui/popover.tsx +87 -0
  165. package/.framework/src/components/ui/progress.tsx +31 -0
  166. package/.framework/src/components/ui/radio-group.tsx +42 -0
  167. package/.framework/src/components/ui/resizable.tsx +50 -0
  168. package/.framework/src/components/ui/scroll-area.tsx +53 -0
  169. package/.framework/src/components/ui/select.tsx +195 -0
  170. package/.framework/src/components/ui/separator.tsx +26 -0
  171. package/.framework/src/components/ui/sheet.tsx +145 -0
  172. package/.framework/src/components/ui/sidebar.tsx +706 -0
  173. package/.framework/src/components/ui/skeleton.tsx +13 -0
  174. package/.framework/src/components/ui/slider.tsx +59 -0
  175. package/.framework/src/components/ui/sonner.tsx +47 -0
  176. package/.framework/src/components/ui/spinner.tsx +10 -0
  177. package/.framework/src/components/ui/switch.tsx +33 -0
  178. package/.framework/src/components/ui/table-primitives.tsx +141 -0
  179. package/.framework/src/components/ui/table.tsx +114 -0
  180. package/.framework/src/components/ui/tabs.tsx +90 -0
  181. package/.framework/src/components/ui/textarea.tsx +18 -0
  182. package/.framework/src/components/ui/toggle-group.tsx +89 -0
  183. package/.framework/src/components/ui/toggle.tsx +45 -0
  184. package/.framework/src/components/ui/tooltip.tsx +57 -0
  185. package/.framework/src/contexts/AppContext.tsx +133 -0
  186. package/.framework/src/contexts/AuthContext.tsx +371 -0
  187. package/.framework/src/hooks/use-mobile.ts +19 -0
  188. package/.framework/src/hooks/useApi.ts +526 -0
  189. package/.framework/src/hooks/useApps.ts +114 -0
  190. package/.framework/src/hooks/useEntityList.ts +190 -0
  191. package/.framework/src/hooks/useEntityRecord.ts +308 -0
  192. package/.framework/src/hooks/useForm.ts +307 -0
  193. package/.framework/src/hooks/useListSchema.ts +264 -0
  194. package/.framework/src/hooks/useSchemaRecord.ts +223 -0
  195. package/.framework/src/index.css +128 -0
  196. package/.framework/src/lib/api.ts +156 -0
  197. package/.framework/src/lib/supabase.ts +94 -0
  198. package/.framework/src/lib/utils.ts +317 -0
  199. package/.framework/src/main.tsx +27 -0
  200. package/.framework/src/pages/DashboardPage.tsx +181 -0
  201. package/.framework/src/pages/NotFoundPage.tsx +39 -0
  202. package/.framework/src/pages/admin/AIAgentDetailPage.tsx +161 -0
  203. package/.framework/src/pages/admin/AIAgentsPage.tsx +318 -0
  204. package/.framework/src/pages/admin/APIKeyDetailPage.tsx +199 -0
  205. package/.framework/src/pages/admin/APIKeysPage.tsx +303 -0
  206. package/.framework/src/pages/admin/AlertsConfigPage.tsx +523 -0
  207. package/.framework/src/pages/admin/AppDetailPage.tsx +493 -0
  208. package/.framework/src/pages/admin/AppsPage.tsx +355 -0
  209. package/.framework/src/pages/admin/DesignedPage.tsx +491 -0
  210. package/.framework/src/pages/admin/EmbeddingDetailPage.tsx +534 -0
  211. package/.framework/src/pages/admin/EmbeddingsPage.tsx +424 -0
  212. package/.framework/src/pages/admin/ExtendedShadcnTestPage.tsx +176 -0
  213. package/.framework/src/pages/admin/IncrementalShadcnTestPage.tsx +109 -0
  214. package/.framework/src/pages/admin/IntegratedDashboard.tsx +402 -0
  215. package/.framework/src/pages/admin/IntegrationDetailPage.tsx +187 -0
  216. package/.framework/src/pages/admin/IntegrationsPage.tsx +301 -0
  217. package/.framework/src/pages/admin/LogsPage.tsx +283 -0
  218. package/.framework/src/pages/admin/MinimalShadcnTestPage.tsx +85 -0
  219. package/.framework/src/pages/admin/ObservabilityDashboard.tsx +470 -0
  220. package/.framework/src/pages/admin/PipelineDetailPage.tsx +183 -0
  221. package/.framework/src/pages/admin/PipelineExecutionsPage.tsx +279 -0
  222. package/.framework/src/pages/admin/PipelinesPage.tsx +390 -0
  223. package/.framework/src/pages/admin/PromptConfigDetailPage.tsx +299 -0
  224. package/.framework/src/pages/admin/PromptConfigsPage.tsx +292 -0
  225. package/.framework/src/pages/admin/ProperlyDesignedPage.tsx +434 -0
  226. package/.framework/src/pages/admin/RoleDetailPage.tsx +273 -0
  227. package/.framework/src/pages/admin/RolesPage.tsx +292 -0
  228. package/.framework/src/pages/admin/SelectTestPage.tsx +61 -0
  229. package/.framework/src/pages/admin/ShadcnTestPage.tsx +588 -0
  230. package/.framework/src/pages/admin/SimpleDashboard.tsx +387 -0
  231. package/.framework/src/pages/admin/TestRunDetailPage.tsx +172 -0
  232. package/.framework/src/pages/admin/TestingDashboard.tsx +257 -0
  233. package/.framework/src/pages/admin/TimerDetailPage.tsx +151 -0
  234. package/.framework/src/pages/admin/TimersPage.tsx +376 -0
  235. package/.framework/src/pages/admin/TriggerDetailPage.tsx +149 -0
  236. package/.framework/src/pages/admin/TriggersPage.tsx +381 -0
  237. package/.framework/src/pages/admin/TypeDetailPage.tsx +694 -0
  238. package/.framework/src/pages/admin/TypesPage.tsx +295 -0
  239. package/.framework/src/pages/auth/LoginPage.tsx +188 -0
  240. package/.framework/src/pages/auth/RegisterPage.tsx +163 -0
  241. package/.framework/src/pages/spine-framework/APIPage.tsx +17 -0
  242. package/.framework/src/pages/spine-framework/CLIPage.tsx +25 -0
  243. package/.framework/src/types/auth.ts +125 -0
  244. package/.framework/src/types/types.ts +407 -0
  245. package/STRUCTURE.md +150 -0
  246. package/config/components.json +25 -0
  247. package/config/deno.lock +108 -0
  248. package/config/package-lock.json +17183 -0
  249. package/config/postcss.config.cjs +10 -0
  250. package/config/tailwind.config.cjs +78 -0
  251. package/config/tsconfig.build.json +32 -0
  252. package/config/tsconfig.cli.json +18 -0
  253. package/config/tsconfig.json +41 -0
  254. package/config/tsconfig.node.json +17 -0
  255. package/config/tsconfig.node.tsbuildinfo +1 -0
  256. package/config/tsconfig.tsbuildinfo +1 -0
  257. package/config/typedoc.json +16 -0
  258. package/config/vite.config.d.ts +2 -0
  259. package/config/vite.config.ts +72 -0
  260. package/dist/cli/commands/agents.d.ts +39 -0
  261. package/dist/cli/commands/agents.d.ts.map +1 -0
  262. package/dist/cli/commands/auth.d.ts +36 -0
  263. package/dist/cli/commands/auth.d.ts.map +1 -0
  264. package/dist/cli/commands/create-app.d.ts +23 -0
  265. package/dist/cli/commands/create-app.d.ts.map +1 -0
  266. package/dist/cli/commands/dev.d.ts +39 -0
  267. package/dist/cli/commands/dev.d.ts.map +1 -0
  268. package/dist/cli/commands/doctor.d.ts +42 -0
  269. package/dist/cli/commands/doctor.d.ts.map +1 -0
  270. package/dist/cli/commands/generate.d.ts +36 -0
  271. package/dist/cli/commands/generate.d.ts.map +1 -0
  272. package/dist/cli/commands/init.d.ts +30 -0
  273. package/dist/cli/commands/init.d.ts.map +1 -0
  274. package/dist/cli/commands/install-app.d.ts +30 -0
  275. package/dist/cli/commands/install-app.d.ts.map +1 -0
  276. package/dist/cli/commands/items.d.ts +45 -0
  277. package/dist/cli/commands/items.d.ts.map +1 -0
  278. package/dist/cli/commands/migrations.d.ts +41 -0
  279. package/dist/cli/commands/migrations.d.ts.map +1 -0
  280. package/dist/cli/commands/pipelines.d.ts +40 -0
  281. package/dist/cli/commands/pipelines.d.ts.map +1 -0
  282. package/dist/cli/commands/status.d.ts +23 -0
  283. package/dist/cli/commands/status.d.ts.map +1 -0
  284. package/dist/cli/commands/system.d.ts +29 -0
  285. package/dist/cli/commands/system.d.ts.map +1 -0
  286. package/dist/cli/commands/test.d.ts +46 -0
  287. package/dist/cli/commands/test.d.ts.map +1 -0
  288. package/dist/cli/commands/uninstall-app.d.ts +23 -0
  289. package/dist/cli/commands/uninstall-app.d.ts.map +1 -0
  290. package/dist/cli/context.d.ts +88 -0
  291. package/dist/cli/context.d.ts.map +1 -0
  292. package/dist/cli/env-loader.d.ts +14 -0
  293. package/dist/cli/env-loader.d.ts.map +1 -0
  294. package/dist/cli/index.d.ts +41 -0
  295. package/dist/cli/index.d.ts.map +1 -0
  296. package/dist/functions/_shared/agent-runner.d.ts +156 -0
  297. package/dist/functions/_shared/agent-runner.d.ts.map +1 -0
  298. package/dist/functions/_shared/app-manifest.d.ts +68 -0
  299. package/dist/functions/_shared/app-manifest.d.ts.map +1 -0
  300. package/dist/functions/_shared/audit.d.ts +91 -0
  301. package/dist/functions/_shared/audit.d.ts.map +1 -0
  302. package/dist/functions/_shared/db.d.ts +125 -0
  303. package/dist/functions/_shared/db.d.ts.map +1 -0
  304. package/dist/functions/_shared/index.d.ts +298 -0
  305. package/dist/functions/_shared/index.d.ts.map +1 -0
  306. package/dist/functions/_shared/middleware.d.ts +315 -0
  307. package/dist/functions/_shared/middleware.d.ts.map +1 -0
  308. package/dist/functions/_shared/permissions.d.ts +626 -0
  309. package/dist/functions/_shared/permissions.d.ts.map +1 -0
  310. package/dist/functions/_shared/pipeline-runner.d.ts +124 -0
  311. package/dist/functions/_shared/pipeline-runner.d.ts.map +1 -0
  312. package/dist/functions/_shared/principal.d.ts +284 -0
  313. package/dist/functions/_shared/principal.d.ts.map +1 -0
  314. package/dist/functions/_shared/schema-utils.d.ts +181 -0
  315. package/dist/functions/_shared/schema-utils.d.ts.map +1 -0
  316. package/dist/functions/_shared/testing.d.ts +172 -0
  317. package/dist/functions/_shared/testing.d.ts.map +1 -0
  318. package/dist/functions/_shared/trigger-engine.d.ts +140 -0
  319. package/dist/functions/_shared/trigger-engine.d.ts.map +1 -0
  320. package/dist/functions/_shared/webhook-registration.d.ts +81 -0
  321. package/dist/functions/_shared/webhook-registration.d.ts.map +1 -0
  322. package/dist/functions/_shared/webhook-registry.d.ts +57 -0
  323. package/dist/functions/_shared/webhook-registry.d.ts.map +1 -0
  324. package/dist/functions/account-nodes.d.ts +48 -0
  325. package/dist/functions/account-nodes.d.ts.map +1 -0
  326. package/dist/functions/admin-data.d.ts +178 -0
  327. package/dist/functions/admin-data.d.ts.map +1 -0
  328. package/dist/functions/ai-agents.d.ts +125 -0
  329. package/dist/functions/ai-agents.d.ts.map +1 -0
  330. package/dist/functions/api-keys.d.ts +140 -0
  331. package/dist/functions/api-keys.d.ts.map +1 -0
  332. package/dist/functions/apps.d.ts +163 -0
  333. package/dist/functions/apps.d.ts.map +1 -0
  334. package/dist/functions/auth.d.ts +74 -0
  335. package/dist/functions/auth.d.ts.map +1 -0
  336. package/dist/functions/debug-auth.d.ts +33 -0
  337. package/dist/functions/debug-auth.d.ts.map +1 -0
  338. package/dist/functions/embeddings.d.ts +205 -0
  339. package/dist/functions/embeddings.d.ts.map +1 -0
  340. package/dist/functions/integration-routes.d.ts +45 -0
  341. package/dist/functions/integration-routes.d.ts.map +1 -0
  342. package/dist/functions/integrations.d.ts +124 -0
  343. package/dist/functions/integrations.d.ts.map +1 -0
  344. package/dist/functions/item-progress.d.ts +41 -0
  345. package/dist/functions/item-progress.d.ts.map +1 -0
  346. package/dist/functions/logs.d.ts +162 -0
  347. package/dist/functions/logs.d.ts.map +1 -0
  348. package/dist/functions/observability.d.ts +123 -0
  349. package/dist/functions/observability.d.ts.map +1 -0
  350. package/dist/functions/pipeline-executions.d.ts +190 -0
  351. package/dist/functions/pipeline-executions.d.ts.map +1 -0
  352. package/dist/functions/pipelines.d.ts +171 -0
  353. package/dist/functions/pipelines.d.ts.map +1 -0
  354. package/dist/functions/prompt-configs.d.ts +125 -0
  355. package/dist/functions/prompt-configs.d.ts.map +1 -0
  356. package/dist/functions/roles.d.ts +118 -0
  357. package/dist/functions/roles.d.ts.map +1 -0
  358. package/dist/functions/system-cron.d.ts +65 -0
  359. package/dist/functions/system-cron.d.ts.map +1 -0
  360. package/dist/functions/system.d.ts +29 -0
  361. package/dist/functions/system.d.ts.map +1 -0
  362. package/dist/functions/tests.d.ts +28 -0
  363. package/dist/functions/tests.d.ts.map +1 -0
  364. package/dist/functions/timers.d.ts +139 -0
  365. package/dist/functions/timers.d.ts.map +1 -0
  366. package/dist/functions/triggers.d.ts +203 -0
  367. package/dist/functions/triggers.d.ts.map +1 -0
  368. package/dist/functions/types.d.ts +151 -0
  369. package/dist/functions/types.d.ts.map +1 -0
  370. package/dist/src/types/types.d.ts +364 -0
  371. package/dist/src/types/types.d.ts.map +1 -0
  372. package/package.json +192 -0
  373. package/scripts/app-install-cli.ts +286 -0
  374. package/scripts/assemble-frontend.sh +79 -0
  375. package/scripts/assemble-functions.sh +62 -0
  376. package/scripts/assemble.sh +35 -0
  377. package/scripts/boundary-check.sh +106 -0
  378. package/scripts/build-manifest.sh +80 -0
  379. package/scripts/check-core-integrity.sh +82 -0
  380. package/scripts/ingest-chunks.cjs +202 -0
  381. package/scripts/kb-chunk-parser.cjs +312 -0
  382. package/scripts/kb-chunk-parser.ts +330 -0
  383. package/scripts/load-test-app-install.ts +484 -0
  384. package/scripts/netlify-dev-wrapper.sh +22 -0
  385. 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
+ })