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.
- package/README.md +67 -42
- package/bin/commands/config.js +19 -0
- package/bin/commands/events.js +40 -0
- package/bin/commands/launch.js +126 -0
- package/bin/commands/manifest.js +283 -0
- package/bin/commands/registry.js +104 -0
- package/bin/commands/update.js +24 -0
- package/bin/lib/helpers.js +229 -0
- package/bin/orbital.js +147 -319
- package/dist/assets/Landing-CfQdHR0N.js +11 -0
- package/dist/assets/PrimitivesConfig-DThSipFy.js +32 -0
- package/dist/assets/QualityGates-B4kxM5UU.js +26 -0
- package/dist/assets/SessionTimeline-Bz1iZnmg.js +1 -0
- package/dist/assets/Settings-DLcZwbCT.js +12 -0
- package/dist/assets/SourceControl-BMNIz7Lt.js +36 -0
- package/dist/assets/WorkflowVisualizer-CxuSBOYu.js +69 -0
- package/dist/assets/arrow-down-DVPp6_qp.js +6 -0
- package/dist/assets/bot-NFaJBDn_.js +6 -0
- package/dist/assets/charts-LGLb8hyU.js +68 -0
- package/dist/assets/circle-x-IsFCkBZu.js +6 -0
- package/dist/assets/file-text-J1cebZXF.js +6 -0
- package/dist/assets/globe-WzeyHsUc.js +6 -0
- package/dist/assets/index-BdJ57EhC.css +1 -0
- package/dist/assets/index-o4ScMAuR.js +349 -0
- package/dist/assets/key-CKR8JJSj.js +6 -0
- package/dist/assets/minus-CHBsJyjp.js +6 -0
- package/dist/assets/radio-xqZaR-Uk.js +6 -0
- package/dist/assets/rocket-D_xvvNG6.js +6 -0
- package/dist/assets/shield-TdB1yv_a.js +6 -0
- package/dist/assets/ui-BmsSg9jU.js +53 -0
- package/dist/assets/useSocketListener-0L5yiN5i.js +1 -0
- package/dist/assets/useWorkflowEditor-CqeRWVQX.js +11 -0
- package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
- package/dist/assets/workflow-constants-Rw-GmgHZ.js +6 -0
- package/dist/assets/zap-C9wqYMpl.js +6 -0
- package/dist/favicon.svg +1 -0
- package/dist/index.html +6 -5
- package/dist/server/server/__tests__/data-routes.test.js +126 -0
- package/dist/server/server/__tests__/helpers/db.js +17 -0
- package/dist/server/server/__tests__/helpers/mock-emitter.js +8 -0
- package/dist/server/server/__tests__/scope-routes.test.js +138 -0
- package/dist/server/server/__tests__/sprint-routes.test.js +102 -0
- package/dist/server/server/__tests__/workflow-routes.test.js +107 -0
- package/dist/server/server/config-migrator.js +135 -0
- package/dist/server/server/config.js +51 -7
- package/dist/server/server/database.js +21 -28
- package/dist/server/server/global-config.js +143 -0
- package/dist/server/server/index.js +118 -276
- package/dist/server/server/init.js +243 -225
- package/dist/server/server/launch.js +29 -0
- package/dist/server/server/manifest-types.js +8 -0
- package/dist/server/server/manifest.js +454 -0
- package/dist/server/server/migrate-legacy.js +229 -0
- package/dist/server/server/parsers/event-parser.js +4 -1
- package/dist/server/server/parsers/event-parser.test.js +117 -0
- package/dist/server/server/parsers/scope-parser.js +74 -28
- package/dist/server/server/parsers/scope-parser.test.js +230 -0
- package/dist/server/server/project-context.js +265 -0
- package/dist/server/server/project-emitter.js +41 -0
- package/dist/server/server/project-manager.js +297 -0
- package/dist/server/server/routes/aggregate-routes.js +871 -0
- package/dist/server/server/routes/config-routes.js +41 -90
- package/dist/server/server/routes/data-routes.js +25 -123
- package/dist/server/server/routes/dispatch-routes.js +37 -15
- package/dist/server/server/routes/git-routes.js +74 -0
- package/dist/server/server/routes/manifest-routes.js +319 -0
- package/dist/server/server/routes/scope-routes.js +45 -28
- package/dist/server/server/routes/sync-routes.js +134 -0
- package/dist/server/server/routes/version-routes.js +1 -15
- package/dist/server/server/routes/workflow-routes.js +9 -3
- package/dist/server/server/schema.js +3 -0
- package/dist/server/server/services/batch-orchestrator.js +41 -17
- package/dist/server/server/services/claude-session-service.js +17 -14
- package/dist/server/server/services/config-service.js +10 -1
- package/dist/server/server/services/deploy-service.test.js +119 -0
- package/dist/server/server/services/event-service.js +64 -1
- package/dist/server/server/services/event-service.test.js +191 -0
- package/dist/server/server/services/gate-service.test.js +105 -0
- package/dist/server/server/services/git-service.js +108 -4
- package/dist/server/server/services/github-service.js +110 -2
- package/dist/server/server/services/readiness-service.test.js +190 -0
- package/dist/server/server/services/scope-cache.js +5 -1
- package/dist/server/server/services/scope-cache.test.js +142 -0
- package/dist/server/server/services/scope-service.js +222 -131
- package/dist/server/server/services/scope-service.test.js +137 -0
- package/dist/server/server/services/sprint-orchestrator.js +29 -15
- package/dist/server/server/services/sprint-service.js +23 -3
- package/dist/server/server/services/sprint-service.test.js +238 -0
- package/dist/server/server/services/sync-service.js +434 -0
- package/dist/server/server/services/sync-types.js +2 -0
- package/dist/server/server/services/workflow-service.js +26 -5
- package/dist/server/server/services/workflow-service.test.js +159 -0
- package/dist/server/server/settings-sync.js +284 -0
- package/dist/server/server/uninstall.js +195 -0
- package/dist/server/server/update-planner.js +279 -0
- package/dist/server/server/update.js +212 -0
- package/dist/server/server/utils/cc-hooks-parser.js +3 -0
- package/dist/server/server/utils/cc-hooks-parser.test.js +86 -0
- package/dist/server/server/utils/dispatch-utils.js +83 -24
- package/dist/server/server/utils/dispatch-utils.test.js +182 -0
- package/dist/server/server/utils/flag-builder.js +54 -0
- package/dist/server/server/utils/json-fields.js +14 -0
- package/dist/server/server/utils/json-fields.test.js +73 -0
- package/dist/server/server/utils/logger.js +37 -3
- package/dist/server/server/utils/package-info.js +30 -0
- package/dist/server/server/utils/route-helpers.js +47 -0
- package/dist/server/server/utils/route-helpers.test.js +115 -0
- package/dist/server/server/utils/terminal-launcher.js +79 -25
- package/dist/server/server/utils/worktree-manager.js +13 -4
- package/dist/server/server/validator.js +230 -0
- package/dist/server/server/watchers/event-watcher.js +28 -13
- package/dist/server/server/watchers/global-watcher.js +63 -0
- package/dist/server/server/watchers/scope-watcher.js +27 -12
- package/dist/server/server/wizard/config-editor.js +237 -0
- package/dist/server/server/wizard/detect.js +96 -0
- package/dist/server/server/wizard/doctor.js +115 -0
- package/dist/server/server/wizard/index.js +340 -0
- package/dist/server/server/wizard/phases/confirm.js +39 -0
- package/dist/server/server/wizard/phases/project-setup.js +90 -0
- package/dist/server/server/wizard/phases/setup-wizard.js +66 -0
- package/dist/server/server/wizard/phases/welcome.js +32 -0
- package/dist/server/server/wizard/phases/workflow-setup.js +22 -0
- package/dist/server/server/wizard/types.js +29 -0
- package/dist/server/server/wizard/ui.js +73 -0
- package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
- package/dist/server/shared/api-types.js +80 -1
- package/dist/server/shared/default-workflow.json +65 -0
- package/dist/server/shared/onboarding-tour.test.js +81 -0
- package/dist/server/shared/project-colors.js +24 -0
- package/dist/server/shared/workflow-config.test.js +84 -0
- package/dist/server/shared/workflow-engine.js +1 -1
- package/dist/server/shared/workflow-engine.test.js +302 -0
- package/dist/server/shared/workflow-normalizer.js +101 -0
- package/dist/server/shared/workflow-normalizer.test.js +100 -0
- package/dist/server/src/components/onboarding/tour-steps.js +84 -0
- package/package.json +34 -29
- package/schemas/orbital.config.schema.json +2 -5
- package/scripts/postinstall.js +18 -6
- package/scripts/release.sh +53 -0
- package/server/__tests__/data-routes.test.ts +151 -0
- package/server/__tests__/helpers/db.ts +19 -0
- package/server/__tests__/helpers/mock-emitter.ts +10 -0
- package/server/__tests__/scope-routes.test.ts +158 -0
- package/server/__tests__/sprint-routes.test.ts +118 -0
- package/server/__tests__/workflow-routes.test.ts +120 -0
- package/server/config-migrator.ts +160 -0
- package/server/config.ts +64 -12
- package/server/database.ts +22 -31
- package/server/global-config.ts +204 -0
- package/server/index.ts +139 -316
- package/server/init.ts +266 -234
- package/server/launch.ts +32 -0
- package/server/manifest-types.ts +145 -0
- package/server/manifest.ts +494 -0
- package/server/migrate-legacy.ts +290 -0
- package/server/parsers/event-parser.test.ts +135 -0
- package/server/parsers/event-parser.ts +4 -1
- package/server/parsers/scope-parser.test.ts +270 -0
- package/server/parsers/scope-parser.ts +79 -31
- package/server/project-context.ts +325 -0
- package/server/project-emitter.ts +50 -0
- package/server/project-manager.ts +368 -0
- package/server/routes/aggregate-routes.ts +968 -0
- package/server/routes/config-routes.ts +43 -85
- package/server/routes/data-routes.ts +34 -156
- package/server/routes/dispatch-routes.ts +46 -17
- package/server/routes/git-routes.ts +77 -0
- package/server/routes/manifest-routes.ts +388 -0
- package/server/routes/scope-routes.ts +39 -30
- package/server/routes/sync-routes.ts +175 -0
- package/server/routes/version-routes.ts +1 -16
- package/server/routes/workflow-routes.ts +9 -3
- package/server/schema.ts +3 -0
- package/server/services/batch-orchestrator.ts +41 -17
- package/server/services/claude-session-service.ts +16 -14
- package/server/services/config-service.ts +10 -1
- package/server/services/deploy-service.test.ts +145 -0
- package/server/services/deploy-service.ts +2 -2
- package/server/services/event-service.test.ts +242 -0
- package/server/services/event-service.ts +92 -3
- package/server/services/gate-service.test.ts +131 -0
- package/server/services/gate-service.ts +2 -2
- package/server/services/git-service.ts +137 -4
- package/server/services/github-service.ts +120 -2
- package/server/services/readiness-service.test.ts +217 -0
- package/server/services/scope-cache.test.ts +167 -0
- package/server/services/scope-cache.ts +4 -1
- package/server/services/scope-service.test.ts +169 -0
- package/server/services/scope-service.ts +224 -130
- package/server/services/sprint-orchestrator.ts +30 -15
- package/server/services/sprint-service.test.ts +271 -0
- package/server/services/sprint-service.ts +29 -5
- package/server/services/sync-service.ts +482 -0
- package/server/services/sync-types.ts +77 -0
- package/server/services/workflow-service.test.ts +190 -0
- package/server/services/workflow-service.ts +29 -9
- package/server/settings-sync.ts +359 -0
- package/server/uninstall.ts +214 -0
- package/server/update-planner.ts +346 -0
- package/server/update.ts +263 -0
- package/server/utils/cc-hooks-parser.test.ts +96 -0
- package/server/utils/cc-hooks-parser.ts +4 -0
- package/server/utils/dispatch-utils.test.ts +245 -0
- package/server/utils/dispatch-utils.ts +102 -30
- package/server/utils/flag-builder.ts +56 -0
- package/server/utils/json-fields.test.ts +83 -0
- package/server/utils/json-fields.ts +14 -0
- package/server/utils/logger.ts +40 -3
- package/server/utils/package-info.ts +32 -0
- package/server/utils/route-helpers.test.ts +144 -0
- package/server/utils/route-helpers.ts +50 -0
- package/server/utils/terminal-launcher.ts +85 -25
- package/server/utils/worktree-manager.ts +9 -4
- package/server/validator.ts +270 -0
- package/server/watchers/event-watcher.ts +24 -12
- package/server/watchers/global-watcher.ts +77 -0
- package/server/watchers/scope-watcher.ts +21 -9
- package/server/wizard/config-editor.ts +248 -0
- package/server/wizard/detect.ts +104 -0
- package/server/wizard/doctor.ts +114 -0
- package/server/wizard/index.ts +438 -0
- package/server/wizard/phases/confirm.ts +45 -0
- package/server/wizard/phases/project-setup.ts +106 -0
- package/server/wizard/phases/setup-wizard.ts +78 -0
- package/server/wizard/phases/welcome.ts +39 -0
- package/server/wizard/phases/workflow-setup.ts +28 -0
- package/server/wizard/types.ts +56 -0
- package/server/wizard/ui.ts +92 -0
- package/shared/__fixtures__/workflow-configs.ts +80 -0
- package/shared/api-types.ts +106 -0
- package/shared/onboarding-tour.test.ts +94 -0
- package/shared/project-colors.ts +24 -0
- package/shared/workflow-config.test.ts +111 -0
- package/shared/workflow-config.ts +7 -0
- package/shared/workflow-engine.test.ts +388 -0
- package/shared/workflow-engine.ts +1 -1
- package/shared/workflow-normalizer.test.ts +119 -0
- package/shared/workflow-normalizer.ts +118 -0
- package/templates/agents/QUICK-REFERENCE.md +1 -0
- package/templates/agents/README.md +1 -0
- package/templates/agents/SKILL-TRIGGERS.md +11 -0
- package/templates/agents/green-team/deep-dive.md +361 -0
- package/templates/hooks/end-session.sh +4 -1
- package/templates/hooks/init-session.sh +1 -0
- package/templates/hooks/orbital-emit.sh +2 -2
- package/templates/hooks/orbital-report-deploy.sh +4 -4
- package/templates/hooks/orbital-report-gates.sh +4 -4
- package/templates/hooks/orbital-scope-update.sh +1 -1
- package/templates/hooks/scope-commit-logger.sh +2 -2
- package/templates/hooks/scope-create-cleanup.sh +2 -2
- package/templates/hooks/scope-create-gate.sh +2 -5
- package/templates/hooks/scope-gate.sh +4 -6
- package/templates/hooks/scope-helpers.sh +28 -1
- package/templates/hooks/scope-lifecycle-gate.sh +14 -5
- package/templates/hooks/scope-prepare.sh +67 -12
- package/templates/hooks/scope-transition.sh +14 -6
- package/templates/hooks/time-tracker.sh +2 -5
- package/templates/migrations/renames.json +1 -0
- package/templates/orbital.config.json +8 -6
- package/{shared/default-workflow.json → templates/presets/default.json} +65 -0
- package/templates/presets/development.json +4 -4
- package/templates/presets/gitflow.json +7 -0
- package/templates/prompts/README.md +23 -0
- package/templates/prompts/deep-dive-audit.md +94 -0
- package/templates/quick/rules.md +56 -5
- package/templates/settings-hooks.json +1 -1
- package/templates/skills/git-commit/SKILL.md +27 -7
- package/templates/skills/git-dev/SKILL.md +13 -4
- package/templates/skills/git-main/SKILL.md +13 -3
- package/templates/skills/git-production/SKILL.md +9 -2
- package/templates/skills/git-staging/SKILL.md +11 -3
- package/templates/skills/scope-create/SKILL.md +17 -3
- package/templates/skills/scope-fix-review/SKILL.md +14 -7
- package/templates/skills/scope-implement/SKILL.md +15 -4
- package/templates/skills/scope-post-review/SKILL.md +77 -7
- package/templates/skills/scope-pre-review/SKILL.md +11 -4
- package/templates/skills/scope-verify/SKILL.md +5 -3
- package/templates/skills/test-code-review/SKILL.md +41 -33
- package/templates/skills/test-scaffold/SKILL.md +222 -0
- package/dist/assets/WorkflowVisualizer-BZ21PIIF.js +0 -84
- package/dist/assets/charts-D__PA1zp.js +0 -72
- package/dist/assets/index-D1G6i0nS.css +0 -1
- package/dist/assets/index-DpItvKpf.js +0 -419
- package/dist/assets/ui-BvF022GT.js +0 -53
- package/index.html +0 -15
- package/postcss.config.js +0 -6
- package/src/App.tsx +0 -33
- package/src/components/AgentBadge.tsx +0 -40
- package/src/components/BatchPreflightModal.tsx +0 -115
- package/src/components/CardDisplayToggle.tsx +0 -74
- package/src/components/ColumnHeaderActions.tsx +0 -55
- package/src/components/ColumnMenu.tsx +0 -99
- package/src/components/DeployHistory.tsx +0 -141
- package/src/components/DispatchModal.tsx +0 -164
- package/src/components/DispatchPopover.tsx +0 -139
- package/src/components/DragOverlay.tsx +0 -25
- package/src/components/DriftSidebar.tsx +0 -140
- package/src/components/EnvironmentStrip.tsx +0 -88
- package/src/components/ErrorBoundary.tsx +0 -62
- package/src/components/FilterChip.tsx +0 -105
- package/src/components/GateIndicator.tsx +0 -33
- package/src/components/IdeaDetailModal.tsx +0 -190
- package/src/components/IdeaFormDialog.tsx +0 -113
- package/src/components/KanbanColumn.tsx +0 -201
- package/src/components/MarkdownRenderer.tsx +0 -114
- package/src/components/NeonGrid.tsx +0 -128
- package/src/components/PromotionQueue.tsx +0 -89
- package/src/components/ScopeCard.tsx +0 -234
- package/src/components/ScopeDetailModal.tsx +0 -255
- package/src/components/ScopeFilterBar.tsx +0 -152
- package/src/components/SearchInput.tsx +0 -102
- package/src/components/SessionPanel.tsx +0 -335
- package/src/components/SprintContainer.tsx +0 -303
- package/src/components/SprintDependencyDialog.tsx +0 -78
- package/src/components/SprintPreflightModal.tsx +0 -138
- package/src/components/StatusBar.tsx +0 -168
- package/src/components/SwimCell.tsx +0 -67
- package/src/components/SwimLaneRow.tsx +0 -94
- package/src/components/SwimlaneBoardView.tsx +0 -108
- package/src/components/VersionBadge.tsx +0 -139
- package/src/components/ViewModeSelector.tsx +0 -114
- package/src/components/config/AgentChip.tsx +0 -53
- package/src/components/config/AgentCreateDialog.tsx +0 -321
- package/src/components/config/AgentEditor.tsx +0 -175
- package/src/components/config/DirectoryTree.tsx +0 -582
- package/src/components/config/FileEditor.tsx +0 -550
- package/src/components/config/HookChip.tsx +0 -50
- package/src/components/config/StageCard.tsx +0 -198
- package/src/components/config/TransitionZone.tsx +0 -173
- package/src/components/config/UnifiedWorkflowPipeline.tsx +0 -216
- package/src/components/config/WorkflowPipeline.tsx +0 -161
- package/src/components/source-control/BranchList.tsx +0 -93
- package/src/components/source-control/BranchPanel.tsx +0 -105
- package/src/components/source-control/CommitLog.tsx +0 -100
- package/src/components/source-control/CommitRow.tsx +0 -47
- package/src/components/source-control/GitHubPanel.tsx +0 -110
- package/src/components/source-control/GitHubSetupGuide.tsx +0 -52
- package/src/components/source-control/GitOverviewBar.tsx +0 -101
- package/src/components/source-control/PullRequestList.tsx +0 -69
- package/src/components/source-control/WorktreeList.tsx +0 -80
- package/src/components/ui/badge.tsx +0 -41
- package/src/components/ui/button.tsx +0 -55
- package/src/components/ui/card.tsx +0 -78
- package/src/components/ui/dialog.tsx +0 -94
- package/src/components/ui/popover.tsx +0 -33
- package/src/components/ui/scroll-area.tsx +0 -54
- package/src/components/ui/separator.tsx +0 -28
- package/src/components/ui/tabs.tsx +0 -52
- package/src/components/ui/toggle-switch.tsx +0 -35
- package/src/components/ui/tooltip.tsx +0 -27
- package/src/components/workflow/AddEdgeDialog.tsx +0 -217
- package/src/components/workflow/AddListDialog.tsx +0 -201
- package/src/components/workflow/ChecklistEditor.tsx +0 -239
- package/src/components/workflow/CommandPrefixManager.tsx +0 -118
- package/src/components/workflow/ConfigSettingsPanel.tsx +0 -189
- package/src/components/workflow/DirectionSelector.tsx +0 -133
- package/src/components/workflow/DispatchConfigPanel.tsx +0 -180
- package/src/components/workflow/EdgeDetailPanel.tsx +0 -236
- package/src/components/workflow/EdgePropertyEditor.tsx +0 -251
- package/src/components/workflow/EditToolbar.tsx +0 -138
- package/src/components/workflow/HookDetailPanel.tsx +0 -250
- package/src/components/workflow/HookExecutionLog.tsx +0 -24
- package/src/components/workflow/HookSourceModal.tsx +0 -129
- package/src/components/workflow/HooksDashboard.tsx +0 -363
- package/src/components/workflow/ListPropertyEditor.tsx +0 -251
- package/src/components/workflow/MigrationPreviewDialog.tsx +0 -237
- package/src/components/workflow/MovementRulesPanel.tsx +0 -188
- package/src/components/workflow/NodeDetailPanel.tsx +0 -245
- package/src/components/workflow/PresetSelector.tsx +0 -414
- package/src/components/workflow/SkillCommandBuilder.tsx +0 -174
- package/src/components/workflow/WorkflowEdgeComponent.tsx +0 -145
- package/src/components/workflow/WorkflowNode.tsx +0 -147
- package/src/components/workflow/graphLayout.ts +0 -186
- package/src/components/workflow/mergeHooks.ts +0 -85
- package/src/components/workflow/useEditHistory.ts +0 -88
- package/src/components/workflow/useWorkflowEditor.ts +0 -262
- package/src/components/workflow/validateConfig.ts +0 -70
- package/src/hooks/useActiveDispatches.ts +0 -198
- package/src/hooks/useBoardSettings.ts +0 -170
- package/src/hooks/useCardDisplay.ts +0 -57
- package/src/hooks/useCcHooks.ts +0 -24
- package/src/hooks/useConfigTree.ts +0 -51
- package/src/hooks/useEnforcementRules.ts +0 -46
- package/src/hooks/useEvents.ts +0 -59
- package/src/hooks/useFileEditor.ts +0 -165
- package/src/hooks/useGates.ts +0 -57
- package/src/hooks/useIdeaActions.ts +0 -53
- package/src/hooks/useKanbanDnd.ts +0 -410
- package/src/hooks/useOrbitalConfig.ts +0 -54
- package/src/hooks/usePipeline.ts +0 -47
- package/src/hooks/usePipelineData.ts +0 -338
- package/src/hooks/useReconnect.ts +0 -25
- package/src/hooks/useScopeFilters.ts +0 -125
- package/src/hooks/useScopeSessions.ts +0 -44
- package/src/hooks/useScopes.ts +0 -67
- package/src/hooks/useSearch.ts +0 -67
- package/src/hooks/useSettings.tsx +0 -187
- package/src/hooks/useSocket.ts +0 -25
- package/src/hooks/useSourceControl.ts +0 -105
- package/src/hooks/useSprintPreflight.ts +0 -55
- package/src/hooks/useSprints.ts +0 -154
- package/src/hooks/useStatusBarHighlight.ts +0 -18
- package/src/hooks/useSwimlaneBoardSettings.ts +0 -104
- package/src/hooks/useTheme.ts +0 -9
- package/src/hooks/useTransitionReadiness.ts +0 -53
- package/src/hooks/useVersion.ts +0 -155
- package/src/hooks/useViolations.ts +0 -65
- package/src/hooks/useWorkflow.tsx +0 -125
- package/src/hooks/useZoomModifier.ts +0 -19
- package/src/index.css +0 -797
- package/src/layouts/DashboardLayout.tsx +0 -113
- package/src/lib/collisionDetection.ts +0 -20
- package/src/lib/scope-fields.ts +0 -61
- package/src/lib/swimlane.ts +0 -146
- package/src/lib/utils.ts +0 -15
- package/src/main.tsx +0 -19
- package/src/socket.ts +0 -11
- package/src/types/index.ts +0 -497
- package/src/views/AgentFeed.tsx +0 -339
- package/src/views/DeployPipeline.tsx +0 -59
- package/src/views/EnforcementView.tsx +0 -378
- package/src/views/PrimitivesConfig.tsx +0 -500
- package/src/views/QualityGates.tsx +0 -1012
- package/src/views/ScopeBoard.tsx +0 -454
- package/src/views/SessionTimeline.tsx +0 -516
- package/src/views/Settings.tsx +0 -183
- package/src/views/SourceControl.tsx +0 -95
- package/src/views/WorkflowVisualizer.tsx +0 -382
- package/tailwind.config.js +0 -161
- package/tsconfig.json +0 -25
- 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,265 @@
|
|
|
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 { ensureDynamicProfiles } from './utils/terminal-launcher.js';
|
|
25
|
+
import { createLogger } from './utils/logger.js';
|
|
26
|
+
const log = createLogger('project-context');
|
|
27
|
+
// ─── Factory ────────────────────────────────────────────────
|
|
28
|
+
/** Resolve the path to the bundled default workflow JSON. */
|
|
29
|
+
function getDefaultConfigPath() {
|
|
30
|
+
const __selfDir = path.dirname(fileURLToPath(import.meta.url));
|
|
31
|
+
return path.resolve(__selfDir, '../shared/default-workflow.json');
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Create a fully wired ProjectContext for a single project.
|
|
35
|
+
*
|
|
36
|
+
* Create a fully wired context for a single project. Each ProjectContext has its own
|
|
37
|
+
* database, services, watchers, and intervals.
|
|
38
|
+
*/
|
|
39
|
+
export async function createProjectContext(projectId, projectRoot, emitter) {
|
|
40
|
+
// Load project config
|
|
41
|
+
const config = loadConfig(projectRoot);
|
|
42
|
+
// Initialize database
|
|
43
|
+
const db = openProjectDatabase(config.dbDir);
|
|
44
|
+
// Initialize workflow engine
|
|
45
|
+
const workflowEngine = new WorkflowEngine(defaultWorkflow);
|
|
46
|
+
// Generate shell manifest for bash hooks
|
|
47
|
+
if (!fs.existsSync(config.configDir))
|
|
48
|
+
fs.mkdirSync(config.configDir, { recursive: true });
|
|
49
|
+
const manifestPath = path.join(config.configDir, 'workflow-manifest.sh');
|
|
50
|
+
fs.writeFileSync(manifestPath, workflowEngine.generateShellManifest(), 'utf-8');
|
|
51
|
+
// Ensure icebox directory exists
|
|
52
|
+
const iceboxDir = path.join(config.scopesDir, 'icebox');
|
|
53
|
+
if (!fs.existsSync(iceboxDir))
|
|
54
|
+
fs.mkdirSync(iceboxDir, { recursive: true });
|
|
55
|
+
// Initialize services
|
|
56
|
+
const scopeCache = new ScopeCache();
|
|
57
|
+
const scopeService = new ScopeService(scopeCache, emitter, config.scopesDir, workflowEngine);
|
|
58
|
+
const eventService = new EventService(db, emitter);
|
|
59
|
+
const gateService = new GateService(db, emitter);
|
|
60
|
+
const deployService = new DeployService(db, emitter);
|
|
61
|
+
const sprintService = new SprintService(db, emitter, scopeService);
|
|
62
|
+
const sprintOrchestrator = new SprintOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot, config);
|
|
63
|
+
const batchOrchestrator = new BatchOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot, config);
|
|
64
|
+
const readinessService = new ReadinessService(scopeService, gateService, workflowEngine, config.projectRoot);
|
|
65
|
+
const workflowService = new WorkflowService(config.configDir, workflowEngine, config.scopesDir, getDefaultConfigPath());
|
|
66
|
+
workflowService.setSocketServer(emitter);
|
|
67
|
+
// Ensure engine reflects active config (may differ from bundled default)
|
|
68
|
+
workflowEngine.reload(workflowService.getActive());
|
|
69
|
+
const gitService = new GitService(config.projectRoot, scopeCache);
|
|
70
|
+
const githubService = new GitHubService(config.projectRoot);
|
|
71
|
+
let telemetryService = null;
|
|
72
|
+
let telemetryRouter = null;
|
|
73
|
+
const telemetryMod = './services/telemetry-service.js';
|
|
74
|
+
try {
|
|
75
|
+
const mod = await import(telemetryMod);
|
|
76
|
+
telemetryService = new mod.TelemetryService(db, config.telemetry, config.projectName, config.projectRoot);
|
|
77
|
+
if (telemetryService?.enabled) {
|
|
78
|
+
telemetryRouter = mod.createTelemetryRoutes({ telemetryService });
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
catch { /* telemetry service not installed */ }
|
|
82
|
+
// Wire active-group guard
|
|
83
|
+
scopeService.setActiveGroupCheck((scopeId) => sprintService.getActiveGroupForScope(scopeId));
|
|
84
|
+
// Wire event inference (Fix 8: diagnostic log lines match index.ts)
|
|
85
|
+
eventService.onIngest((eventType, scopeId, data) => {
|
|
86
|
+
if (eventType === 'SESSION_START' && typeof data.dispatch_id === 'string' && typeof data.pid === 'number') {
|
|
87
|
+
linkPidToDispatch(db, data.dispatch_id, data.pid);
|
|
88
|
+
log.debug('Linked PID to dispatch', { pid: data.pid, dispatch_id: data.dispatch_id });
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
if (eventType === 'SCOPE_GATE_LIFTED' && scopeId != null) {
|
|
92
|
+
const id = Number(scopeId);
|
|
93
|
+
if (!isNaN(id) && id > 0) {
|
|
94
|
+
resolveActiveDispatchesForScope(db, emitter, id, 'completed');
|
|
95
|
+
log.debug('Resolved dispatches for scope gate lift', { scope_id: id });
|
|
96
|
+
}
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
if (eventType === 'SESSION_END') {
|
|
100
|
+
const outcome = data.normal_exit === true ? 'completed' : 'abandoned';
|
|
101
|
+
let resolvedIds = [];
|
|
102
|
+
if (typeof data.dispatch_id === 'string') {
|
|
103
|
+
resolvedIds = resolveDispatchesByDispatchId(db, emitter, data.dispatch_id, outcome);
|
|
104
|
+
}
|
|
105
|
+
if (resolvedIds.length === 0 && typeof data.pid === 'number') {
|
|
106
|
+
resolvedIds = resolveDispatchesByPid(db, emitter, data.pid, outcome);
|
|
107
|
+
}
|
|
108
|
+
if (resolvedIds.length > 0)
|
|
109
|
+
log.info('Session resolved', { count: resolvedIds.length, outcome });
|
|
110
|
+
// For abandoned dispatches, immediately try auto-revert so the scope
|
|
111
|
+
// returns to its pre-dispatch status without requiring user interaction
|
|
112
|
+
if (outcome === 'abandoned') {
|
|
113
|
+
for (const eventId of resolvedIds) {
|
|
114
|
+
tryAutoRevertAndClear(db, emitter, scopeService, workflowEngine, eventId);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
if (resolvedIds.length > 0)
|
|
118
|
+
batchOrchestrator.resolveStaleBatches();
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
// Status inference
|
|
122
|
+
if (scopeId == null)
|
|
123
|
+
return;
|
|
124
|
+
const id = Number(scopeId);
|
|
125
|
+
if (isNaN(id) || id <= 0)
|
|
126
|
+
return;
|
|
127
|
+
const current = scopeService.getById(id);
|
|
128
|
+
if (current?.status === 'icebox')
|
|
129
|
+
return;
|
|
130
|
+
const currentStatus = current?.status ?? '';
|
|
131
|
+
const result = workflowEngine.inferStatus(eventType, currentStatus, data);
|
|
132
|
+
if (result === null)
|
|
133
|
+
return;
|
|
134
|
+
if (typeof result === 'object' && 'dispatchResolution' in result) {
|
|
135
|
+
resolveActiveDispatchesForScope(db, emitter, id, result.resolution);
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
scopeService.updateStatus(id, result, 'event');
|
|
139
|
+
});
|
|
140
|
+
// Wire status change callbacks
|
|
141
|
+
scopeService.onStatusChange((scopeId, newStatus) => {
|
|
142
|
+
if (workflowEngine.isTerminalStatus(newStatus))
|
|
143
|
+
sprintOrchestrator.onScopeReachedDev(scopeId);
|
|
144
|
+
batchOrchestrator.onScopeStatusChanged(scopeId, newStatus);
|
|
145
|
+
});
|
|
146
|
+
scopeService.onStatusChange((scopeId, newStatus) => {
|
|
147
|
+
if (workflowEngine.isTerminalStatus(newStatus)) {
|
|
148
|
+
resolveActiveDispatchesForScope(db, emitter, scopeId, 'completed');
|
|
149
|
+
}
|
|
150
|
+
});
|
|
151
|
+
// Load scopes from filesystem and reconcile directory mismatches
|
|
152
|
+
const scopeCount = scopeService.syncFromFilesystem();
|
|
153
|
+
const reconciled = scopeService.reconcileDirectories();
|
|
154
|
+
if (reconciled > 0)
|
|
155
|
+
log.info('Reconciled scope directories', { id: projectId, count: reconciled });
|
|
156
|
+
// Start watchers
|
|
157
|
+
const scopeWatcher = startScopeWatcher(config.scopesDir, scopeService);
|
|
158
|
+
const eventWatcher = startEventWatcher(config.eventsDir, eventService);
|
|
159
|
+
// Write iTerm2 dispatch profiles (Fix 2 + Fix 5: per-project prefix)
|
|
160
|
+
ensureDynamicProfiles(workflowEngine, config.terminal.profilePrefix);
|
|
161
|
+
// Recover active sprints/batches
|
|
162
|
+
await sprintOrchestrator.recoverActiveSprints();
|
|
163
|
+
await batchOrchestrator.recoverActiveBatches();
|
|
164
|
+
// Resolve stale batches on startup (Fix 6: catches stuck dispatches from previous runs)
|
|
165
|
+
const staleBatchesResolved = batchOrchestrator.resolveStaleBatches();
|
|
166
|
+
if (staleBatchesResolved > 0)
|
|
167
|
+
log.info('Resolved stale batches', { count: staleBatchesResolved });
|
|
168
|
+
// Resolve stale dispatches
|
|
169
|
+
resolveStaleDispatches(db, emitter, scopeService, workflowEngine, config.dispatch.staleTimeoutMinutes);
|
|
170
|
+
// Initial session sync + legacy purge (Fix 7)
|
|
171
|
+
syncClaudeSessionsToDB(db, scopeService, config.projectRoot).then((count) => {
|
|
172
|
+
if (count > 0)
|
|
173
|
+
log.info('Synced sessions', { id: projectId, count });
|
|
174
|
+
const purged = db.prepare("DELETE FROM sessions WHERE action IS NULL AND id LIKE 'claude-%'").run();
|
|
175
|
+
if (purged.changes > 0)
|
|
176
|
+
log.info('Purged legacy session rows', { count: purged.changes });
|
|
177
|
+
if (telemetryService?.enabled) {
|
|
178
|
+
telemetryService.uploadChangedSessions().catch(() => { });
|
|
179
|
+
}
|
|
180
|
+
}).catch(err => log.error('Session sync failed', { error: err.message }));
|
|
181
|
+
// Start periodic intervals
|
|
182
|
+
const intervals = [];
|
|
183
|
+
// Fix 11: periodic batch recovery (two-phase completion B-1)
|
|
184
|
+
intervals.push(setInterval(() => {
|
|
185
|
+
batchOrchestrator.recoverActiveBatches().catch(err => log.error('Batch recovery failed', { error: err.message }));
|
|
186
|
+
}, 30_000));
|
|
187
|
+
intervals.push(setInterval(() => {
|
|
188
|
+
batchOrchestrator.resolveStaleBatches();
|
|
189
|
+
}, 30_000));
|
|
190
|
+
intervals.push(setInterval(() => {
|
|
191
|
+
resolveStaleDispatches(db, emitter, scopeService, workflowEngine, config.dispatch.staleTimeoutMinutes);
|
|
192
|
+
}, 30_000));
|
|
193
|
+
intervals.push(setInterval(async () => {
|
|
194
|
+
const count = await syncClaudeSessionsToDB(db, scopeService, config.projectRoot);
|
|
195
|
+
if (count > 0)
|
|
196
|
+
emitter.emit('session:updated', { type: 'resync', count });
|
|
197
|
+
if (telemetryService?.enabled) {
|
|
198
|
+
telemetryService.uploadChangedSessions().catch(() => { });
|
|
199
|
+
}
|
|
200
|
+
}, 5 * 60_000));
|
|
201
|
+
let lastGitHash = '';
|
|
202
|
+
intervals.push(setInterval(async () => {
|
|
203
|
+
try {
|
|
204
|
+
const hash = await gitService.getStatusHash();
|
|
205
|
+
if (lastGitHash && hash !== lastGitHash) {
|
|
206
|
+
gitService.clearCache();
|
|
207
|
+
emitter.emit('git:status:changed');
|
|
208
|
+
}
|
|
209
|
+
lastGitHash = hash;
|
|
210
|
+
}
|
|
211
|
+
catch { /* ok */ }
|
|
212
|
+
}, 10_000));
|
|
213
|
+
log.info('Project ready', { id: projectId, scopes: scopeCount });
|
|
214
|
+
const ctx = {
|
|
215
|
+
id: projectId,
|
|
216
|
+
config,
|
|
217
|
+
db,
|
|
218
|
+
workflowEngine,
|
|
219
|
+
emitter,
|
|
220
|
+
scopeCache,
|
|
221
|
+
scopeService,
|
|
222
|
+
eventService,
|
|
223
|
+
gateService,
|
|
224
|
+
deployService,
|
|
225
|
+
sprintService,
|
|
226
|
+
sprintOrchestrator,
|
|
227
|
+
batchOrchestrator,
|
|
228
|
+
readinessService,
|
|
229
|
+
workflowService,
|
|
230
|
+
gitService,
|
|
231
|
+
githubService,
|
|
232
|
+
telemetryService,
|
|
233
|
+
telemetryRouter,
|
|
234
|
+
scopeWatcher,
|
|
235
|
+
eventWatcher,
|
|
236
|
+
intervals,
|
|
237
|
+
status: 'active',
|
|
238
|
+
async shutdown() {
|
|
239
|
+
log.info('Shutting down project context', { id: projectId });
|
|
240
|
+
for (const interval of intervals)
|
|
241
|
+
clearInterval(interval);
|
|
242
|
+
intervals.length = 0;
|
|
243
|
+
try {
|
|
244
|
+
await scopeWatcher.close();
|
|
245
|
+
}
|
|
246
|
+
catch (e) {
|
|
247
|
+
log.error('Scope watcher close failed', { id: projectId, error: String(e) });
|
|
248
|
+
}
|
|
249
|
+
try {
|
|
250
|
+
await eventWatcher.close();
|
|
251
|
+
}
|
|
252
|
+
catch (e) {
|
|
253
|
+
log.error('Event watcher close failed', { id: projectId, error: String(e) });
|
|
254
|
+
}
|
|
255
|
+
try {
|
|
256
|
+
db.close();
|
|
257
|
+
}
|
|
258
|
+
catch (e) {
|
|
259
|
+
log.error('DB close failed', { id: projectId, error: String(e) });
|
|
260
|
+
}
|
|
261
|
+
ctx.status = 'offline';
|
|
262
|
+
},
|
|
263
|
+
};
|
|
264
|
+
return ctx;
|
|
265
|
+
}
|
|
@@ -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
|
+
}
|