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