orbital-command 0.2.0 → 0.3.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 (380) hide show
  1. package/bin/orbital.js +640 -37
  2. package/dist/assets/PrimitivesConfig-CrmQXYh4.js +32 -0
  3. package/dist/assets/QualityGates-BbasOsF3.js +21 -0
  4. package/dist/assets/SessionTimeline-CGeJsVvy.js +1 -0
  5. package/dist/assets/Settings-oiM496mc.js +12 -0
  6. package/dist/assets/SourceControl-B1fP2nJL.js +41 -0
  7. package/dist/assets/WorkflowVisualizer-CWLYf-f0.js +74 -0
  8. package/dist/assets/arrow-down-CPy85_J6.js +6 -0
  9. package/dist/assets/charts-DbDg0Psc.js +68 -0
  10. package/dist/assets/circle-x-Cwz6ZQDV.js +6 -0
  11. package/dist/assets/file-text-C46Xr65c.js +6 -0
  12. package/dist/assets/formatDistanceToNow-BMqsSP44.js +1 -0
  13. package/dist/assets/globe-Cn2yNZUD.js +6 -0
  14. package/dist/assets/index-Aj4sV8Al.css +1 -0
  15. package/dist/assets/index-Bc9dK3MW.js +354 -0
  16. package/dist/assets/key-OPaNTWJ5.js +6 -0
  17. package/dist/assets/minus-GMsbpKym.js +6 -0
  18. package/dist/assets/shield-DwAFkDYI.js +6 -0
  19. package/dist/assets/ui-BmsSg9jU.js +53 -0
  20. package/dist/assets/useWorkflowEditor-BJkTX_NR.js +16 -0
  21. package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
  22. package/dist/assets/zap-DfbUoOty.js +11 -0
  23. package/dist/favicon.svg +1 -0
  24. package/dist/index.html +6 -5
  25. package/dist/server/server/__tests__/data-routes.test.js +124 -0
  26. package/dist/server/server/__tests__/helpers/db.js +17 -0
  27. package/dist/server/server/__tests__/helpers/mock-emitter.js +8 -0
  28. package/dist/server/server/__tests__/scope-routes.test.js +137 -0
  29. package/dist/server/server/__tests__/sprint-routes.test.js +102 -0
  30. package/dist/server/server/__tests__/workflow-routes.test.js +107 -0
  31. package/dist/server/server/config-migrator.js +138 -0
  32. package/dist/server/server/config.js +17 -2
  33. package/dist/server/server/database.js +27 -12
  34. package/dist/server/server/global-config.js +143 -0
  35. package/dist/server/server/index.js +882 -252
  36. package/dist/server/server/init.js +579 -194
  37. package/dist/server/server/launch.js +29 -0
  38. package/dist/server/server/manifest-types.js +8 -0
  39. package/dist/server/server/manifest.js +454 -0
  40. package/dist/server/server/migrate-legacy.js +229 -0
  41. package/dist/server/server/parsers/event-parser.test.js +117 -0
  42. package/dist/server/server/parsers/scope-parser.js +74 -28
  43. package/dist/server/server/parsers/scope-parser.test.js +230 -0
  44. package/dist/server/server/project-context.js +255 -0
  45. package/dist/server/server/project-emitter.js +41 -0
  46. package/dist/server/server/project-manager.js +297 -0
  47. package/dist/server/server/routes/config-routes.js +1 -3
  48. package/dist/server/server/routes/data-routes.js +22 -110
  49. package/dist/server/server/routes/dispatch-routes.js +15 -9
  50. package/dist/server/server/routes/git-routes.js +74 -0
  51. package/dist/server/server/routes/manifest-routes.js +319 -0
  52. package/dist/server/server/routes/scope-routes.js +37 -23
  53. package/dist/server/server/routes/sync-routes.js +134 -0
  54. package/dist/server/server/routes/version-routes.js +1 -15
  55. package/dist/server/server/routes/workflow-routes.js +9 -3
  56. package/dist/server/server/schema.js +2 -0
  57. package/dist/server/server/services/batch-orchestrator.js +26 -16
  58. package/dist/server/server/services/claude-session-service.js +17 -14
  59. package/dist/server/server/services/deploy-service.test.js +119 -0
  60. package/dist/server/server/services/event-service.js +64 -1
  61. package/dist/server/server/services/event-service.test.js +191 -0
  62. package/dist/server/server/services/gate-service.test.js +105 -0
  63. package/dist/server/server/services/git-service.js +108 -4
  64. package/dist/server/server/services/github-service.js +110 -2
  65. package/dist/server/server/services/readiness-service.test.js +190 -0
  66. package/dist/server/server/services/scope-cache.js +5 -1
  67. package/dist/server/server/services/scope-cache.test.js +142 -0
  68. package/dist/server/server/services/scope-service.js +217 -126
  69. package/dist/server/server/services/scope-service.test.js +137 -0
  70. package/dist/server/server/services/sprint-orchestrator.js +7 -6
  71. package/dist/server/server/services/sprint-service.js +21 -1
  72. package/dist/server/server/services/sprint-service.test.js +238 -0
  73. package/dist/server/server/services/sync-service.js +434 -0
  74. package/dist/server/server/services/sync-types.js +2 -0
  75. package/dist/server/server/services/telemetry-service.js +143 -0
  76. package/dist/server/server/services/workflow-service.js +26 -5
  77. package/dist/server/server/services/workflow-service.test.js +159 -0
  78. package/dist/server/server/settings-sync.js +284 -0
  79. package/dist/server/server/update-planner.js +279 -0
  80. package/dist/server/server/utils/cc-hooks-parser.js +3 -0
  81. package/dist/server/server/utils/cc-hooks-parser.test.js +86 -0
  82. package/dist/server/server/utils/dispatch-utils.js +77 -20
  83. package/dist/server/server/utils/dispatch-utils.test.js +182 -0
  84. package/dist/server/server/utils/logger.js +37 -3
  85. package/dist/server/server/utils/package-info.js +30 -0
  86. package/dist/server/server/utils/route-helpers.js +10 -0
  87. package/dist/server/server/utils/terminal-launcher.js +79 -25
  88. package/dist/server/server/utils/worktree-manager.js +13 -4
  89. package/dist/server/server/validator.js +230 -0
  90. package/dist/server/server/watchers/global-watcher.js +63 -0
  91. package/dist/server/server/watchers/scope-watcher.js +27 -12
  92. package/dist/server/server/wizard/config-editor.js +237 -0
  93. package/dist/server/server/wizard/detect.js +96 -0
  94. package/dist/server/server/wizard/doctor.js +115 -0
  95. package/dist/server/server/wizard/index.js +155 -0
  96. package/dist/server/server/wizard/phases/confirm.js +39 -0
  97. package/dist/server/server/wizard/phases/project-setup.js +90 -0
  98. package/dist/server/server/wizard/phases/setup-wizard.js +66 -0
  99. package/dist/server/server/wizard/phases/welcome.js +35 -0
  100. package/dist/server/server/wizard/phases/workflow-setup.js +22 -0
  101. package/dist/server/server/wizard/types.js +29 -0
  102. package/dist/server/server/wizard/ui.js +74 -0
  103. package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
  104. package/dist/server/shared/default-workflow.json +65 -0
  105. package/dist/server/shared/onboarding-tour.test.js +81 -0
  106. package/dist/server/shared/project-colors.js +24 -0
  107. package/dist/server/shared/workflow-config.test.js +84 -0
  108. package/dist/server/shared/workflow-engine.test.js +302 -0
  109. package/dist/server/shared/workflow-normalizer.js +101 -0
  110. package/dist/server/shared/workflow-normalizer.test.js +100 -0
  111. package/dist/server/src/components/onboarding/tour-steps.js +84 -0
  112. package/package.json +20 -15
  113. package/schemas/orbital.config.schema.json +16 -1
  114. package/scripts/postinstall.js +55 -7
  115. package/server/__tests__/data-routes.test.ts +149 -0
  116. package/server/__tests__/helpers/db.ts +19 -0
  117. package/server/__tests__/helpers/mock-emitter.ts +10 -0
  118. package/server/__tests__/scope-routes.test.ts +157 -0
  119. package/server/__tests__/sprint-routes.test.ts +118 -0
  120. package/server/__tests__/workflow-routes.test.ts +120 -0
  121. package/server/config-migrator.ts +163 -0
  122. package/server/config.ts +26 -2
  123. package/server/database.ts +35 -18
  124. package/server/global-config.ts +200 -0
  125. package/server/index.ts +975 -287
  126. package/server/init.ts +625 -182
  127. package/server/launch.ts +32 -0
  128. package/server/manifest-types.ts +145 -0
  129. package/server/manifest.ts +494 -0
  130. package/server/migrate-legacy.ts +290 -0
  131. package/server/parsers/event-parser.test.ts +135 -0
  132. package/server/parsers/scope-parser.test.ts +270 -0
  133. package/server/parsers/scope-parser.ts +79 -31
  134. package/server/project-context.ts +309 -0
  135. package/server/project-emitter.ts +50 -0
  136. package/server/project-manager.ts +369 -0
  137. package/server/routes/config-routes.ts +3 -5
  138. package/server/routes/data-routes.ts +28 -141
  139. package/server/routes/dispatch-routes.ts +19 -11
  140. package/server/routes/git-routes.ts +77 -0
  141. package/server/routes/manifest-routes.ts +388 -0
  142. package/server/routes/scope-routes.ts +29 -25
  143. package/server/routes/sync-routes.ts +175 -0
  144. package/server/routes/version-routes.ts +1 -16
  145. package/server/routes/workflow-routes.ts +9 -3
  146. package/server/schema.ts +2 -0
  147. package/server/services/batch-orchestrator.ts +24 -16
  148. package/server/services/claude-session-service.ts +16 -14
  149. package/server/services/deploy-service.test.ts +145 -0
  150. package/server/services/deploy-service.ts +2 -2
  151. package/server/services/event-service.test.ts +242 -0
  152. package/server/services/event-service.ts +92 -3
  153. package/server/services/gate-service.test.ts +131 -0
  154. package/server/services/gate-service.ts +2 -2
  155. package/server/services/git-service.ts +137 -4
  156. package/server/services/github-service.ts +120 -2
  157. package/server/services/readiness-service.test.ts +217 -0
  158. package/server/services/scope-cache.test.ts +167 -0
  159. package/server/services/scope-cache.ts +4 -1
  160. package/server/services/scope-service.test.ts +169 -0
  161. package/server/services/scope-service.ts +220 -126
  162. package/server/services/sprint-orchestrator.ts +7 -7
  163. package/server/services/sprint-service.test.ts +271 -0
  164. package/server/services/sprint-service.ts +27 -3
  165. package/server/services/sync-service.ts +482 -0
  166. package/server/services/sync-types.ts +77 -0
  167. package/server/services/telemetry-service.ts +195 -0
  168. package/server/services/workflow-service.test.ts +190 -0
  169. package/server/services/workflow-service.ts +29 -9
  170. package/server/settings-sync.ts +359 -0
  171. package/server/update-planner.ts +346 -0
  172. package/server/utils/cc-hooks-parser.test.ts +96 -0
  173. package/server/utils/cc-hooks-parser.ts +4 -0
  174. package/server/utils/dispatch-utils.test.ts +245 -0
  175. package/server/utils/dispatch-utils.ts +97 -27
  176. package/server/utils/logger.ts +40 -3
  177. package/server/utils/package-info.ts +32 -0
  178. package/server/utils/route-helpers.ts +12 -0
  179. package/server/utils/terminal-launcher.ts +85 -25
  180. package/server/utils/worktree-manager.ts +9 -4
  181. package/server/validator.ts +270 -0
  182. package/server/watchers/global-watcher.ts +77 -0
  183. package/server/watchers/scope-watcher.ts +21 -9
  184. package/server/wizard/config-editor.ts +248 -0
  185. package/server/wizard/detect.ts +104 -0
  186. package/server/wizard/doctor.ts +114 -0
  187. package/server/wizard/index.ts +187 -0
  188. package/server/wizard/phases/confirm.ts +45 -0
  189. package/server/wizard/phases/project-setup.ts +106 -0
  190. package/server/wizard/phases/setup-wizard.ts +78 -0
  191. package/server/wizard/phases/welcome.ts +43 -0
  192. package/server/wizard/phases/workflow-setup.ts +28 -0
  193. package/server/wizard/types.ts +56 -0
  194. package/server/wizard/ui.ts +93 -0
  195. package/shared/__fixtures__/workflow-configs.ts +80 -0
  196. package/shared/default-workflow.json +65 -0
  197. package/shared/onboarding-tour.test.ts +94 -0
  198. package/shared/project-colors.ts +24 -0
  199. package/shared/workflow-config.test.ts +111 -0
  200. package/shared/workflow-config.ts +7 -0
  201. package/shared/workflow-engine.test.ts +388 -0
  202. package/shared/workflow-normalizer.test.ts +119 -0
  203. package/shared/workflow-normalizer.ts +118 -0
  204. package/templates/hooks/end-session.sh +3 -1
  205. package/templates/hooks/orbital-emit.sh +2 -2
  206. package/templates/hooks/orbital-report-deploy.sh +4 -4
  207. package/templates/hooks/orbital-report-gates.sh +4 -4
  208. package/templates/hooks/orbital-scope-update.sh +1 -1
  209. package/templates/hooks/scope-create-cleanup.sh +2 -2
  210. package/templates/hooks/scope-create-gate.sh +0 -1
  211. package/templates/hooks/scope-helpers.sh +18 -0
  212. package/templates/hooks/scope-prepare.sh +66 -11
  213. package/templates/migrations/renames.json +1 -0
  214. package/templates/orbital.config.json +7 -2
  215. package/templates/settings-hooks.json +1 -1
  216. package/templates/skills/git-commit/SKILL.md +9 -4
  217. package/templates/skills/git-dev/SKILL.md +8 -3
  218. package/templates/skills/git-main/SKILL.md +8 -2
  219. package/templates/skills/git-production/SKILL.md +6 -2
  220. package/templates/skills/git-staging/SKILL.md +8 -3
  221. package/templates/skills/scope-create/SKILL.md +17 -3
  222. package/templates/skills/scope-fix-review/SKILL.md +6 -3
  223. package/templates/skills/scope-implement/SKILL.md +4 -1
  224. package/templates/skills/scope-post-review/SKILL.md +63 -5
  225. package/templates/skills/scope-pre-review/SKILL.md +5 -2
  226. package/templates/skills/scope-verify/SKILL.md +5 -3
  227. package/templates/skills/test-code-review/SKILL.md +41 -33
  228. package/templates/skills/test-scaffold/SKILL.md +222 -0
  229. package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +0 -84
  230. package/dist/assets/charts-D__PA1zp.js +0 -72
  231. package/dist/assets/index-D1G6i0nS.css +0 -1
  232. package/dist/assets/index-DpItvKpf.js +0 -419
  233. package/dist/assets/ui-BvF022GT.js +0 -53
  234. package/index.html +0 -15
  235. package/postcss.config.js +0 -6
  236. package/src/App.tsx +0 -33
  237. package/src/components/AgentBadge.tsx +0 -40
  238. package/src/components/BatchPreflightModal.tsx +0 -115
  239. package/src/components/CardDisplayToggle.tsx +0 -74
  240. package/src/components/ColumnHeaderActions.tsx +0 -55
  241. package/src/components/ColumnMenu.tsx +0 -99
  242. package/src/components/DeployHistory.tsx +0 -141
  243. package/src/components/DispatchModal.tsx +0 -164
  244. package/src/components/DispatchPopover.tsx +0 -139
  245. package/src/components/DragOverlay.tsx +0 -25
  246. package/src/components/DriftSidebar.tsx +0 -140
  247. package/src/components/EnvironmentStrip.tsx +0 -88
  248. package/src/components/ErrorBoundary.tsx +0 -62
  249. package/src/components/FilterChip.tsx +0 -105
  250. package/src/components/GateIndicator.tsx +0 -33
  251. package/src/components/IdeaDetailModal.tsx +0 -190
  252. package/src/components/IdeaFormDialog.tsx +0 -113
  253. package/src/components/KanbanColumn.tsx +0 -201
  254. package/src/components/MarkdownRenderer.tsx +0 -114
  255. package/src/components/NeonGrid.tsx +0 -128
  256. package/src/components/PromotionQueue.tsx +0 -89
  257. package/src/components/ScopeCard.tsx +0 -234
  258. package/src/components/ScopeDetailModal.tsx +0 -255
  259. package/src/components/ScopeFilterBar.tsx +0 -152
  260. package/src/components/SearchInput.tsx +0 -102
  261. package/src/components/SessionPanel.tsx +0 -335
  262. package/src/components/SprintContainer.tsx +0 -303
  263. package/src/components/SprintDependencyDialog.tsx +0 -78
  264. package/src/components/SprintPreflightModal.tsx +0 -138
  265. package/src/components/StatusBar.tsx +0 -168
  266. package/src/components/SwimCell.tsx +0 -67
  267. package/src/components/SwimLaneRow.tsx +0 -94
  268. package/src/components/SwimlaneBoardView.tsx +0 -108
  269. package/src/components/VersionBadge.tsx +0 -139
  270. package/src/components/ViewModeSelector.tsx +0 -114
  271. package/src/components/config/AgentChip.tsx +0 -53
  272. package/src/components/config/AgentCreateDialog.tsx +0 -321
  273. package/src/components/config/AgentEditor.tsx +0 -175
  274. package/src/components/config/DirectoryTree.tsx +0 -582
  275. package/src/components/config/FileEditor.tsx +0 -550
  276. package/src/components/config/HookChip.tsx +0 -50
  277. package/src/components/config/StageCard.tsx +0 -198
  278. package/src/components/config/TransitionZone.tsx +0 -173
  279. package/src/components/config/UnifiedWorkflowPipeline.tsx +0 -216
  280. package/src/components/config/WorkflowPipeline.tsx +0 -161
  281. package/src/components/source-control/BranchList.tsx +0 -93
  282. package/src/components/source-control/BranchPanel.tsx +0 -105
  283. package/src/components/source-control/CommitLog.tsx +0 -100
  284. package/src/components/source-control/CommitRow.tsx +0 -47
  285. package/src/components/source-control/GitHubPanel.tsx +0 -110
  286. package/src/components/source-control/GitHubSetupGuide.tsx +0 -52
  287. package/src/components/source-control/GitOverviewBar.tsx +0 -101
  288. package/src/components/source-control/PullRequestList.tsx +0 -69
  289. package/src/components/source-control/WorktreeList.tsx +0 -80
  290. package/src/components/ui/badge.tsx +0 -41
  291. package/src/components/ui/button.tsx +0 -55
  292. package/src/components/ui/card.tsx +0 -78
  293. package/src/components/ui/dialog.tsx +0 -94
  294. package/src/components/ui/popover.tsx +0 -33
  295. package/src/components/ui/scroll-area.tsx +0 -54
  296. package/src/components/ui/separator.tsx +0 -28
  297. package/src/components/ui/tabs.tsx +0 -52
  298. package/src/components/ui/toggle-switch.tsx +0 -35
  299. package/src/components/ui/tooltip.tsx +0 -27
  300. package/src/components/workflow/AddEdgeDialog.tsx +0 -217
  301. package/src/components/workflow/AddListDialog.tsx +0 -201
  302. package/src/components/workflow/ChecklistEditor.tsx +0 -239
  303. package/src/components/workflow/CommandPrefixManager.tsx +0 -118
  304. package/src/components/workflow/ConfigSettingsPanel.tsx +0 -189
  305. package/src/components/workflow/DirectionSelector.tsx +0 -133
  306. package/src/components/workflow/DispatchConfigPanel.tsx +0 -180
  307. package/src/components/workflow/EdgeDetailPanel.tsx +0 -236
  308. package/src/components/workflow/EdgePropertyEditor.tsx +0 -251
  309. package/src/components/workflow/EditToolbar.tsx +0 -138
  310. package/src/components/workflow/HookDetailPanel.tsx +0 -250
  311. package/src/components/workflow/HookExecutionLog.tsx +0 -24
  312. package/src/components/workflow/HookSourceModal.tsx +0 -129
  313. package/src/components/workflow/HooksDashboard.tsx +0 -363
  314. package/src/components/workflow/ListPropertyEditor.tsx +0 -251
  315. package/src/components/workflow/MigrationPreviewDialog.tsx +0 -237
  316. package/src/components/workflow/MovementRulesPanel.tsx +0 -188
  317. package/src/components/workflow/NodeDetailPanel.tsx +0 -245
  318. package/src/components/workflow/PresetSelector.tsx +0 -414
  319. package/src/components/workflow/SkillCommandBuilder.tsx +0 -174
  320. package/src/components/workflow/WorkflowEdgeComponent.tsx +0 -145
  321. package/src/components/workflow/WorkflowNode.tsx +0 -147
  322. package/src/components/workflow/graphLayout.ts +0 -186
  323. package/src/components/workflow/mergeHooks.ts +0 -85
  324. package/src/components/workflow/useEditHistory.ts +0 -88
  325. package/src/components/workflow/useWorkflowEditor.ts +0 -262
  326. package/src/components/workflow/validateConfig.ts +0 -70
  327. package/src/hooks/useActiveDispatches.ts +0 -198
  328. package/src/hooks/useBoardSettings.ts +0 -170
  329. package/src/hooks/useCardDisplay.ts +0 -57
  330. package/src/hooks/useCcHooks.ts +0 -24
  331. package/src/hooks/useConfigTree.ts +0 -51
  332. package/src/hooks/useEnforcementRules.ts +0 -46
  333. package/src/hooks/useEvents.ts +0 -59
  334. package/src/hooks/useFileEditor.ts +0 -165
  335. package/src/hooks/useGates.ts +0 -57
  336. package/src/hooks/useIdeaActions.ts +0 -53
  337. package/src/hooks/useKanbanDnd.ts +0 -410
  338. package/src/hooks/useOrbitalConfig.ts +0 -54
  339. package/src/hooks/usePipeline.ts +0 -47
  340. package/src/hooks/usePipelineData.ts +0 -338
  341. package/src/hooks/useReconnect.ts +0 -25
  342. package/src/hooks/useScopeFilters.ts +0 -125
  343. package/src/hooks/useScopeSessions.ts +0 -44
  344. package/src/hooks/useScopes.ts +0 -67
  345. package/src/hooks/useSearch.ts +0 -67
  346. package/src/hooks/useSettings.tsx +0 -187
  347. package/src/hooks/useSocket.ts +0 -25
  348. package/src/hooks/useSourceControl.ts +0 -105
  349. package/src/hooks/useSprintPreflight.ts +0 -55
  350. package/src/hooks/useSprints.ts +0 -154
  351. package/src/hooks/useStatusBarHighlight.ts +0 -18
  352. package/src/hooks/useSwimlaneBoardSettings.ts +0 -104
  353. package/src/hooks/useTheme.ts +0 -9
  354. package/src/hooks/useTransitionReadiness.ts +0 -53
  355. package/src/hooks/useVersion.ts +0 -155
  356. package/src/hooks/useViolations.ts +0 -65
  357. package/src/hooks/useWorkflow.tsx +0 -125
  358. package/src/hooks/useZoomModifier.ts +0 -19
  359. package/src/index.css +0 -797
  360. package/src/layouts/DashboardLayout.tsx +0 -113
  361. package/src/lib/collisionDetection.ts +0 -20
  362. package/src/lib/scope-fields.ts +0 -61
  363. package/src/lib/swimlane.ts +0 -146
  364. package/src/lib/utils.ts +0 -15
  365. package/src/main.tsx +0 -19
  366. package/src/socket.ts +0 -11
  367. package/src/types/index.ts +0 -497
  368. package/src/views/AgentFeed.tsx +0 -339
  369. package/src/views/DeployPipeline.tsx +0 -59
  370. package/src/views/EnforcementView.tsx +0 -378
  371. package/src/views/PrimitivesConfig.tsx +0 -500
  372. package/src/views/QualityGates.tsx +0 -1012
  373. package/src/views/ScopeBoard.tsx +0 -454
  374. package/src/views/SessionTimeline.tsx +0 -516
  375. package/src/views/Settings.tsx +0 -183
  376. package/src/views/SourceControl.tsx +0 -95
  377. package/src/views/WorkflowVisualizer.tsx +0 -382
  378. package/tailwind.config.js +0 -161
  379. package/tsconfig.json +0 -25
  380. package/vite.config.ts +0 -38
