orbital-command 0.2.0 → 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 (431) hide show
  1. package/README.md +67 -42
  2. package/bin/commands/config.js +19 -0
  3. package/bin/commands/events.js +40 -0
  4. package/bin/commands/launch.js +126 -0
  5. package/bin/commands/manifest.js +283 -0
  6. package/bin/commands/registry.js +104 -0
  7. package/bin/commands/update.js +24 -0
  8. package/bin/lib/helpers.js +229 -0
  9. package/bin/orbital.js +147 -319
  10. package/dist/assets/Landing-CfQdHR0N.js +11 -0
  11. package/dist/assets/PrimitivesConfig-DThSipFy.js +32 -0
  12. package/dist/assets/QualityGates-B4kxM5UU.js +26 -0
  13. package/dist/assets/SessionTimeline-Bz1iZnmg.js +1 -0
  14. package/dist/assets/Settings-DLcZwbCT.js +12 -0
  15. package/dist/assets/SourceControl-BMNIz7Lt.js +36 -0
  16. package/dist/assets/WorkflowVisualizer-CxuSBOYu.js +69 -0
  17. package/dist/assets/arrow-down-DVPp6_qp.js +6 -0
  18. package/dist/assets/bot-NFaJBDn_.js +6 -0
  19. package/dist/assets/charts-LGLb8hyU.js +68 -0
  20. package/dist/assets/circle-x-IsFCkBZu.js +6 -0
  21. package/dist/assets/file-text-J1cebZXF.js +6 -0
  22. package/dist/assets/globe-WzeyHsUc.js +6 -0
  23. package/dist/assets/index-BdJ57EhC.css +1 -0
  24. package/dist/assets/index-o4ScMAuR.js +349 -0
  25. package/dist/assets/key-CKR8JJSj.js +6 -0
  26. package/dist/assets/minus-CHBsJyjp.js +6 -0
  27. package/dist/assets/radio-xqZaR-Uk.js +6 -0
  28. package/dist/assets/rocket-D_xvvNG6.js +6 -0
  29. package/dist/assets/shield-TdB1yv_a.js +6 -0
  30. package/dist/assets/ui-BmsSg9jU.js +53 -0
  31. package/dist/assets/useSocketListener-0L5yiN5i.js +1 -0
  32. package/dist/assets/useWorkflowEditor-CqeRWVQX.js +11 -0
  33. package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
  34. package/dist/assets/workflow-constants-Rw-GmgHZ.js +6 -0
  35. package/dist/assets/zap-C9wqYMpl.js +6 -0
  36. package/dist/favicon.svg +1 -0
  37. package/dist/index.html +6 -5
  38. package/dist/server/server/__tests__/data-routes.test.js +126 -0
  39. package/dist/server/server/__tests__/helpers/db.js +17 -0
  40. package/dist/server/server/__tests__/helpers/mock-emitter.js +8 -0
  41. package/dist/server/server/__tests__/scope-routes.test.js +138 -0
  42. package/dist/server/server/__tests__/sprint-routes.test.js +102 -0
  43. package/dist/server/server/__tests__/workflow-routes.test.js +107 -0
  44. package/dist/server/server/config-migrator.js +135 -0
  45. package/dist/server/server/config.js +51 -7
  46. package/dist/server/server/database.js +21 -28
  47. package/dist/server/server/global-config.js +143 -0
  48. package/dist/server/server/index.js +118 -276
  49. package/dist/server/server/init.js +243 -225
  50. package/dist/server/server/launch.js +29 -0
  51. package/dist/server/server/manifest-types.js +8 -0
  52. package/dist/server/server/manifest.js +454 -0
  53. package/dist/server/server/migrate-legacy.js +229 -0
  54. package/dist/server/server/parsers/event-parser.js +4 -1
  55. package/dist/server/server/parsers/event-parser.test.js +117 -0
  56. package/dist/server/server/parsers/scope-parser.js +74 -28
  57. package/dist/server/server/parsers/scope-parser.test.js +230 -0
  58. package/dist/server/server/project-context.js +265 -0
  59. package/dist/server/server/project-emitter.js +41 -0
  60. package/dist/server/server/project-manager.js +297 -0
  61. package/dist/server/server/routes/aggregate-routes.js +871 -0
  62. package/dist/server/server/routes/config-routes.js +41 -90
  63. package/dist/server/server/routes/data-routes.js +25 -123
  64. package/dist/server/server/routes/dispatch-routes.js +37 -15
  65. package/dist/server/server/routes/git-routes.js +74 -0
  66. package/dist/server/server/routes/manifest-routes.js +319 -0
  67. package/dist/server/server/routes/scope-routes.js +45 -28
  68. package/dist/server/server/routes/sync-routes.js +134 -0
  69. package/dist/server/server/routes/version-routes.js +1 -15
  70. package/dist/server/server/routes/workflow-routes.js +9 -3
  71. package/dist/server/server/schema.js +3 -0
  72. package/dist/server/server/services/batch-orchestrator.js +41 -17
  73. package/dist/server/server/services/claude-session-service.js +17 -14
  74. package/dist/server/server/services/config-service.js +10 -1
  75. package/dist/server/server/services/deploy-service.test.js +119 -0
  76. package/dist/server/server/services/event-service.js +64 -1
  77. package/dist/server/server/services/event-service.test.js +191 -0
  78. package/dist/server/server/services/gate-service.test.js +105 -0
  79. package/dist/server/server/services/git-service.js +108 -4
  80. package/dist/server/server/services/github-service.js +110 -2
  81. package/dist/server/server/services/readiness-service.test.js +190 -0
  82. package/dist/server/server/services/scope-cache.js +5 -1
  83. package/dist/server/server/services/scope-cache.test.js +142 -0
  84. package/dist/server/server/services/scope-service.js +222 -131
  85. package/dist/server/server/services/scope-service.test.js +137 -0
  86. package/dist/server/server/services/sprint-orchestrator.js +29 -15
  87. package/dist/server/server/services/sprint-service.js +23 -3
  88. package/dist/server/server/services/sprint-service.test.js +238 -0
  89. package/dist/server/server/services/sync-service.js +434 -0
  90. package/dist/server/server/services/sync-types.js +2 -0
  91. package/dist/server/server/services/workflow-service.js +26 -5
  92. package/dist/server/server/services/workflow-service.test.js +159 -0
  93. package/dist/server/server/settings-sync.js +284 -0
  94. package/dist/server/server/uninstall.js +195 -0
  95. package/dist/server/server/update-planner.js +279 -0
  96. package/dist/server/server/update.js +212 -0
  97. package/dist/server/server/utils/cc-hooks-parser.js +3 -0
  98. package/dist/server/server/utils/cc-hooks-parser.test.js +86 -0
  99. package/dist/server/server/utils/dispatch-utils.js +83 -24
  100. package/dist/server/server/utils/dispatch-utils.test.js +182 -0
  101. package/dist/server/server/utils/flag-builder.js +54 -0
  102. package/dist/server/server/utils/json-fields.js +14 -0
  103. package/dist/server/server/utils/json-fields.test.js +73 -0
  104. package/dist/server/server/utils/logger.js +37 -3
  105. package/dist/server/server/utils/package-info.js +30 -0
  106. package/dist/server/server/utils/route-helpers.js +47 -0
  107. package/dist/server/server/utils/route-helpers.test.js +115 -0
  108. package/dist/server/server/utils/terminal-launcher.js +79 -25
  109. package/dist/server/server/utils/worktree-manager.js +13 -4
  110. package/dist/server/server/validator.js +230 -0
  111. package/dist/server/server/watchers/event-watcher.js +28 -13
  112. package/dist/server/server/watchers/global-watcher.js +63 -0
  113. package/dist/server/server/watchers/scope-watcher.js +27 -12
  114. package/dist/server/server/wizard/config-editor.js +237 -0
  115. package/dist/server/server/wizard/detect.js +96 -0
  116. package/dist/server/server/wizard/doctor.js +115 -0
  117. package/dist/server/server/wizard/index.js +340 -0
  118. package/dist/server/server/wizard/phases/confirm.js +39 -0
  119. package/dist/server/server/wizard/phases/project-setup.js +90 -0
  120. package/dist/server/server/wizard/phases/setup-wizard.js +66 -0
  121. package/dist/server/server/wizard/phases/welcome.js +32 -0
  122. package/dist/server/server/wizard/phases/workflow-setup.js +22 -0
  123. package/dist/server/server/wizard/types.js +29 -0
  124. package/dist/server/server/wizard/ui.js +73 -0
  125. package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
  126. package/dist/server/shared/api-types.js +80 -1
  127. package/dist/server/shared/default-workflow.json +65 -0
  128. package/dist/server/shared/onboarding-tour.test.js +81 -0
  129. package/dist/server/shared/project-colors.js +24 -0
  130. package/dist/server/shared/workflow-config.test.js +84 -0
  131. package/dist/server/shared/workflow-engine.js +1 -1
  132. package/dist/server/shared/workflow-engine.test.js +302 -0
  133. package/dist/server/shared/workflow-normalizer.js +101 -0
  134. package/dist/server/shared/workflow-normalizer.test.js +100 -0
  135. package/dist/server/src/components/onboarding/tour-steps.js +84 -0
  136. package/package.json +34 -29
  137. package/schemas/orbital.config.schema.json +2 -5
  138. package/scripts/postinstall.js +18 -6
  139. package/scripts/release.sh +53 -0
  140. package/server/__tests__/data-routes.test.ts +151 -0
  141. package/server/__tests__/helpers/db.ts +19 -0
  142. package/server/__tests__/helpers/mock-emitter.ts +10 -0
  143. package/server/__tests__/scope-routes.test.ts +158 -0
  144. package/server/__tests__/sprint-routes.test.ts +118 -0
  145. package/server/__tests__/workflow-routes.test.ts +120 -0
  146. package/server/config-migrator.ts +160 -0
  147. package/server/config.ts +64 -12
  148. package/server/database.ts +22 -31
  149. package/server/global-config.ts +204 -0
  150. package/server/index.ts +139 -316
  151. package/server/init.ts +266 -234
  152. package/server/launch.ts +32 -0
  153. package/server/manifest-types.ts +145 -0
  154. package/server/manifest.ts +494 -0
  155. package/server/migrate-legacy.ts +290 -0
  156. package/server/parsers/event-parser.test.ts +135 -0
  157. package/server/parsers/event-parser.ts +4 -1
  158. package/server/parsers/scope-parser.test.ts +270 -0
  159. package/server/parsers/scope-parser.ts +79 -31
  160. package/server/project-context.ts +325 -0
  161. package/server/project-emitter.ts +50 -0
  162. package/server/project-manager.ts +368 -0
  163. package/server/routes/aggregate-routes.ts +968 -0
  164. package/server/routes/config-routes.ts +43 -85
  165. package/server/routes/data-routes.ts +34 -156
  166. package/server/routes/dispatch-routes.ts +46 -17
  167. package/server/routes/git-routes.ts +77 -0
  168. package/server/routes/manifest-routes.ts +388 -0
  169. package/server/routes/scope-routes.ts +39 -30
  170. package/server/routes/sync-routes.ts +175 -0
  171. package/server/routes/version-routes.ts +1 -16
  172. package/server/routes/workflow-routes.ts +9 -3
  173. package/server/schema.ts +3 -0
  174. package/server/services/batch-orchestrator.ts +41 -17
  175. package/server/services/claude-session-service.ts +16 -14
  176. package/server/services/config-service.ts +10 -1
  177. package/server/services/deploy-service.test.ts +145 -0
  178. package/server/services/deploy-service.ts +2 -2
  179. package/server/services/event-service.test.ts +242 -0
  180. package/server/services/event-service.ts +92 -3
  181. package/server/services/gate-service.test.ts +131 -0
  182. package/server/services/gate-service.ts +2 -2
  183. package/server/services/git-service.ts +137 -4
  184. package/server/services/github-service.ts +120 -2
  185. package/server/services/readiness-service.test.ts +217 -0
  186. package/server/services/scope-cache.test.ts +167 -0
  187. package/server/services/scope-cache.ts +4 -1
  188. package/server/services/scope-service.test.ts +169 -0
  189. package/server/services/scope-service.ts +224 -130
  190. package/server/services/sprint-orchestrator.ts +30 -15
  191. package/server/services/sprint-service.test.ts +271 -0
  192. package/server/services/sprint-service.ts +29 -5
  193. package/server/services/sync-service.ts +482 -0
  194. package/server/services/sync-types.ts +77 -0
  195. package/server/services/workflow-service.test.ts +190 -0
  196. package/server/services/workflow-service.ts +29 -9
  197. package/server/settings-sync.ts +359 -0
  198. package/server/uninstall.ts +214 -0
  199. package/server/update-planner.ts +346 -0
  200. package/server/update.ts +263 -0
  201. package/server/utils/cc-hooks-parser.test.ts +96 -0
  202. package/server/utils/cc-hooks-parser.ts +4 -0
  203. package/server/utils/dispatch-utils.test.ts +245 -0
  204. package/server/utils/dispatch-utils.ts +102 -30
  205. package/server/utils/flag-builder.ts +56 -0
  206. package/server/utils/json-fields.test.ts +83 -0
  207. package/server/utils/json-fields.ts +14 -0
  208. package/server/utils/logger.ts +40 -3
  209. package/server/utils/package-info.ts +32 -0
  210. package/server/utils/route-helpers.test.ts +144 -0
  211. package/server/utils/route-helpers.ts +50 -0
  212. package/server/utils/terminal-launcher.ts +85 -25
  213. package/server/utils/worktree-manager.ts +9 -4
  214. package/server/validator.ts +270 -0
  215. package/server/watchers/event-watcher.ts +24 -12
  216. package/server/watchers/global-watcher.ts +77 -0
  217. package/server/watchers/scope-watcher.ts +21 -9
  218. package/server/wizard/config-editor.ts +248 -0
  219. package/server/wizard/detect.ts +104 -0
  220. package/server/wizard/doctor.ts +114 -0
  221. package/server/wizard/index.ts +438 -0
  222. package/server/wizard/phases/confirm.ts +45 -0
  223. package/server/wizard/phases/project-setup.ts +106 -0
  224. package/server/wizard/phases/setup-wizard.ts +78 -0
  225. package/server/wizard/phases/welcome.ts +39 -0
  226. package/server/wizard/phases/workflow-setup.ts +28 -0
  227. package/server/wizard/types.ts +56 -0
  228. package/server/wizard/ui.ts +92 -0
  229. package/shared/__fixtures__/workflow-configs.ts +80 -0
  230. package/shared/api-types.ts +106 -0
  231. package/shared/onboarding-tour.test.ts +94 -0
  232. package/shared/project-colors.ts +24 -0
  233. package/shared/workflow-config.test.ts +111 -0
  234. package/shared/workflow-config.ts +7 -0
  235. package/shared/workflow-engine.test.ts +388 -0
  236. package/shared/workflow-engine.ts +1 -1
  237. package/shared/workflow-normalizer.test.ts +119 -0
  238. package/shared/workflow-normalizer.ts +118 -0
  239. package/templates/agents/QUICK-REFERENCE.md +1 -0
  240. package/templates/agents/README.md +1 -0
  241. package/templates/agents/SKILL-TRIGGERS.md +11 -0
  242. package/templates/agents/green-team/deep-dive.md +361 -0
  243. package/templates/hooks/end-session.sh +4 -1
  244. package/templates/hooks/init-session.sh +1 -0
  245. package/templates/hooks/orbital-emit.sh +2 -2
  246. package/templates/hooks/orbital-report-deploy.sh +4 -4
  247. package/templates/hooks/orbital-report-gates.sh +4 -4
  248. package/templates/hooks/orbital-scope-update.sh +1 -1
  249. package/templates/hooks/scope-commit-logger.sh +2 -2
  250. package/templates/hooks/scope-create-cleanup.sh +2 -2
  251. package/templates/hooks/scope-create-gate.sh +2 -5
  252. package/templates/hooks/scope-gate.sh +4 -6
  253. package/templates/hooks/scope-helpers.sh +28 -1
  254. package/templates/hooks/scope-lifecycle-gate.sh +14 -5
  255. package/templates/hooks/scope-prepare.sh +67 -12
  256. package/templates/hooks/scope-transition.sh +14 -6
  257. package/templates/hooks/time-tracker.sh +2 -5
  258. package/templates/migrations/renames.json +1 -0
  259. package/templates/orbital.config.json +8 -6
  260. package/{shared/default-workflow.json → templates/presets/default.json} +65 -0
  261. package/templates/presets/development.json +4 -4
  262. package/templates/presets/gitflow.json +7 -0
  263. package/templates/prompts/README.md +23 -0
  264. package/templates/prompts/deep-dive-audit.md +94 -0
  265. package/templates/quick/rules.md +56 -5
  266. package/templates/settings-hooks.json +1 -1
  267. package/templates/skills/git-commit/SKILL.md +27 -7
  268. package/templates/skills/git-dev/SKILL.md +13 -4
  269. package/templates/skills/git-main/SKILL.md +13 -3
  270. package/templates/skills/git-production/SKILL.md +9 -2
  271. package/templates/skills/git-staging/SKILL.md +11 -3
  272. package/templates/skills/scope-create/SKILL.md +17 -3
  273. package/templates/skills/scope-fix-review/SKILL.md +14 -7
  274. package/templates/skills/scope-implement/SKILL.md +15 -4
  275. package/templates/skills/scope-post-review/SKILL.md +77 -7
  276. package/templates/skills/scope-pre-review/SKILL.md +11 -4
  277. package/templates/skills/scope-verify/SKILL.md +5 -3
  278. package/templates/skills/test-code-review/SKILL.md +41 -33
  279. package/templates/skills/test-scaffold/SKILL.md +222 -0
  280. package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +0 -84
  281. package/dist/assets/charts-D__PA1zp.js +0 -72
  282. package/dist/assets/index-D1G6i0nS.css +0 -1
  283. package/dist/assets/index-DpItvKpf.js +0 -419
  284. package/dist/assets/ui-BvF022GT.js +0 -53
  285. package/index.html +0 -15
  286. package/postcss.config.js +0 -6
  287. package/src/App.tsx +0 -33
  288. package/src/components/AgentBadge.tsx +0 -40
  289. package/src/components/BatchPreflightModal.tsx +0 -115
  290. package/src/components/CardDisplayToggle.tsx +0 -74
  291. package/src/components/ColumnHeaderActions.tsx +0 -55
  292. package/src/components/ColumnMenu.tsx +0 -99
  293. package/src/components/DeployHistory.tsx +0 -141
  294. package/src/components/DispatchModal.tsx +0 -164
  295. package/src/components/DispatchPopover.tsx +0 -139
  296. package/src/components/DragOverlay.tsx +0 -25
  297. package/src/components/DriftSidebar.tsx +0 -140
  298. package/src/components/EnvironmentStrip.tsx +0 -88
  299. package/src/components/ErrorBoundary.tsx +0 -62
  300. package/src/components/FilterChip.tsx +0 -105
  301. package/src/components/GateIndicator.tsx +0 -33
  302. package/src/components/IdeaDetailModal.tsx +0 -190
  303. package/src/components/IdeaFormDialog.tsx +0 -113
  304. package/src/components/KanbanColumn.tsx +0 -201
  305. package/src/components/MarkdownRenderer.tsx +0 -114
  306. package/src/components/NeonGrid.tsx +0 -128
  307. package/src/components/PromotionQueue.tsx +0 -89
  308. package/src/components/ScopeCard.tsx +0 -234
  309. package/src/components/ScopeDetailModal.tsx +0 -255
  310. package/src/components/ScopeFilterBar.tsx +0 -152
  311. package/src/components/SearchInput.tsx +0 -102
  312. package/src/components/SessionPanel.tsx +0 -335
  313. package/src/components/SprintContainer.tsx +0 -303
  314. package/src/components/SprintDependencyDialog.tsx +0 -78
  315. package/src/components/SprintPreflightModal.tsx +0 -138
  316. package/src/components/StatusBar.tsx +0 -168
  317. package/src/components/SwimCell.tsx +0 -67
  318. package/src/components/SwimLaneRow.tsx +0 -94
  319. package/src/components/SwimlaneBoardView.tsx +0 -108
  320. package/src/components/VersionBadge.tsx +0 -139
  321. package/src/components/ViewModeSelector.tsx +0 -114
  322. package/src/components/config/AgentChip.tsx +0 -53
  323. package/src/components/config/AgentCreateDialog.tsx +0 -321
  324. package/src/components/config/AgentEditor.tsx +0 -175
  325. package/src/components/config/DirectoryTree.tsx +0 -582
  326. package/src/components/config/FileEditor.tsx +0 -550
  327. package/src/components/config/HookChip.tsx +0 -50
  328. package/src/components/config/StageCard.tsx +0 -198
  329. package/src/components/config/TransitionZone.tsx +0 -173
  330. package/src/components/config/UnifiedWorkflowPipeline.tsx +0 -216
  331. package/src/components/config/WorkflowPipeline.tsx +0 -161
  332. package/src/components/source-control/BranchList.tsx +0 -93
  333. package/src/components/source-control/BranchPanel.tsx +0 -105
  334. package/src/components/source-control/CommitLog.tsx +0 -100
  335. package/src/components/source-control/CommitRow.tsx +0 -47
  336. package/src/components/source-control/GitHubPanel.tsx +0 -110
  337. package/src/components/source-control/GitHubSetupGuide.tsx +0 -52
  338. package/src/components/source-control/GitOverviewBar.tsx +0 -101
  339. package/src/components/source-control/PullRequestList.tsx +0 -69
  340. package/src/components/source-control/WorktreeList.tsx +0 -80
  341. package/src/components/ui/badge.tsx +0 -41
  342. package/src/components/ui/button.tsx +0 -55
  343. package/src/components/ui/card.tsx +0 -78
  344. package/src/components/ui/dialog.tsx +0 -94
  345. package/src/components/ui/popover.tsx +0 -33
  346. package/src/components/ui/scroll-area.tsx +0 -54
  347. package/src/components/ui/separator.tsx +0 -28
  348. package/src/components/ui/tabs.tsx +0 -52
  349. package/src/components/ui/toggle-switch.tsx +0 -35
  350. package/src/components/ui/tooltip.tsx +0 -27
  351. package/src/components/workflow/AddEdgeDialog.tsx +0 -217
  352. package/src/components/workflow/AddListDialog.tsx +0 -201
  353. package/src/components/workflow/ChecklistEditor.tsx +0 -239
  354. package/src/components/workflow/CommandPrefixManager.tsx +0 -118
  355. package/src/components/workflow/ConfigSettingsPanel.tsx +0 -189
  356. package/src/components/workflow/DirectionSelector.tsx +0 -133
  357. package/src/components/workflow/DispatchConfigPanel.tsx +0 -180
  358. package/src/components/workflow/EdgeDetailPanel.tsx +0 -236
  359. package/src/components/workflow/EdgePropertyEditor.tsx +0 -251
  360. package/src/components/workflow/EditToolbar.tsx +0 -138
  361. package/src/components/workflow/HookDetailPanel.tsx +0 -250
  362. package/src/components/workflow/HookExecutionLog.tsx +0 -24
  363. package/src/components/workflow/HookSourceModal.tsx +0 -129
  364. package/src/components/workflow/HooksDashboard.tsx +0 -363
  365. package/src/components/workflow/ListPropertyEditor.tsx +0 -251
  366. package/src/components/workflow/MigrationPreviewDialog.tsx +0 -237
  367. package/src/components/workflow/MovementRulesPanel.tsx +0 -188
  368. package/src/components/workflow/NodeDetailPanel.tsx +0 -245
  369. package/src/components/workflow/PresetSelector.tsx +0 -414
  370. package/src/components/workflow/SkillCommandBuilder.tsx +0 -174
  371. package/src/components/workflow/WorkflowEdgeComponent.tsx +0 -145
  372. package/src/components/workflow/WorkflowNode.tsx +0 -147
  373. package/src/components/workflow/graphLayout.ts +0 -186
  374. package/src/components/workflow/mergeHooks.ts +0 -85
  375. package/src/components/workflow/useEditHistory.ts +0 -88
  376. package/src/components/workflow/useWorkflowEditor.ts +0 -262
  377. package/src/components/workflow/validateConfig.ts +0 -70
  378. package/src/hooks/useActiveDispatches.ts +0 -198
  379. package/src/hooks/useBoardSettings.ts +0 -170
  380. package/src/hooks/useCardDisplay.ts +0 -57
  381. package/src/hooks/useCcHooks.ts +0 -24
  382. package/src/hooks/useConfigTree.ts +0 -51
  383. package/src/hooks/useEnforcementRules.ts +0 -46
  384. package/src/hooks/useEvents.ts +0 -59
  385. package/src/hooks/useFileEditor.ts +0 -165
  386. package/src/hooks/useGates.ts +0 -57
  387. package/src/hooks/useIdeaActions.ts +0 -53
  388. package/src/hooks/useKanbanDnd.ts +0 -410
  389. package/src/hooks/useOrbitalConfig.ts +0 -54
  390. package/src/hooks/usePipeline.ts +0 -47
  391. package/src/hooks/usePipelineData.ts +0 -338
  392. package/src/hooks/useReconnect.ts +0 -25
  393. package/src/hooks/useScopeFilters.ts +0 -125
  394. package/src/hooks/useScopeSessions.ts +0 -44
  395. package/src/hooks/useScopes.ts +0 -67
  396. package/src/hooks/useSearch.ts +0 -67
  397. package/src/hooks/useSettings.tsx +0 -187
  398. package/src/hooks/useSocket.ts +0 -25
  399. package/src/hooks/useSourceControl.ts +0 -105
  400. package/src/hooks/useSprintPreflight.ts +0 -55
  401. package/src/hooks/useSprints.ts +0 -154
  402. package/src/hooks/useStatusBarHighlight.ts +0 -18
  403. package/src/hooks/useSwimlaneBoardSettings.ts +0 -104
  404. package/src/hooks/useTheme.ts +0 -9
  405. package/src/hooks/useTransitionReadiness.ts +0 -53
  406. package/src/hooks/useVersion.ts +0 -155
  407. package/src/hooks/useViolations.ts +0 -65
  408. package/src/hooks/useWorkflow.tsx +0 -125
  409. package/src/hooks/useZoomModifier.ts +0 -19
  410. package/src/index.css +0 -797
  411. package/src/layouts/DashboardLayout.tsx +0 -113
  412. package/src/lib/collisionDetection.ts +0 -20
  413. package/src/lib/scope-fields.ts +0 -61
  414. package/src/lib/swimlane.ts +0 -146
  415. package/src/lib/utils.ts +0 -15
  416. package/src/main.tsx +0 -19
  417. package/src/socket.ts +0 -11
  418. package/src/types/index.ts +0 -497
  419. package/src/views/AgentFeed.tsx +0 -339
  420. package/src/views/DeployPipeline.tsx +0 -59
  421. package/src/views/EnforcementView.tsx +0 -378
  422. package/src/views/PrimitivesConfig.tsx +0 -500
  423. package/src/views/QualityGates.tsx +0 -1012
  424. package/src/views/ScopeBoard.tsx +0 -454
  425. package/src/views/SessionTimeline.tsx +0 -516
  426. package/src/views/Settings.tsx +0 -183
  427. package/src/views/SourceControl.tsx +0 -95
  428. package/src/views/WorkflowVisualizer.tsx +0 -382
  429. package/tailwind.config.js +0 -161
  430. package/tsconfig.json +0 -25
  431. package/vite.config.ts +0 -38
@@ -0,0 +1,229 @@
1
+ /**
2
+ * Legacy migration — creates an orbital-manifest.json for projects
3
+ * that were initialized before the manifest system existed.
4
+ *
5
+ * Classifies existing files as synced/modified/user-owned by comparing
6
+ * their content hashes against the current template set.
7
+ */
8
+ import fs from 'fs';
9
+ import path from 'path';
10
+ import { loadManifest, saveManifest, createManifest, hashFile, buildTemplateInventory, templateFileRecord, userFileRecord, isSelfHosting, getSymlinkTarget, } from './manifest.js';
11
+ // ─── Constants ──────────────────────────────────────────────
12
+ /** Directories that contain managed primitives */
13
+ const MANAGED_DIRS = ['hooks', 'skills', 'agents'];
14
+ /** Gitignore entries that Orbital adds */
15
+ const GITIGNORE_ENTRIES = [
16
+ 'scopes/',
17
+ '.claude/orbital/',
18
+ '.claude/orbital-events/',
19
+ '.claude/config/workflow-manifest.sh',
20
+ ];
21
+ /**
22
+ * Detect whether a project needs legacy migration.
23
+ * Returns true if it has an orbital config but no manifest.
24
+ */
25
+ export function needsLegacyMigration(projectRoot) {
26
+ const claudeDir = path.join(projectRoot, '.claude');
27
+ const hasConfig = fs.existsSync(path.join(claudeDir, 'orbital.config.json'));
28
+ const hasManifest = loadManifest(projectRoot) !== null;
29
+ return hasConfig && !hasManifest;
30
+ }
31
+ /**
32
+ * Create a manifest for an existing project that was initialized
33
+ * before the manifest system. Classifies every file in .claude/
34
+ * managed directories.
35
+ */
36
+ export function migrateFromLegacy(projectRoot, templatesDir, packageVersion) {
37
+ const claudeDir = path.join(projectRoot, '.claude');
38
+ // If manifest already exists, skip
39
+ if (loadManifest(projectRoot) !== null) {
40
+ return { migrated: false, synced: 0, modified: 0, userOwned: 0, importedPins: 0 };
41
+ }
42
+ // Read existing config to get preset info
43
+ const preset = readPresetFromConfig(claudeDir);
44
+ // Read existing templateVersion from config, fall back to packageVersion
45
+ const configVersion = readTemplateVersion(claudeDir) || packageVersion;
46
+ const manifest = createManifest(configVersion, preset);
47
+ const selfHosting = isSelfHosting(projectRoot);
48
+ // Build template inventory (what files the current package ships)
49
+ const templateInventory = buildTemplateInventory(templatesDir);
50
+ let synced = 0;
51
+ let modified = 0;
52
+ let userOwned = 0;
53
+ // Walk managed directories and classify each file
54
+ for (const dir of MANAGED_DIRS) {
55
+ const dirPath = path.join(claudeDir, dir);
56
+ if (!fs.existsSync(dirPath))
57
+ continue;
58
+ walkDir(dirPath, dir, (relPath, absPath) => {
59
+ const templateHash = templateInventory.get(relPath);
60
+ if (templateHash) {
61
+ // File matches a known template path
62
+ if (selfHosting) {
63
+ const symlinkTarget = getSymlinkTarget(claudeDir, relPath);
64
+ if (symlinkTarget) {
65
+ manifest.files[relPath] = {
66
+ ...templateFileRecord(templateHash, symlinkTarget),
67
+ };
68
+ synced++;
69
+ return;
70
+ }
71
+ }
72
+ const fileHash = hashFile(absPath);
73
+ if (fileHash === templateHash) {
74
+ manifest.files[relPath] = templateFileRecord(templateHash);
75
+ synced++;
76
+ }
77
+ else {
78
+ manifest.files[relPath] = {
79
+ origin: 'template',
80
+ status: 'modified',
81
+ templateHash,
82
+ installedHash: fileHash,
83
+ };
84
+ modified++;
85
+ }
86
+ }
87
+ else {
88
+ // File doesn't match any template — user-created
89
+ const fileHash = hashFile(absPath);
90
+ manifest.files[relPath] = userFileRecord(fileHash);
91
+ userOwned++;
92
+ }
93
+ });
94
+ }
95
+ // Also classify non-managed template files (quick/, anti-patterns/, config/, etc.)
96
+ classifyNonManagedFiles(claudeDir, templateInventory, manifest, selfHosting);
97
+ // Import from orbital-sync.json if it exists (override → pinned)
98
+ const importedPins = importFromSyncManifest(claudeDir, manifest);
99
+ // Record gitignore entries
100
+ manifest.gitignoreEntries = [...GITIGNORE_ENTRIES];
101
+ // Record settings hooks checksum
102
+ const settingsHooksPath = path.join(templatesDir, 'settings-hooks.json');
103
+ if (fs.existsSync(settingsHooksPath)) {
104
+ manifest.settingsHooksChecksum = hashFile(settingsHooksPath);
105
+ }
106
+ saveManifest(projectRoot, manifest);
107
+ return { migrated: true, synced, modified, userOwned, importedPins };
108
+ }
109
+ // ─── Internal Helpers ───────────────────────────────────────
110
+ /** Recursively walk a directory, calling fn with relative and absolute paths. */
111
+ function walkDir(dirPath, prefix, fn) {
112
+ if (!fs.existsSync(dirPath))
113
+ return;
114
+ for (const entry of fs.readdirSync(dirPath, { withFileTypes: true })) {
115
+ if (entry.name.startsWith('.'))
116
+ continue;
117
+ const absPath = path.join(dirPath, entry.name);
118
+ const relPath = `${prefix}/${entry.name}`;
119
+ // Follow symlinks: use stat() to check if target is a directory
120
+ const stat = fs.statSync(absPath);
121
+ if (stat.isDirectory()) {
122
+ walkDir(absPath, relPath, fn);
123
+ }
124
+ else {
125
+ fn(relPath, absPath);
126
+ }
127
+ }
128
+ }
129
+ /** Classify template files outside hooks/skills/agents (quick/, anti-patterns/, config/, etc.) */
130
+ function classifyNonManagedFiles(claudeDir, templateInventory, manifest, selfHosting) {
131
+ const nonManagedDirs = ['quick', 'anti-patterns', 'config'];
132
+ for (const dir of nonManagedDirs) {
133
+ const dirPath = path.join(claudeDir, dir);
134
+ if (!fs.existsSync(dirPath))
135
+ continue;
136
+ walkDir(dirPath, dir, (relPath, absPath) => {
137
+ const templateHash = templateInventory.get(relPath);
138
+ if (!templateHash)
139
+ return; // Not a template file, skip
140
+ if (selfHosting) {
141
+ const symlinkTarget = getSymlinkTarget(claudeDir, relPath);
142
+ if (symlinkTarget) {
143
+ manifest.files[relPath] = templateFileRecord(templateHash, symlinkTarget);
144
+ return;
145
+ }
146
+ }
147
+ const fileHash = hashFile(absPath);
148
+ if (fileHash === templateHash) {
149
+ manifest.files[relPath] = templateFileRecord(templateHash);
150
+ }
151
+ else {
152
+ manifest.files[relPath] = {
153
+ origin: 'template',
154
+ status: 'modified',
155
+ templateHash,
156
+ installedHash: fileHash,
157
+ };
158
+ }
159
+ });
160
+ }
161
+ // Top-level template files
162
+ const topLevel = ['lessons-learned.md'];
163
+ for (const file of topLevel) {
164
+ const filePath = path.join(claudeDir, file);
165
+ const templateHash = templateInventory.get(file);
166
+ if (!templateHash || !fs.existsSync(filePath))
167
+ continue;
168
+ const fileHash = hashFile(filePath);
169
+ if (fileHash === templateHash) {
170
+ manifest.files[file] = templateFileRecord(templateHash);
171
+ }
172
+ else {
173
+ manifest.files[file] = {
174
+ origin: 'template',
175
+ status: 'modified',
176
+ templateHash,
177
+ installedHash: fileHash,
178
+ };
179
+ }
180
+ }
181
+ }
182
+ /** Import pin information from the legacy orbital-sync.json manifest. */
183
+ function importFromSyncManifest(claudeDir, manifest) {
184
+ const syncManifestPath = path.join(claudeDir, 'orbital-sync.json');
185
+ if (!fs.existsSync(syncManifestPath))
186
+ return 0;
187
+ let imported = 0;
188
+ try {
189
+ const raw = fs.readFileSync(syncManifestPath, 'utf-8');
190
+ const syncManifest = JSON.parse(raw);
191
+ for (const [relPath, record] of Object.entries(syncManifest.files)) {
192
+ if (record.mode === 'override' && manifest.files[relPath]) {
193
+ manifest.files[relPath].status = 'pinned';
194
+ manifest.files[relPath].pinnedAt = record.overriddenAt || new Date().toISOString();
195
+ manifest.files[relPath].pinnedReason = record.reason || 'Imported from orbital-sync.json override';
196
+ imported++;
197
+ }
198
+ }
199
+ }
200
+ catch {
201
+ // Malformed sync manifest — skip import
202
+ }
203
+ return imported;
204
+ }
205
+ /** Read the preset name from orbital.config.json, defaulting to "default". */
206
+ function readPresetFromConfig(claudeDir) {
207
+ try {
208
+ const workflowPath = path.join(claudeDir, 'config', 'workflow.json');
209
+ if (fs.existsSync(workflowPath)) {
210
+ const workflow = JSON.parse(fs.readFileSync(workflowPath, 'utf-8'));
211
+ if (workflow.name)
212
+ return workflow.name.toLowerCase().replace(/\s+/g, '-');
213
+ }
214
+ }
215
+ catch { /* fall through */ }
216
+ return 'default';
217
+ }
218
+ /** Read the templateVersion from orbital.config.json. */
219
+ function readTemplateVersion(claudeDir) {
220
+ try {
221
+ const configPath = path.join(claudeDir, 'orbital.config.json');
222
+ if (fs.existsSync(configPath)) {
223
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf-8'));
224
+ return config.templateVersion || null;
225
+ }
226
+ }
227
+ catch { /* fall through */ }
228
+ return null;
229
+ }
@@ -32,7 +32,10 @@ export function parseEventFile(filePath) {
32
32
  };