@@ -0,0 +1,230 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import { normalizeStatus, inferStatusFromDir, setValidStatuses, parseScopeFile, parseAllScopes } from './scope-parser.js';
3
+ import fs from 'fs';
4
+ import os from 'os';
5
+ import path from 'path';
6
+ // ─── Pure function tests (no I/O) ──────────────────────────
7
+ describe('normalizeStatus()', () => {
8
+ it('maps "in-progress" to "implementing"', () => {
9
+ expect(normalizeStatus('in-progress')).toBe('implementing');
10
+ });
11
+ it('maps "in_progress" to "implementing"', () => {
12
+ expect(normalizeStatus('in_progress')).toBe('implementing');
13
+ });
14
+ it('maps "complete" to "completed"', () => {
15
+ expect(normalizeStatus('complete')).toBe('completed');
16
+ });
17
+ it('maps "done" to "production"', () => {
18
+ expect(normalizeStatus('done')).toBe('production');
19
+ });
20
+ it('maps "exploring" to "planning"', () => {
21
+ expect(normalizeStatus('exploring')).toBe('planning');
22
+ });
23
+ it('maps "blocked" to "backlog"', () => {
24
+ expect(normalizeStatus('blocked')).toBe('backlog');
25
+ });
26
+ it('maps "testing" to "review"', () => {
27
+ expect(normalizeStatus('testing')).toBe('review');
28
+ });
29
+ it('returns identity for already-valid statuses', () => {
30
+ expect(normalizeStatus('implementing')).toBe('implementing');
31
+ expect(normalizeStatus('icebox')).toBe('icebox');
32
+ expect(normalizeStatus('staging')).toBe('staging');
33
+ });
34
+ it('returns raw value for unknown statuses', () => {
35
+ expect(normalizeStatus('custom-status')).toBe('custom-status');
36
+ });
37
+ });
38
+ describe('setValidStatuses() + inferStatusFromDir()', () => {
39
+ beforeEach(() => {
40
+ setValidStatuses(['icebox', 'planning', 'backlog', 'implementing', 'review', 'completed', 'main']);
41
+ });
42
+ it('returns dir name when it is a valid status', () => {
43
+ expect(inferStatusFromDir('implementing')).toBe('implementing');
44
+ expect(inferStatusFromDir('backlog')).toBe('backlog');
45
+ });
46
+ it('returns "planning" for unknown dir name', () => {
47
+ expect(inferStatusFromDir('unknown-dir')).toBe('planning');
48
+ });
49
+ it('returns dir name as-is when validDirStatuses not yet set', () => {
50
+ // Reset by setting to a set that doesn't include our test value
51
+ setValidStatuses([]);
52
+ expect(inferStatusFromDir('anything')).toBe('planning');
53
+ });
54
+ });
55
+ // ─── File-based tests ───────────────────────────────────────
56
+ describe('parseScopeFile()', () => {
57
+ let tmpDir;
58
+ beforeEach(() => {
59
+ tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'scope-test-'));
60
+ setValidStatuses(['icebox', 'planning', 'backlog', 'implementing', 'review', 'completed', 'main']);
61
+ });
62
+ afterEach(() => {
63
+ fs.rmSync(tmpDir, { recursive: true, force: true });
64
+ });
65
+ function writeScopeFile(subDir, filename, content) {
66
+ const dir = path.join(tmpDir, subDir);
67
+ fs.mkdirSync(dir, { recursive: true });
68
+ const filePath = path.join(dir, filename);
69
+ fs.writeFileSync(filePath, content);
70
+ return filePath;
71
+ }
72
+ it('parses YAML frontmatter scope', () => {
73
+ const file = writeScopeFile('backlog', '001-test-scope.md', `---
74
+ title: Test Scope
75
+ status: backlog
76
+ priority: high
77
+ tags: [feature, backend]
78
+ blocked_by: [2, 3]
79
+ ---
80
+ # Test Scope
81
+
82
+ Some content here.
83
+ `);
84
+ const result = parseScopeFile(file);
85
+ expect(result).not.toBeNull();
86
+ expect(result.id).toBe(1);
87
+ expect(result.title).toBe('Test Scope');
88
+ expect(result.status).toBe('backlog');
89
+ expect(result.priority).toBe('high');
90
+ expect(result.tags).toEqual(['feature', 'backend']);
91
+ expect(result.blocked_by).toEqual([2, 3]);
92
+ expect(result.raw_content).toContain('Some content here.');
93
+ });
94
+ it('extracts ID with suffix encoding: a→1000+base', () => {
95
+ const file = writeScopeFile('implementing', '047a-variant.md', `---
96
+ title: Variant A
97
+ status: implementing
98
+ ---
99
+ Content
100
+ `);
101
+ const result = parseScopeFile(file);
102
+ expect(result.id).toBe(1047); // 1000 + 47
103
+ });
104
+ it('extracts ID with suffix encoding: X→9000+base', () => {
105
+ const file = writeScopeFile('review', '075X-experimental.md', `---
106
+ title: Experimental
107
+ status: review
108
+ ---
109
+ Content
110
+ `);
111
+ const result = parseScopeFile(file);
112
+ expect(result.id).toBe(9075); // 9000 + 75
113
+ });
114
+ it('generates negative hash ID for slug-only icebox files', () => {
115
+ const file = writeScopeFile('icebox', 'onboarding-flow.md', `---
116
+ title: Onboarding Flow
117
+ status: icebox
118
+ ---
119
+ An idea for onboarding.
120
+ `);
121
+ const result = parseScopeFile(file);
122
+ expect(result).not.toBeNull();
123
+ expect(result.id).toBeLessThan(0);
124
+ expect(result.slug).toBe('onboarding-flow');
125
+ });
126
+ it('parses markdown-only scope (no YAML)', () => {
127
+ const file = writeScopeFile('planning', '010-markdown-only.md', `# Scope 010: Markdown Feature
128
+ ## Priority: high
129
+ ## Estimated Effort: 3 days
130
+ ## Category: Backend
131
+
132
+ Implementation details...
133
+ `);
134
+ const result = parseScopeFile(file);
135
+ expect(result).not.toBeNull();
136
+ expect(result.id).toBe(10);
137
+ expect(result.title).toBe('Markdown Feature');
138
+ expect(result.priority).toBe('high');
139
+ expect(result.effort_estimate).toBe('3 days');
140
+ expect(result.category).toBe('Backend');
141
+ });
142
+ it('rejects invalid priority', () => {
143
+ const file = writeScopeFile('backlog', '002-test.md', `---
144
+ title: Bad Priority
145
+ status: backlog
146
+ priority: urgent
147
+ ---
148
+ Content
149
+ `);
150
+ const result = parseScopeFile(file);
151
+ expect(result.priority).toBeNull();
152
+ });
153
+ it('validates session keys', () => {
154
+ const file = writeScopeFile('implementing', '003-sessions.md', `---
155
+ title: With Sessions
156
+ status: implementing
157
+ sessions:
158
+ implementScope: ["session-1"]
159
+ invalidKey: ["session-2"]
160
+ ---
161
+ Content
162
+ `);
163
+ const result = parseScopeFile(file);
164
+ expect(result.sessions).toHaveProperty('implementScope');
165
+ expect(result.sessions).not.toHaveProperty('invalidKey');
166
+ });
167
+ it('handles ghost scopes', () => {
168
+ const file = writeScopeFile('icebox', '004-ghost.md', `---
169
+ title: Ghost Idea
170
+ status: icebox
171
+ ghost: true
172
+ ---
173
+ AI-generated idea
174
+ `);
175
+ const result = parseScopeFile(file);
176
+ expect(result.is_ghost).toBe(true);
177
+ });
178
+ it('returns null for template/non-scope files', () => {
179
+ const file = writeScopeFile('backlog', '_template.md', `---
180
+ title: Template
181
+ ---
182
+ Template content
183
+ `);
184
+ expect(parseScopeFile(file)).toBeNull();
185
+ });
186
+ });
187
+ describe('parseAllScopes()', () => {
188
+ let tmpDir;
189
+ beforeEach(() => {
190
+ tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'scopes-test-'));
191
+ setValidStatuses(['icebox', 'planning', 'backlog', 'implementing']);
192
+ });
193
+ afterEach(() => {
194
+ fs.rmSync(tmpDir, { recursive: true, force: true });
195
+ });
196
+ it('recursively scans and parses all .md files', () => {
197
+ const backlogDir = path.join(tmpDir, 'backlog');
198
+ const planningDir = path.join(tmpDir, 'planning');
199
+ fs.mkdirSync(backlogDir, { recursive: true });
200
+ fs.mkdirSync(planningDir, { recursive: true });
201
+ fs.writeFileSync(path.join(backlogDir, '001-first.md'), '---\ntitle: First\nstatus: backlog\n---\nContent\n');
202
+ fs.writeFileSync(path.join(planningDir, '002-second.md'), '---\ntitle: Second\nstatus: planning\n---\nContent\n');
203
+ const scopes = parseAllScopes(tmpDir);
204
+ expect(scopes).toHaveLength(2);
205
+ expect(scopes[0].id).toBe(1);
206
+ expect(scopes[1].id).toBe(2);
207
+ });
208
+ it('deduplicates by ID (first-seen wins)', () => {
209
+ const dir1 = path.join(tmpDir, 'backlog');
210
+ const dir2 = path.join(tmpDir, 'planning');
211
+ fs.mkdirSync(dir1, { recursive: true });
212
+ fs.mkdirSync(dir2, { recursive: true });
213
+ fs.writeFileSync(path.join(dir1, '001-original.md'), '---\ntitle: Original\nstatus: backlog\n---\n');
214
+ fs.writeFileSync(path.join(dir2, '001-duplicate.md'), '---\ntitle: Duplicate\nstatus: planning\n---\n');
215
+ const scopes = parseAllScopes(tmpDir);
216
+ expect(scopes).toHaveLength(1);
217
+ });
218
+ it('returns empty array for non-existent directory', () => {
219
+ expect(parseAllScopes('/tmp/nonexistent-scopes-dir')).toEqual([]);
220
+ });
221
+ it('returns sorted by ID', () => {
222
+ const dir = path.join(tmpDir, 'backlog');
223
+ fs.mkdirSync(dir, { recursive: true });
224
+ fs.writeFileSync(path.join(dir, '003-third.md'), '---\ntitle: Third\nstatus: backlog\n---\n');
225
+ fs.writeFileSync(path.join(dir, '001-first.md'), '---\ntitle: First\nstatus: backlog\n---\n');
226
+ fs.writeFileSync(path.join(dir, '002-second.md'), '---\ntitle: Second\nstatus: backlog\n---\n');
227
+ const scopes = parseAllScopes(tmpDir);
228
+ expect(scopes.map(s => s.id)).toEqual([1, 2, 3]);
229
+ });
230
+ });
@@ -0,0 +1,255 @@
1
+ import path from 'path';
2
+ import fs from 'fs';
3
+ import { fileURLToPath } from 'url';
4
+ import { openProjectDatabase } from './database.js';
5
+ import { loadConfig } from './config.js';
6
+ import { ScopeCache } from './services/scope-cache.js';
7
+ import { ScopeService } from './services/scope-service.js';
8
+ import { EventService } from './services/event-service.js';
9
+ import { GateService } from './services/gate-service.js';
10
+ import { DeployService } from './services/deploy-service.js';
11
+ import { SprintService } from './services/sprint-service.js';
12
+ import { SprintOrchestrator } from './services/sprint-orchestrator.js';
13
+ import { BatchOrchestrator } from './services/batch-orchestrator.js';
14
+ import { ReadinessService } from './services/readiness-service.js';
15
+ import { WorkflowService } from './services/workflow-service.js';
16
+ import { GitService } from './services/git-service.js';
17
+ import { GitHubService } from './services/github-service.js';
18
+ import { WorkflowEngine } from '../shared/workflow-engine.js';
19
+ import defaultWorkflow from '../shared/default-workflow.json' with { type: 'json' };
20
+ import { startScopeWatcher } from './watchers/scope-watcher.js';
21
+ import { startEventWatcher } from './watchers/event-watcher.js';
22
+ import { resolveStaleDispatches, resolveActiveDispatchesForScope, resolveDispatchesByPid, resolveDispatchesByDispatchId, linkPidToDispatch, tryAutoRevertAndClear } from './utils/dispatch-utils.js';
23
+ import { syncClaudeSessionsToDB } from './services/claude-session-service.js';
24
+ import { TelemetryService } from './services/telemetry-service.js';
25
+ import { ensureDynamicProfiles } from './utils/terminal-launcher.js';
26
+ import { createLogger } from './utils/logger.js';
27
+ const log = createLogger('project-context');
28
+ // ─── Factory ────────────────────────────────────────────────
29
+ /** Resolve the path to the bundled default workflow JSON. */
30
+ function getDefaultConfigPath() {
31
+ const __selfDir = path.dirname(fileURLToPath(import.meta.url));
32
+ return path.resolve(__selfDir, '../shared/default-workflow.json');
33
+ }
34
+ /**
35
+ * Create a fully wired ProjectContext for a single project.
36
+ *
37
+ * Create a fully wired context for a single project. Each ProjectContext has its own
38
+ * database, services, watchers, and intervals.
39
+ */
40
+ export async function createProjectContext(projectId, projectRoot, emitter) {
41
+ // Load project config
42
+ const config = loadConfig(projectRoot);
43
+ // Initialize database
44
+ const db = openProjectDatabase(config.dbDir);
45
+ // Initialize workflow engine
46
+ const workflowEngine = new WorkflowEngine(defaultWorkflow);
47
+ // Generate shell manifest for bash hooks
48
+ if (!fs.existsSync(config.configDir))
49
+ fs.mkdirSync(config.configDir, { recursive: true });
50
+ const manifestPath = path.join(config.configDir, 'workflow-manifest.sh');
51
+ fs.writeFileSync(manifestPath, workflowEngine.generateShellManifest(), 'utf-8');
52
+ // Ensure icebox directory exists
53
+ const iceboxDir = path.join(config.scopesDir, 'icebox');
54
+ if (!fs.existsSync(iceboxDir))
55
+ fs.mkdirSync(iceboxDir, { recursive: true });
56
+ // Initialize services
57
+ const scopeCache = new ScopeCache();
58
+ const scopeService = new ScopeService(scopeCache, emitter, config.scopesDir, workflowEngine);
59
+ const eventService = new EventService(db, emitter);
60
+ const gateService = new GateService(db, emitter);
61
+ const deployService = new DeployService(db, emitter);
62
+ const sprintService = new SprintService(db, emitter, scopeService);
63
+ const sprintOrchestrator = new SprintOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot);
64
+ const batchOrchestrator = new BatchOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot);
65
+ const readinessService = new ReadinessService(scopeService, gateService, workflowEngine, config.projectRoot);
66
+ const workflowService = new WorkflowService(config.configDir, workflowEngine, config.scopesDir, getDefaultConfigPath());
67
+ workflowService.setSocketServer(emitter);
68
+ // Ensure engine reflects active config (may differ from bundled default)
69
+ workflowEngine.reload(workflowService.getActive());
70
+ const gitService = new GitService(config.projectRoot, scopeCache);
71
+ const githubService = new GitHubService(config.projectRoot);
72
+ const telemetryService = new TelemetryService(db, config.telemetry, config.projectName, config.projectRoot);
73
+ // Wire active-group guard
74
+ scopeService.setActiveGroupCheck((scopeId) => sprintService.getActiveGroupForScope(scopeId));
75
+ // Wire event inference (Fix 8: diagnostic log lines match index.ts)
76
+ eventService.onIngest((eventType, scopeId, data) => {
77
+ if (eventType === 'SESSION_START' && typeof data.dispatch_id === 'string' && typeof data.pid === 'number') {
78
+ linkPidToDispatch(db, data.dispatch_id, data.pid);
79
+ log.debug('Linked PID to dispatch', { pid: data.pid, dispatch_id: data.dispatch_id });
80
+ return;
81
+ }
82
+ if (eventType === 'SCOPE_GATE_LIFTED' && scopeId != null) {
83
+ const id = Number(scopeId);
84
+ if (!isNaN(id) && id > 0) {
85
+ resolveActiveDispatchesForScope(db, emitter, id, 'completed');
86
+ log.debug('Resolved dispatches for scope gate lift', { scope_id: id });
87
+ }
88
+ return;
89
+ }
90
+ if (eventType === 'SESSION_END') {
91
+ const outcome = data.normal_exit === true ? 'completed' : 'abandoned';
92
+ let resolvedIds = [];
93
+ if (typeof data.dispatch_id === 'string') {
94
+ resolvedIds = resolveDispatchesByDispatchId(db, emitter, data.dispatch_id, outcome);
95
+ }
96
+ if (resolvedIds.length === 0 && typeof data.pid === 'number') {
97
+ resolvedIds = resolveDispatchesByPid(db, emitter, data.pid, outcome);
98
+ }
99
+ if (resolvedIds.length > 0)
100
+ log.info('Session resolved', { count: resolvedIds.length, outcome });
101
+ // For abandoned dispatches, immediately try auto-revert so the scope
102
+ // returns to its pre-dispatch status without requiring user interaction
103
+ if (outcome === 'abandoned') {
104
+ for (const eventId of resolvedIds) {
105
+ tryAutoRevertAndClear(db, emitter, scopeService, workflowEngine, eventId);
106
+ }
107
+ }
108
+ if (resolvedIds.length > 0)
109
+ batchOrchestrator.resolveStaleBatches();
110
+ return;
111
+ }
112
+ // Status inference
113
+ if (scopeId == null)
114
+ return;
115
+ const id = Number(scopeId);
116
+ if (isNaN(id) || id <= 0)
117
+ return;
118
+ const current = scopeService.getById(id);
119
+ if (current?.status === 'icebox')
120
+ return;
121
+ const currentStatus = current?.status ?? '';
122
+ const result = workflowEngine.inferStatus(eventType, currentStatus, data);
123
+ if (result === null)
124
+ return;
125
+ if (typeof result === 'object' && 'dispatchResolution' in result) {
126
+ resolveActiveDispatchesForScope(db, emitter, id, result.resolution);
127
+ return;
128
+ }
129
+ scopeService.updateStatus(id, result, 'event');
130
+ });
131
+ // Wire status change callbacks
132
+ scopeService.onStatusChange((scopeId, newStatus) => {
133
+ if (newStatus === 'dev')
134
+ sprintOrchestrator.onScopeReachedDev(scopeId);
135
+ batchOrchestrator.onScopeStatusChanged(scopeId, newStatus);
136
+ });
137
+ scopeService.onStatusChange((scopeId, newStatus) => {
138
+ if (workflowEngine.isTerminalStatus(newStatus)) {
139
+ resolveActiveDispatchesForScope(db, emitter, scopeId, 'completed');
140
+ }
141
+ });
142
+ // Load scopes from filesystem and reconcile directory mismatches
143
+ const scopeCount = scopeService.syncFromFilesystem();
144
+ const reconciled = scopeService.reconcileDirectories();
145
+ if (reconciled > 0)
146
+ log.info('Reconciled scope directories', { id: projectId, count: reconciled });
147
+ // Start watchers
148
+ const scopeWatcher = startScopeWatcher(config.scopesDir, scopeService);
149
+ const eventWatcher = startEventWatcher(config.eventsDir, eventService);
150
+ // Write iTerm2 dispatch profiles (Fix 2 + Fix 5: per-project prefix)
151
+ ensureDynamicProfiles(workflowEngine, config.terminal.profilePrefix);
152
+ // Recover active sprints/batches
153
+ await sprintOrchestrator.recoverActiveSprints();
154
+ await batchOrchestrator.recoverActiveBatches();
155
+ // Resolve stale batches on startup (Fix 6: catches stuck dispatches from previous runs)
156
+ const staleBatchesResolved = batchOrchestrator.resolveStaleBatches();
157
+ if (staleBatchesResolved > 0)
158
+ log.info('Resolved stale batches', { count: staleBatchesResolved });
159
+ // Resolve stale dispatches
160
+ resolveStaleDispatches(db, emitter, scopeService, workflowEngine);
161
+ // Initial session sync + legacy purge (Fix 7)
162
+ syncClaudeSessionsToDB(db, scopeService, config.projectRoot).then((count) => {
163
+ if (count > 0)
164
+ log.info('Synced sessions', { id: projectId, count });
165
+ const purged = db.prepare("DELETE FROM sessions WHERE action IS NULL AND id LIKE 'claude-%'").run();
166
+ if (purged.changes > 0)
167
+ log.info('Purged legacy session rows', { count: purged.changes });
168
+ if (telemetryService.enabled) {
169
+ telemetryService.uploadChangedSessions().catch(() => { });
170
+ }
171
+ }).catch(err => log.error('Session sync failed', { error: err.message }));
172
+ // Start periodic intervals
173
+ const intervals = [];
174
+ // Fix 11: periodic batch recovery (two-phase completion B-1)
175
+ intervals.push(setInterval(() => {
176
+ batchOrchestrator.recoverActiveBatches().catch(err => log.error('Batch recovery failed', { error: err.message }));
177
+ }, 30_000));
178
+ intervals.push(setInterval(() => {
179
+ batchOrchestrator.resolveStaleBatches();
180
+ }, 30_000));
181
+ intervals.push(setInterval(() => {
182
+ resolveStaleDispatches(db, emitter, scopeService, workflowEngine);
183
+ }, 30_000));
184
+ intervals.push(setInterval(async () => {
185
+ const count = await syncClaudeSessionsToDB(db, scopeService, config.projectRoot);
186
+ if (count > 0)
187
+ emitter.emit('session:updated', { type: 'resync', count });
188
+ if (telemetryService.enabled) {
189
+ telemetryService.uploadChangedSessions().catch(() => { });
190
+ }
191
+ }, 5 * 60_000));
192
+ let lastGitHash = '';
193
+ intervals.push(setInterval(async () => {
194
+ try {
195
+ const hash = await gitService.getStatusHash();
196
+ if (lastGitHash && hash !== lastGitHash) {
197
+ gitService.clearCache();
198
+ emitter.emit('git:status:changed');
199
+ }
200
+ lastGitHash = hash;
201
+ }
202
+ catch { /* ok */ }
203
+ }, 10_000));
204
+ log.info('Project ready', { id: projectId, scopes: scopeCount });
205
+ const ctx = {
206
+ id: projectId,
207
+ config,
208
+ db,
209
+ workflowEngine,
210
+ emitter,
211
+ scopeCache,
212
+ scopeService,
213
+ eventService,
214
+ gateService,
215
+ deployService,
216
+ sprintService,
217
+ sprintOrchestrator,
218
+ batchOrchestrator,
219
+ readinessService,
220
+ workflowService,
221
+ gitService,
222
+ githubService,
223
+ telemetryService,
224
+ scopeWatcher,
225
+ eventWatcher,
226
+ intervals,
227
+ status: 'active',
228
+ async shutdown() {
229
+ log.info('Shutting down project context', { id: projectId });
230
+ for (const interval of intervals)
231
+ clearInterval(interval);
232
+ intervals.length = 0;
233
+ try {
234
+ await scopeWatcher.close();
235
+ }
236
+ catch (e) {
237
+ log.error('Scope watcher close failed', { id: projectId, error: String(e) });
238
+ }
239
+ try {
240
+ await eventWatcher.close();
241
+ }
242
+ catch (e) {
243
+ log.error('Event watcher close failed', { id: projectId, error: String(e) });
244
+ }
245
+ try {
246
+ db.close();
247
+ }
248
+ catch (e) {
249
+ log.error('DB close failed', { id: projectId, error: String(e) });
250
+ }
251
+ ctx.status = 'offline';
252
+ },
253
+ };
254
+ return ctx;
255
+ }
@@ -0,0 +1,41 @@
1
+ /**
2
+ * A project-scoped Socket.io emitter.
3
+ *
4
+ * Services use this instead of the raw Socket.io Server so that events
5
+ * are automatically scoped to the correct project room. Events are emitted
6
+ * to both the project-specific room (`project:{id}`) and the aggregate
7
+ * room (`all-projects`) so that the All Projects dashboard view receives
8
+ * updates from every project.
9
+ *
10
+ * The emit() signature matches Socket.io's Server.emit() so existing
11
+ * service code requires only a type change (Server → ProjectEmitter).
12
+ */
13
+ export class ProjectEmitter {
14
+ io;
15
+ projectId;
16
+ constructor(io, projectId) {
17
+ this.io = io;
18
+ this.projectId = projectId;
19
+ }
20
+ /** Emit an event to this project's room and the all-projects room. */
21
+ emit(event, ...args) {
22
+ // Inject project_id into the first data argument if it's an object
23
+ const enrichedArgs = args.map((arg, i) => {
24
+ if (i === 0 && arg !== null && typeof arg === 'object' && !Array.isArray(arg)) {
25
+ return { ...arg, project_id: this.projectId };
26
+ }
27
+ return arg;
28
+ });
29
+ this.io.to(`project:${this.projectId}`).emit(event, ...enrichedArgs);
30
+ this.io.to('all-projects').emit(event, ...enrichedArgs);
31
+ return true;
32
+ }
33
+ /** Get the underlying Socket.io server (for operations that need it, e.g., connection handling). */
34
+ getServer() {
35
+ return this.io;
36
+ }
37
+ /** Get the project ID this emitter is scoped to. */
38
+ getProjectId() {
39
+ return this.projectId;
40
+ }
41
+ }