33
33
  }
34
34
  catch (err) {
35
- log.warn('Failed to parse event file', { file: filePath, error: err.message });
35
+ const code = err.code;
36
+ if (code !== 'ENOENT') {
37
+ log.warn('Failed to parse event file', { file: filePath, error: err.message });
38
+ }
36
39
  return null;
37
40
  }
38
41
  }
@@ -0,0 +1,117 @@
1
+ import { describe, it, expect, afterEach } from 'vitest';
2
+ import { parseEventFile } from './event-parser.js';
3
+ import fs from 'fs';
4
+ import os from 'os';
5
+ import path from 'path';
6
+ describe('parseEventFile', () => {
7
+ let tmpFile;
8
+ function writeEvent(data) {
9
+ tmpFile = path.join(os.tmpdir(), `test-event-${Date.now()}-${Math.random().toString(36).slice(2)}.json`);
10
+ fs.writeFileSync(tmpFile, JSON.stringify(data));
11
+ return tmpFile;
12
+ }
13
+ afterEach(() => {
14
+ if (tmpFile && fs.existsSync(tmpFile))
15
+ fs.unlinkSync(tmpFile);
16
+ });
17
+ // ─── Full format parsing ─────────────────────────────────
18
+ describe('full format (top-level fields)', () => {
19
+ it('extracts all fields correctly', () => {
20
+ const file = writeEvent({
21
+ id: 'evt-1', type: 'SCOPE_STATUS_CHANGED', scope_id: 42,
22
+ session_id: 'sess-abc', agent: 'architect',
23
+ data: { from: 'backlog', to: 'implementing' },
24
+ timestamp: '2026-04-08T10:00:00Z',
25
+ });
26
+ const result = parseEventFile(file);
27
+ expect(result).not.toBeNull();
28
+ expect(result.id).toBe('evt-1');
29
+ expect(result.type).toBe('SCOPE_STATUS_CHANGED');
30
+ expect(result.scope_id).toBe(42);
31
+ expect(result.session_id).toBe('sess-abc');
32
+ expect(result.agent).toBe('architect');
33
+ expect(result.data).toEqual({ from: 'backlog', to: 'implementing' });
34
+ });
35
+ it('preserves data payload as-is', () => {
36
+ const file = writeEvent({
37
+ id: 'evt-2', type: 'CUSTOM', timestamp: '2026-04-08T10:00:00Z',
38
+ data: { nested: { deep: true }, arr: [1, 2, 3] },
39
+ });
40
+ const result = parseEventFile(file);
41
+ expect(result.data).toEqual({ nested: { deep: true }, arr: [1, 2, 3] });
42
+ });
43
+ });
44
+ // ─── Minimal format parsing ──────────────────────────────
45
+ describe('minimal format (fields in data)', () => {
46
+ it('extracts scope_id and session_id from data', () => {
47
+ const file = writeEvent({
48
+ id: 'evt-3', type: 'AGENT_STARTED', timestamp: '2026-04-08T10:00:00Z',
49
+ data: { scope_id: 99, session_id: 'sess-xyz' },
50
+ });
51
+ const result = parseEventFile(file);
52
+ expect(result.scope_id).toBe(99);
53
+ expect(result.session_id).toBe('sess-xyz');
54
+ });
55
+ it('extracts agent from data.agents[0]', () => {
56
+ const file = writeEvent({
57
+ id: 'evt-4', type: 'AGENT_COMPLETED', timestamp: '2026-04-08T10:00:00Z',
58
+ data: { agents: ['attacker', 'chaos'] },
59
+ });
60
+ const result = parseEventFile(file);
61
+ expect(result.agent).toBe('attacker');
62
+ });
63
+ });
64
+ // ─── Field fallback priority ─────────────────────────────
65
+ describe('field fallback priority', () => {
66
+ it('top-level scope_id wins over data.scope_id', () => {
67
+ const file = writeEvent({
68
+ id: 'evt-5', type: 'TEST', timestamp: 'now',
69
+ scope_id: 10, data: { scope_id: 20 },
70
+ });
71
+ expect(parseEventFile(file).scope_id).toBe(10);
72
+ });
73
+ it('top-level agent wins over data.agent', () => {
74
+ const file = writeEvent({
75
+ id: 'evt-6', type: 'TEST', timestamp: 'now',
76
+ agent: 'top', data: { agent: 'nested' },
77
+ });
78
+ expect(parseEventFile(file).agent).toBe('top');
79
+ });
80
+ it('data.agent wins over data.agents[0]', () => {
81
+ const file = writeEvent({
82
+ id: 'evt-7', type: 'TEST', timestamp: 'now',
83
+ data: { agent: 'single', agents: ['array-first'] },
84
+ });
85
+ expect(parseEventFile(file).agent).toBe('single');
86
+ });
87
+ });
88
+ // ─── Edge cases ──────────────────────────────────────────
89
+ describe('edge cases', () => {
90
+ it('empty string scope_id falls through to data', () => {
91
+ const file = writeEvent({
92
+ id: 'evt-8', type: 'TEST', timestamp: 'now',
93
+ scope_id: '', data: { scope_id: 42 },
94
+ });
95
+ expect(parseEventFile(file).scope_id).toBe(42);
96
+ });
97
+ it('null scope_id with no data fallback returns null', () => {
98
+ const file = writeEvent({
99
+ id: 'evt-9', type: 'TEST', timestamp: 'now',
100
+ scope_id: null, data: {},
101
+ });
102
+ expect(parseEventFile(file).scope_id).toBeNull();
103
+ });
104
+ it('returns null for malformed JSON', () => {
105
+ tmpFile = path.join(os.tmpdir(), `test-bad-${Date.now()}.json`);
106
+ fs.writeFileSync(tmpFile, '{ not valid json');
107
+ expect(parseEventFile(tmpFile)).toBeNull();
108
+ });
109
+ it('returns null for missing required fields', () => {
110
+ const file = writeEvent({ id: 'evt-10' }); // missing type and timestamp
111
+ expect(parseEventFile(file)).toBeNull();
112
+ });
113
+ it('returns null for non-existent file', () => {
114
+ expect(parseEventFile('/tmp/nonexistent-event-file.json')).toBeNull();
115
+ });
116
+ });
117
+ });
@@ -45,31 +45,68 @@ export const STATUS_MAP = {
45
45
  export function normalizeStatus(raw) {
46
46
  return STATUS_MAP[raw] ?? raw;
47
47
  }
48
+ /** Generate a stable positive integer hash from a string (for synthetic icebox IDs) */
49
+ function slugHash(s) {
50
+ let hash = 5381;
51
+ for (let i = 0; i < s.length; i++) {
52
+ hash = ((hash << 5) + hash + s.charCodeAt(i)) >>> 0;
53
+ }
54
+ // Keep in range 10000-2147483647 to avoid collisions with real scope IDs and suffix-encoded IDs
55
+ return 10000 + (hash % 2137483647);
56
+ }
48
57
  /**
49
58
  * Parse a scope markdown file into structured data.
50
59
  * Handles both YAML frontmatter and plain markdown formats.
51
60
  */
52
61
  export function parseScopeFile(filePath) {
53
- const content = fs.readFileSync(filePath, 'utf-8');
54
- const fileName = path.basename(filePath, '.md');
55
- const dirName = path.basename(path.dirname(filePath));
56
- // Extract ID from filename pattern: NNN[suffix]-description.md
57
- // Suffixes (a-d, X) encode as thousands offset for unique DB keys
58
- const idMatch = fileName.match(/^(\d+)([a-dA-DxX])?/);
59
- const fileId = idMatch ? scopeFileId(parseInt(idMatch[1], 10), idMatch[2]) : 0;
60
- // Skip non-scope files
61
- if (fileId === 0 && !fileName.startsWith('0')) {
62
- // Files like _template.md, technical-debt.md, backlog_plan.md
63
- if (fileName.startsWith('_') || !idMatch)
62
+ try {
63
+ const content = fs.readFileSync(filePath, 'utf-8');
64
+ const fileName = path.basename(filePath, '.md');
65
+ const dirName = path.basename(path.dirname(filePath));
66
+ // Extract ID from filename pattern: NNN[suffix]-description.md
67
+ // Suffixes (a-d, X) encode as thousands offset for unique DB keys
68
+ const idMatch = fileName.match(/^(\d+)([a-dA-DxX])?/);
69
+ const fileId = idMatch ? scopeFileId(parseInt(idMatch[1], 10), idMatch[2]) : 0;
70
+ // Slug-only icebox files: no numeric prefix, e.g. "onboarding-flow.md"
71
+ const isSlugOnly = !idMatch && dirName === 'icebox';
72
+ // Skip non-scope files (but allow slug-only icebox files)
73
+ if (fileId === 0 && !fileName.startsWith('0') && !isSlugOnly) {
74
+ // Files like _template.md, technical-debt.md, backlog_plan.md
75
+ if (fileName.startsWith('_') || !idMatch)
76
+ return null;
77
+ }
78
+ // For slug-only icebox files, generate a stable negative ID for internal cache indexing.
79
+ // This avoids collisions with real scope IDs (positive) and between icebox items.
80
+ const effectiveId = isSlugOnly ? -slugHash(fileName) : fileId;
81
+ // Try YAML frontmatter first
82
+ let frontmatter;
83
+ let markdownBody;
84
+ try {
85
+ ({ data: frontmatter, content: markdownBody } = matter(content));
86
+ }
87
+ catch (err) {
88
+ log.warn('Skipping scope with malformed YAML frontmatter', { file: filePath, error: err.message });
64
89
  return null;
90
+ }
91
+ if (frontmatter && Object.keys(frontmatter).length > 0) {
92
+ const scope = parseFrontmatterScope(frontmatter, markdownBody, filePath, effectiveId, dirName);
93
+ // Populate slug for icebox items
94
+ if (isSlugOnly || dirName === 'icebox') {
95
+ scope.slug = isSlugOnly ? fileName : fileName.replace(/^\d+[a-dA-DxX]?-/, '');
96
+ }
97
+ return scope;
98
+ }
99
+ // Fallback: extract from markdown structure
100
+ const fallbackScope = parseMarkdownScope(content, filePath, effectiveId, dirName);
101
+ if (isSlugOnly || dirName === 'icebox') {
102
+ fallbackScope.slug = isSlugOnly ? fileName : fileName.replace(/^\d+[a-dA-DxX]?-/, '');
103
+ }
104
+ return fallbackScope;
65
105
  }
66
- // Try YAML frontmatter first
67
- const { data: frontmatter, content: markdownBody } = matter(content);
68
- if (frontmatter && Object.keys(frontmatter).length > 0) {
69
- return parseFrontmatterScope(frontmatter, markdownBody, filePath, fileId, dirName);
106
+ catch (err) {
107
+ log.warn('Failed to parse scope file', { file: filePath, error: err.message });
108
+ return null;
70
109
  }
71
- // Fallback: extract from markdown structure
72
- return parseMarkdownScope(content, filePath, fileId, dirName);
73
110
  }
74
111
  function parseFrontmatterScope(fm, body, filePath, fallbackId, dirName) {
75
112
  // Prefer filename-derived ID (includes suffix encoding) over frontmatter
@@ -92,6 +129,7 @@ function parseFrontmatterScope(fm, body, filePath, fallbackId, dirName) {
92
129
  raw_content: body.trim(),
93
130
  sessions: parseSessions(fm.sessions),
94
131
  is_ghost: fm.ghost === true,
132
+ favourite: fm.favourite === true,
95
133
  };
96
134
  }
97
135
  function parseMarkdownScope(content, filePath, id, dirName) {
@@ -126,6 +164,7 @@ function parseMarkdownScope(content, filePath, id, dirName) {
126
164
  raw_content: content,
127
165
  sessions: {},
128
166
  is_ghost: false,
167
+ favourite: false,
129
168
  };
130
169
  }
131
170
  /** Map filename suffix (a-d, X) to a thousands-digit offset for unique IDs */
@@ -145,7 +184,7 @@ let validDirStatuses = null;
145
184
  export function setValidStatuses(statuses) {
146
185
  validDirStatuses = new Set(statuses);
147
186
  }
148
- function inferStatusFromDir(dirName) {
187
+ export function inferStatusFromDir(dirName) {
149
188
  if (validDirStatuses) {
150
189
  return validDirStatuses.has(dirName) ? dirName : 'planning';
151
190
  }
@@ -164,25 +203,32 @@ export function parseAllScopes(scopesDir) {
164
203
  const entries = fs.readdirSync(dir, { withFileTypes: true });
165
204
  for (const entry of entries) {
166
205
  const fullPath = path.join(dir, entry.name);
167
- if (entry.isDirectory()) {
168
- scanDir(fullPath);
206
+ try {
207
+ if (entry.isDirectory()) {
208
+ scanDir(fullPath);
209
+ }
210
+ else if (entry.name.endsWith('.md') && !entry.name.startsWith('_')) {
211
+ const parsed = parseScopeFile(fullPath);
212
+ if (parsed)
213
+ scopes.push(parsed);
214
+ }
169
215
  }
170
- else if (entry.name.endsWith('.md') && !entry.name.startsWith('_')) {
171
- const parsed = parseScopeFile(fullPath);
172
- if (parsed)
173
- scopes.push(parsed);
216
+ catch (err) {
217
+ log.warn('Failed to process scope entry', { file: fullPath, error: err.message });
174
218
  }
175
219
  }
176
220
  }
177
221
  scanDir(scopesDir);
178
- // Detect ID collisions — last-write-wins but warn on stderr
222
+ // Detect ID collisions — first-seen wins, duplicates are dropped
179
223
  const seen = new Map();
180
- for (const scope of scopes) {
224
+ const deduped = scopes.filter(scope => {
181
225
  const existing = seen.get(scope.id);
182
226
  if (existing) {
183
227
  log.error('Scope ID collision — renumber one of them', { id: scope.id, existing, duplicate: scope.file_path });
228
+ return false;
184
229
  }
185
230
  seen.set(scope.id, scope.file_path);
186
- }
187
- return scopes.sort((a, b) => a.id - b.id);
231
+ return true;
232
+ });
233
+ return deduped.sort((a, b) => a.id - b.id);
188
234
  }