orbital-command 0.1.4 → 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.
- package/bin/orbital.js +676 -53
- package/dist/assets/PrimitivesConfig-CrmQXYh4.js +32 -0
- package/dist/assets/QualityGates-BbasOsF3.js +21 -0
- package/dist/assets/SessionTimeline-CGeJsVvy.js +1 -0
- package/dist/assets/Settings-oiM496mc.js +12 -0
- package/dist/assets/SourceControl-B1fP2nJL.js +41 -0
- package/dist/assets/WorkflowVisualizer-CWLYf-f0.js +74 -0
- package/dist/assets/arrow-down-CPy85_J6.js +6 -0
- package/dist/assets/charts-DbDg0Psc.js +68 -0
- package/dist/assets/circle-x-Cwz6ZQDV.js +6 -0
- package/dist/assets/file-text-C46Xr65c.js +6 -0
- package/dist/assets/formatDistanceToNow-BMqsSP44.js +1 -0
- package/dist/assets/globe-Cn2yNZUD.js +6 -0
- package/dist/assets/index-Aj4sV8Al.css +1 -0
- package/dist/assets/index-Bc9dK3MW.js +354 -0
- package/dist/assets/key-OPaNTWJ5.js +6 -0
- package/dist/assets/minus-GMsbpKym.js +6 -0
- package/dist/assets/shield-DwAFkDYI.js +6 -0
- package/dist/assets/ui-BmsSg9jU.js +53 -0
- package/dist/assets/useWorkflowEditor-BJkTX_NR.js +16 -0
- package/dist/assets/{vendor-Dzv9lrRc.js → vendor-Bqt8AJn2.js} +1 -1
- package/dist/assets/zap-DfbUoOty.js +11 -0
- package/dist/favicon.svg +1 -0
- package/dist/index.html +6 -5
- package/dist/server/server/__tests__/data-routes.test.js +124 -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 +137 -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 +138 -0
- package/dist/server/server/config.js +17 -2
- package/dist/server/server/database.js +27 -12
- package/dist/server/server/global-config.js +143 -0
- package/dist/server/server/index.js +882 -252
- package/dist/server/server/init.js +579 -194
- 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.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 +255 -0
- package/dist/server/server/project-emitter.js +41 -0
- package/dist/server/server/project-manager.js +297 -0
- package/dist/server/server/routes/config-routes.js +1 -3
- package/dist/server/server/routes/data-routes.js +22 -110
- package/dist/server/server/routes/dispatch-routes.js +15 -9
- 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 +37 -23
- 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 +2 -0
- package/dist/server/server/services/batch-orchestrator.js +26 -16
- package/dist/server/server/services/claude-session-service.js +17 -14
- 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 +217 -126
- package/dist/server/server/services/scope-service.test.js +137 -0
- package/dist/server/server/services/sprint-orchestrator.js +7 -6
- package/dist/server/server/services/sprint-service.js +21 -1
- 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/telemetry-service.js +143 -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/update-planner.js +279 -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 +77 -20
- package/dist/server/server/utils/dispatch-utils.test.js +182 -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 +10 -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/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 +155 -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 +35 -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 +74 -0
- package/dist/server/shared/__fixtures__/workflow-configs.js +75 -0
- 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.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 +20 -15
- package/schemas/orbital.config.schema.json +16 -1
- package/scripts/postinstall.js +55 -7
- package/server/__tests__/data-routes.test.ts +149 -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 +157 -0
- package/server/__tests__/sprint-routes.test.ts +118 -0
- package/server/__tests__/workflow-routes.test.ts +120 -0
- package/server/config-migrator.ts +163 -0
- package/server/config.ts +26 -2
- package/server/database.ts +35 -18
- package/server/global-config.ts +200 -0
- package/server/index.ts +975 -287
- package/server/init.ts +625 -182
- 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/scope-parser.test.ts +270 -0
- package/server/parsers/scope-parser.ts +79 -31
- package/server/project-context.ts +309 -0
- package/server/project-emitter.ts +50 -0
- package/server/project-manager.ts +369 -0
- package/server/routes/config-routes.ts +3 -5
- package/server/routes/data-routes.ts +28 -141
- package/server/routes/dispatch-routes.ts +19 -11
- package/server/routes/git-routes.ts +77 -0
- package/server/routes/manifest-routes.ts +388 -0
- package/server/routes/scope-routes.ts +29 -25
- 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 +2 -0
- package/server/services/batch-orchestrator.ts +24 -16
- package/server/services/claude-session-service.ts +16 -14
- 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 +220 -126
- package/server/services/sprint-orchestrator.ts +7 -7
- package/server/services/sprint-service.test.ts +271 -0
- package/server/services/sprint-service.ts +27 -3
- package/server/services/sync-service.ts +482 -0
- package/server/services/sync-types.ts +77 -0
- package/server/services/telemetry-service.ts +195 -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/update-planner.ts +346 -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 +97 -27
- package/server/utils/logger.ts +40 -3
- package/server/utils/package-info.ts +32 -0
- package/server/utils/route-helpers.ts +12 -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/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 +187 -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 +43 -0
- package/server/wizard/phases/workflow-setup.ts +28 -0
- package/server/wizard/types.ts +56 -0
- package/server/wizard/ui.ts +93 -0
- package/shared/__fixtures__/workflow-configs.ts +80 -0
- package/shared/default-workflow.json +65 -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-normalizer.test.ts +119 -0
- package/shared/workflow-normalizer.ts +118 -0
- package/templates/hooks/end-session.sh +3 -1
- 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-create-cleanup.sh +2 -2
- package/templates/hooks/scope-create-gate.sh +0 -1
- package/templates/hooks/scope-helpers.sh +18 -0
- package/templates/hooks/scope-prepare.sh +66 -11
- package/templates/migrations/renames.json +1 -0
- package/templates/orbital.config.json +7 -2
- package/templates/settings-hooks.json +1 -1
- package/templates/skills/git-commit/SKILL.md +9 -4
- package/templates/skills/git-dev/SKILL.md +8 -3
- package/templates/skills/git-main/SKILL.md +8 -2
- package/templates/skills/git-production/SKILL.md +6 -2
- package/templates/skills/git-staging/SKILL.md +8 -3
- package/templates/skills/scope-create/SKILL.md +17 -3
- package/templates/skills/scope-fix-review/SKILL.md +6 -3
- package/templates/skills/scope-implement/SKILL.md +4 -1
- package/templates/skills/scope-post-review/SKILL.md +63 -5
- package/templates/skills/scope-pre-review/SKILL.md +5 -2
- 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 -49
|
@@ -8,6 +8,7 @@ const log = createLogger('scope');
|
|
|
8
8
|
export interface ParsedScope {
|
|
9
9
|
id: number;
|
|
10
10
|
title: string;
|
|
11
|
+
slug?: string;
|
|
11
12
|
status: string;
|
|
12
13
|
priority: string | null;
|
|
13
14
|
effort_estimate: string | null;
|
|
@@ -21,6 +22,7 @@ export interface ParsedScope {
|
|
|
21
22
|
raw_content: string;
|
|
22
23
|
sessions: Record<string, string[]>;
|
|
23
24
|
is_ghost: boolean;
|
|
25
|
+
favourite: boolean;
|
|
24
26
|
}
|
|
25
27
|
|
|
26
28
|
const VALID_PRIORITIES = new Set(['critical', 'high', 'medium', 'low']);
|
|
@@ -69,35 +71,73 @@ export function normalizeStatus(raw: string): string {
|
|
|
69
71
|
return STATUS_MAP[raw] ?? raw;
|
|
70
72
|
}
|
|
71
73
|
|
|
74
|
+
/** Generate a stable positive integer hash from a string (for synthetic icebox IDs) */
|
|
75
|
+
function slugHash(s: string): number {
|
|
76
|
+
let hash = 5381;
|
|
77
|
+
for (let i = 0; i < s.length; i++) {
|
|
78
|
+
hash = ((hash << 5) + hash + s.charCodeAt(i)) >>> 0;
|
|
79
|
+
}
|
|
80
|
+
// Keep in range 10000-2147483647 to avoid collisions with real scope IDs and suffix-encoded IDs
|
|
81
|
+
return 10000 + (hash % 2137483647);
|
|
82
|
+
}
|
|
83
|
+
|
|
72
84
|
/**
|
|
73
85
|
* Parse a scope markdown file into structured data.
|
|
74
86
|
* Handles both YAML frontmatter and plain markdown formats.
|
|
75
87
|
*/
|
|
76
88
|
export function parseScopeFile(filePath: string): ParsedScope | null {
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
// Extract ID from filename pattern: NNN[suffix]-description.md
|
|
82
|
-
// Suffixes (a-d, X) encode as thousands offset for unique DB keys
|
|
83
|
-
const idMatch = fileName.match(/^(\d+)([a-dA-DxX])?/);
|
|
84
|
-
const fileId = idMatch ? scopeFileId(parseInt(idMatch[1], 10), idMatch[2]) : 0;
|
|
85
|
-
|
|
86
|
-
// Skip non-scope files
|
|
87
|
-
if (fileId === 0 && !fileName.startsWith('0')) {
|
|
88
|
-
// Files like _template.md, technical-debt.md, backlog_plan.md
|
|
89
|
-
if (fileName.startsWith('_') || !idMatch) return null;
|
|
90
|
-
}
|
|
89
|
+
try {
|
|
90
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
91
|
+
const fileName = path.basename(filePath, '.md');
|
|
92
|
+
const dirName = path.basename(path.dirname(filePath));
|
|
91
93
|
|
|
92
|
-
|
|
93
|
-
|
|
94
|
+
// Extract ID from filename pattern: NNN[suffix]-description.md
|
|
95
|
+
// Suffixes (a-d, X) encode as thousands offset for unique DB keys
|
|
96
|
+
const idMatch = fileName.match(/^(\d+)([a-dA-DxX])?/);
|
|
97
|
+
const fileId = idMatch ? scopeFileId(parseInt(idMatch[1], 10), idMatch[2]) : 0;
|
|
94
98
|
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
99
|
+
// Slug-only icebox files: no numeric prefix, e.g. "onboarding-flow.md"
|
|
100
|
+
const isSlugOnly = !idMatch && dirName === 'icebox';
|
|
101
|
+
|
|
102
|
+
// Skip non-scope files (but allow slug-only icebox files)
|
|
103
|
+
if (fileId === 0 && !fileName.startsWith('0') && !isSlugOnly) {
|
|
104
|
+
// Files like _template.md, technical-debt.md, backlog_plan.md
|
|
105
|
+
if (fileName.startsWith('_') || !idMatch) return null;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
// For slug-only icebox files, generate a stable negative ID for internal cache indexing.
|
|
109
|
+
// This avoids collisions with real scope IDs (positive) and between icebox items.
|
|
110
|
+
const effectiveId = isSlugOnly ? -slugHash(fileName) : fileId;
|
|
98
111
|
|
|
99
|
-
|
|
100
|
-
|
|
112
|
+
// Try YAML frontmatter first
|
|
113
|
+
let frontmatter: Record<string, unknown>;
|
|
114
|
+
let markdownBody: string;
|
|
115
|
+
try {
|
|
116
|
+
({ data: frontmatter, content: markdownBody } = matter(content));
|
|
117
|
+
} catch (err) {
|
|
118
|
+
log.warn('Skipping scope with malformed YAML frontmatter', { file: filePath, error: (err as Error).message });
|
|
119
|
+
return null;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
if (frontmatter && Object.keys(frontmatter).length > 0) {
|
|
123
|
+
const scope = parseFrontmatterScope(frontmatter, markdownBody, filePath, effectiveId, dirName);
|
|
124
|
+
// Populate slug for icebox items
|
|
125
|
+
if (isSlugOnly || dirName === 'icebox') {
|
|
126
|
+
scope.slug = isSlugOnly ? fileName : fileName.replace(/^\d+[a-dA-DxX]?-/, '');
|
|
127
|
+
}
|
|
128
|
+
return scope;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
// Fallback: extract from markdown structure
|
|
132
|
+
const fallbackScope = parseMarkdownScope(content, filePath, effectiveId, dirName);
|
|
133
|
+
if (isSlugOnly || dirName === 'icebox') {
|
|
134
|
+
fallbackScope.slug = isSlugOnly ? fileName : fileName.replace(/^\d+[a-dA-DxX]?-/, '');
|
|
135
|
+
}
|
|
136
|
+
return fallbackScope;
|
|
137
|
+
} catch (err) {
|
|
138
|
+
log.warn('Failed to parse scope file', { file: filePath, error: (err as Error).message });
|
|
139
|
+
return null;
|
|
140
|
+
}
|
|
101
141
|
}
|
|
102
142
|
|
|
103
143
|
function parseFrontmatterScope(
|
|
@@ -128,6 +168,7 @@ function parseFrontmatterScope(
|
|
|
128
168
|
raw_content: body.trim(),
|
|
129
169
|
sessions: parseSessions(fm.sessions),
|
|
130
170
|
is_ghost: fm.ghost === true,
|
|
171
|
+
favourite: fm.favourite === true,
|
|
131
172
|
};
|
|
132
173
|
}
|
|
133
174
|
|
|
@@ -173,6 +214,7 @@ function parseMarkdownScope(
|
|
|
173
214
|
raw_content: content,
|
|
174
215
|
sessions: {},
|
|
175
216
|
is_ghost: false,
|
|
217
|
+
favourite: false,
|
|
176
218
|
};
|
|
177
219
|
}
|
|
178
220
|
|
|
@@ -194,7 +236,7 @@ export function setValidStatuses(statuses: Iterable<string>): void {
|
|
|
194
236
|
validDirStatuses = new Set(statuses);
|
|
195
237
|
}
|
|
196
238
|
|
|
197
|
-
function inferStatusFromDir(dirName: string): string {
|
|
239
|
+
export function inferStatusFromDir(dirName: string): string {
|
|
198
240
|
if (validDirStatuses) {
|
|
199
241
|
return validDirStatuses.has(dirName) ? dirName : 'planning';
|
|
200
242
|
}
|
|
@@ -215,26 +257,32 @@ export function parseAllScopes(scopesDir: string): ParsedScope[] {
|
|
|
215
257
|
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
216
258
|
for (const entry of entries) {
|
|
217
259
|
const fullPath = path.join(dir, entry.name);
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
260
|
+
try {
|
|
261
|
+
if (entry.isDirectory()) {
|
|
262
|
+
scanDir(fullPath);
|
|
263
|
+
} else if (entry.name.endsWith('.md') && !entry.name.startsWith('_')) {
|
|
264
|
+
const parsed = parseScopeFile(fullPath);
|
|
265
|
+
if (parsed) scopes.push(parsed);
|
|
266
|
+
}
|
|
267
|
+
} catch (err) {
|
|
268
|
+
log.warn('Failed to process scope entry', { file: fullPath, error: (err as Error).message });
|
|
223
269
|
}
|
|
224
270
|
}
|
|
225
271
|
}
|
|
226
272
|
|
|
227
273
|
scanDir(scopesDir);
|
|
228
274
|
|
|
229
|
-
// Detect ID collisions —
|
|
275
|
+
// Detect ID collisions — first-seen wins, duplicates are dropped
|
|
230
276
|
const seen = new Map<number, string>();
|
|
231
|
-
|
|
277
|
+
const deduped = scopes.filter(scope => {
|
|
232
278
|
const existing = seen.get(scope.id);
|
|
233
279
|
if (existing) {
|
|
234
280
|
log.error('Scope ID collision — renumber one of them', { id: scope.id, existing, duplicate: scope.file_path });
|
|
281
|
+
return false;
|
|
235
282
|
}
|
|
236
283
|
seen.set(scope.id, scope.file_path);
|
|
237
|
-
|
|
284
|
+
return true;
|
|
285
|
+
});
|
|
238
286
|
|
|
239
|
-
return
|
|
287
|
+
return deduped.sort((a, b) => a.id - b.id);
|
|
240
288
|
}
|
|
@@ -0,0 +1,309 @@
|
|
|
1
|
+
import path from 'path';
|
|
2
|
+
import fs from 'fs';
|
|
3
|
+
import { fileURLToPath } from 'url';
|
|
4
|
+
import type Database from 'better-sqlite3';
|
|
5
|
+
import type { FSWatcher } from 'chokidar';
|
|
6
|
+
import { openProjectDatabase } from './database.js';
|
|
7
|
+
import { loadConfig } from './config.js';
|
|
8
|
+
import type { OrbitalConfig } from './config.js';
|
|
9
|
+
import { ProjectEmitter } from './project-emitter.js';
|
|
10
|
+
import { ScopeCache } from './services/scope-cache.js';
|
|
11
|
+
import { ScopeService } from './services/scope-service.js';
|
|
12
|
+
import { EventService } from './services/event-service.js';
|
|
13
|
+
import { GateService } from './services/gate-service.js';
|
|
14
|
+
import { DeployService } from './services/deploy-service.js';
|
|
15
|
+
import { SprintService } from './services/sprint-service.js';
|
|
16
|
+
import { SprintOrchestrator } from './services/sprint-orchestrator.js';
|
|
17
|
+
import { BatchOrchestrator } from './services/batch-orchestrator.js';
|
|
18
|
+
import { ReadinessService } from './services/readiness-service.js';
|
|
19
|
+
import { WorkflowService } from './services/workflow-service.js';
|
|
20
|
+
import { GitService } from './services/git-service.js';
|
|
21
|
+
import { GitHubService } from './services/github-service.js';
|
|
22
|
+
import { WorkflowEngine } from '../shared/workflow-engine.js';
|
|
23
|
+
import type { WorkflowConfig } from '../shared/workflow-config.js';
|
|
24
|
+
import defaultWorkflow from '../shared/default-workflow.json' with { type: 'json' };
|
|
25
|
+
import { startScopeWatcher } from './watchers/scope-watcher.js';
|
|
26
|
+
import { startEventWatcher } from './watchers/event-watcher.js';
|
|
27
|
+
import { resolveStaleDispatches, resolveActiveDispatchesForScope, resolveDispatchesByPid, resolveDispatchesByDispatchId, linkPidToDispatch, tryAutoRevertAndClear } from './utils/dispatch-utils.js';
|
|
28
|
+
import { syncClaudeSessionsToDB } from './services/claude-session-service.js';
|
|
29
|
+
import { TelemetryService } from './services/telemetry-service.js';
|
|
30
|
+
import { ensureDynamicProfiles } from './utils/terminal-launcher.js';
|
|
31
|
+
import { createLogger } from './utils/logger.js';
|
|
32
|
+
|
|
33
|
+
const log = createLogger('project-context');
|
|
34
|
+
|
|
35
|
+
// ─── Types ──────────────────────────────────────────────────
|
|
36
|
+
|
|
37
|
+
export type ProjectStatus = 'active' | 'error' | 'offline';
|
|
38
|
+
|
|
39
|
+
export interface ProjectContext {
|
|
40
|
+
/** Project slug ID (derived from directory name) */
|
|
41
|
+
id: string;
|
|
42
|
+
/** Loaded project config */
|
|
43
|
+
config: OrbitalConfig;
|
|
44
|
+
/** Per-project SQLite database */
|
|
45
|
+
db: Database.Database;
|
|
46
|
+
/** Per-project workflow engine */
|
|
47
|
+
workflowEngine: WorkflowEngine;
|
|
48
|
+
/** Project-scoped socket emitter */
|
|
49
|
+
emitter: ProjectEmitter;
|
|
50
|
+
|
|
51
|
+
// Services
|
|
52
|
+
scopeCache: ScopeCache;
|
|
53
|
+
scopeService: ScopeService;
|
|
54
|
+
eventService: EventService;
|
|
55
|
+
gateService: GateService;
|
|
56
|
+
deployService: DeployService;
|
|
57
|
+
sprintService: SprintService;
|
|
58
|
+
sprintOrchestrator: SprintOrchestrator;
|
|
59
|
+
batchOrchestrator: BatchOrchestrator;
|
|
60
|
+
readinessService: ReadinessService;
|
|
61
|
+
workflowService: WorkflowService;
|
|
62
|
+
gitService: GitService;
|
|
63
|
+
githubService: GitHubService;
|
|
64
|
+
telemetryService: TelemetryService;
|
|
65
|
+
|
|
66
|
+
// Watchers
|
|
67
|
+
scopeWatcher: FSWatcher;
|
|
68
|
+
eventWatcher: FSWatcher;
|
|
69
|
+
|
|
70
|
+
// Intervals (cleanup, sync, polling)
|
|
71
|
+
intervals: ReturnType<typeof setInterval>[];
|
|
72
|
+
|
|
73
|
+
// Status
|
|
74
|
+
status: ProjectStatus;
|
|
75
|
+
error?: string;
|
|
76
|
+
|
|
77
|
+
// Lifecycle
|
|
78
|
+
shutdown(): Promise<void>;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// ─── Factory ────────────────────────────────────────────────
|
|
82
|
+
|
|
83
|
+
/** Resolve the path to the bundled default workflow JSON. */
|
|
84
|
+
function getDefaultConfigPath(): string {
|
|
85
|
+
const __selfDir = path.dirname(fileURLToPath(import.meta.url));
|
|
86
|
+
return path.resolve(__selfDir, '../shared/default-workflow.json');
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Create a fully wired ProjectContext for a single project.
|
|
91
|
+
*
|
|
92
|
+
* Create a fully wired context for a single project. Each ProjectContext has its own
|
|
93
|
+
* database, services, watchers, and intervals.
|
|
94
|
+
*/
|
|
95
|
+
export async function createProjectContext(
|
|
96
|
+
projectId: string,
|
|
97
|
+
projectRoot: string,
|
|
98
|
+
emitter: ProjectEmitter,
|
|
99
|
+
): Promise<ProjectContext> {
|
|
100
|
+
// Load project config
|
|
101
|
+
const config = loadConfig(projectRoot);
|
|
102
|
+
|
|
103
|
+
// Initialize database
|
|
104
|
+
const db = openProjectDatabase(config.dbDir);
|
|
105
|
+
|
|
106
|
+
// Initialize workflow engine
|
|
107
|
+
const workflowEngine = new WorkflowEngine(defaultWorkflow as WorkflowConfig);
|
|
108
|
+
|
|
109
|
+
// Generate shell manifest for bash hooks
|
|
110
|
+
if (!fs.existsSync(config.configDir)) fs.mkdirSync(config.configDir, { recursive: true });
|
|
111
|
+
const manifestPath = path.join(config.configDir, 'workflow-manifest.sh');
|
|
112
|
+
fs.writeFileSync(manifestPath, workflowEngine.generateShellManifest(), 'utf-8');
|
|
113
|
+
|
|
114
|
+
// Ensure icebox directory exists
|
|
115
|
+
const iceboxDir = path.join(config.scopesDir, 'icebox');
|
|
116
|
+
if (!fs.existsSync(iceboxDir)) fs.mkdirSync(iceboxDir, { recursive: true });
|
|
117
|
+
|
|
118
|
+
// Initialize services
|
|
119
|
+
const scopeCache = new ScopeCache();
|
|
120
|
+
const scopeService = new ScopeService(scopeCache, emitter, config.scopesDir, workflowEngine);
|
|
121
|
+
const eventService = new EventService(db, emitter);
|
|
122
|
+
const gateService = new GateService(db, emitter);
|
|
123
|
+
const deployService = new DeployService(db, emitter);
|
|
124
|
+
const sprintService = new SprintService(db, emitter, scopeService);
|
|
125
|
+
const sprintOrchestrator = new SprintOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot);
|
|
126
|
+
const batchOrchestrator = new BatchOrchestrator(db, emitter, sprintService, scopeService, workflowEngine, config.projectRoot);
|
|
127
|
+
const readinessService = new ReadinessService(scopeService, gateService, workflowEngine, config.projectRoot);
|
|
128
|
+
const workflowService = new WorkflowService(config.configDir, workflowEngine, config.scopesDir, getDefaultConfigPath());
|
|
129
|
+
workflowService.setSocketServer(emitter);
|
|
130
|
+
|
|
131
|
+
// Ensure engine reflects active config (may differ from bundled default)
|
|
132
|
+
workflowEngine.reload(workflowService.getActive());
|
|
133
|
+
const gitService = new GitService(config.projectRoot, scopeCache);
|
|
134
|
+
const githubService = new GitHubService(config.projectRoot);
|
|
135
|
+
const telemetryService = new TelemetryService(db, config.telemetry, config.projectName, config.projectRoot);
|
|
136
|
+
|
|
137
|
+
// Wire active-group guard
|
|
138
|
+
scopeService.setActiveGroupCheck((scopeId) => sprintService.getActiveGroupForScope(scopeId));
|
|
139
|
+
|
|
140
|
+
// Wire event inference (Fix 8: diagnostic log lines match index.ts)
|
|
141
|
+
eventService.onIngest((eventType, scopeId, data) => {
|
|
142
|
+
if (eventType === 'SESSION_START' && typeof data.dispatch_id === 'string' && typeof data.pid === 'number') {
|
|
143
|
+
linkPidToDispatch(db, data.dispatch_id, data.pid);
|
|
144
|
+
log.debug('Linked PID to dispatch', { pid: data.pid, dispatch_id: data.dispatch_id });
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
if (eventType === 'SCOPE_GATE_LIFTED' && scopeId != null) {
|
|
148
|
+
const id = Number(scopeId);
|
|
149
|
+
if (!isNaN(id) && id > 0) {
|
|
150
|
+
resolveActiveDispatchesForScope(db, emitter, id, 'completed');
|
|
151
|
+
log.debug('Resolved dispatches for scope gate lift', { scope_id: id });
|
|
152
|
+
}
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
if (eventType === 'SESSION_END') {
|
|
156
|
+
const outcome = data.normal_exit === true ? 'completed' : 'abandoned';
|
|
157
|
+
let resolvedIds: string[] = [];
|
|
158
|
+
if (typeof data.dispatch_id === 'string') {
|
|
159
|
+
resolvedIds = resolveDispatchesByDispatchId(db, emitter, data.dispatch_id, outcome);
|
|
160
|
+
}
|
|
161
|
+
if (resolvedIds.length === 0 && typeof data.pid === 'number') {
|
|
162
|
+
resolvedIds = resolveDispatchesByPid(db, emitter, data.pid, outcome);
|
|
163
|
+
}
|
|
164
|
+
if (resolvedIds.length > 0) log.info('Session resolved', { count: resolvedIds.length, outcome });
|
|
165
|
+
// For abandoned dispatches, immediately try auto-revert so the scope
|
|
166
|
+
// returns to its pre-dispatch status without requiring user interaction
|
|
167
|
+
if (outcome === 'abandoned') {
|
|
168
|
+
for (const eventId of resolvedIds) {
|
|
169
|
+
tryAutoRevertAndClear(db, emitter, scopeService, workflowEngine, eventId);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
if (resolvedIds.length > 0) batchOrchestrator.resolveStaleBatches();
|
|
173
|
+
return;
|
|
174
|
+
}
|
|
175
|
+
// Status inference
|
|
176
|
+
if (scopeId == null) return;
|
|
177
|
+
const id = Number(scopeId);
|
|
178
|
+
if (isNaN(id) || id <= 0) return;
|
|
179
|
+
const current = scopeService.getById(id);
|
|
180
|
+
if (current?.status === 'icebox') return;
|
|
181
|
+
const currentStatus = current?.status ?? '';
|
|
182
|
+
const result = workflowEngine.inferStatus(eventType, currentStatus, data);
|
|
183
|
+
if (result === null) return;
|
|
184
|
+
if (typeof result === 'object' && 'dispatchResolution' in result) {
|
|
185
|
+
resolveActiveDispatchesForScope(db, emitter, id, result.resolution as 'completed' | 'failed');
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
scopeService.updateStatus(id, result, 'event');
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
// Wire status change callbacks
|
|
192
|
+
scopeService.onStatusChange((scopeId, newStatus) => {
|
|
193
|
+
if (newStatus === 'dev') sprintOrchestrator.onScopeReachedDev(scopeId);
|
|
194
|
+
batchOrchestrator.onScopeStatusChanged(scopeId, newStatus);
|
|
195
|
+
});
|
|
196
|
+
scopeService.onStatusChange((scopeId, newStatus) => {
|
|
197
|
+
if (workflowEngine.isTerminalStatus(newStatus)) {
|
|
198
|
+
resolveActiveDispatchesForScope(db, emitter, scopeId, 'completed');
|
|
199
|
+
}
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
// Load scopes from filesystem and reconcile directory mismatches
|
|
203
|
+
const scopeCount = scopeService.syncFromFilesystem();
|
|
204
|
+
const reconciled = scopeService.reconcileDirectories();
|
|
205
|
+
if (reconciled > 0) log.info('Reconciled scope directories', { id: projectId, count: reconciled });
|
|
206
|
+
|
|
207
|
+
// Start watchers
|
|
208
|
+
const scopeWatcher = startScopeWatcher(config.scopesDir, scopeService);
|
|
209
|
+
const eventWatcher = startEventWatcher(config.eventsDir, eventService);
|
|
210
|
+
|
|
211
|
+
// Write iTerm2 dispatch profiles (Fix 2 + Fix 5: per-project prefix)
|
|
212
|
+
ensureDynamicProfiles(workflowEngine, config.terminal.profilePrefix);
|
|
213
|
+
|
|
214
|
+
// Recover active sprints/batches
|
|
215
|
+
await sprintOrchestrator.recoverActiveSprints();
|
|
216
|
+
await batchOrchestrator.recoverActiveBatches();
|
|
217
|
+
|
|
218
|
+
// Resolve stale batches on startup (Fix 6: catches stuck dispatches from previous runs)
|
|
219
|
+
const staleBatchesResolved = batchOrchestrator.resolveStaleBatches();
|
|
220
|
+
if (staleBatchesResolved > 0) log.info('Resolved stale batches', { count: staleBatchesResolved });
|
|
221
|
+
|
|
222
|
+
// Resolve stale dispatches
|
|
223
|
+
resolveStaleDispatches(db, emitter, scopeService, workflowEngine);
|
|
224
|
+
|
|
225
|
+
// Initial session sync + legacy purge (Fix 7)
|
|
226
|
+
syncClaudeSessionsToDB(db, scopeService, config.projectRoot).then((count) => {
|
|
227
|
+
if (count > 0) log.info('Synced sessions', { id: projectId, count });
|
|
228
|
+
const purged = db.prepare("DELETE FROM sessions WHERE action IS NULL AND id LIKE 'claude-%'").run();
|
|
229
|
+
if (purged.changes > 0) log.info('Purged legacy session rows', { count: purged.changes });
|
|
230
|
+
if (telemetryService.enabled) {
|
|
231
|
+
telemetryService.uploadChangedSessions().catch(() => {});
|
|
232
|
+
}
|
|
233
|
+
}).catch(err => log.error('Session sync failed', { error: err.message }));
|
|
234
|
+
|
|
235
|
+
// Start periodic intervals
|
|
236
|
+
const intervals: ReturnType<typeof setInterval>[] = [];
|
|
237
|
+
|
|
238
|
+
// Fix 11: periodic batch recovery (two-phase completion B-1)
|
|
239
|
+
intervals.push(setInterval(() => {
|
|
240
|
+
batchOrchestrator.recoverActiveBatches().catch(err => log.error('Batch recovery failed', { error: err.message }));
|
|
241
|
+
}, 30_000));
|
|
242
|
+
|
|
243
|
+
intervals.push(setInterval(() => {
|
|
244
|
+
batchOrchestrator.resolveStaleBatches();
|
|
245
|
+
}, 30_000));
|
|
246
|
+
|
|
247
|
+
intervals.push(setInterval(() => {
|
|
248
|
+
resolveStaleDispatches(db, emitter, scopeService, workflowEngine);
|
|
249
|
+
}, 30_000));
|
|
250
|
+
|
|
251
|
+
intervals.push(setInterval(async () => {
|
|
252
|
+
const count = await syncClaudeSessionsToDB(db, scopeService, config.projectRoot);
|
|
253
|
+
if (count > 0) emitter.emit('session:updated', { type: 'resync', count });
|
|
254
|
+
if (telemetryService.enabled) {
|
|
255
|
+
telemetryService.uploadChangedSessions().catch(() => {});
|
|
256
|
+
}
|
|
257
|
+
}, 5 * 60_000));
|
|
258
|
+
|
|
259
|
+
let lastGitHash = '';
|
|
260
|
+
intervals.push(setInterval(async () => {
|
|
261
|
+
try {
|
|
262
|
+
const hash = await gitService.getStatusHash();
|
|
263
|
+
if (lastGitHash && hash !== lastGitHash) {
|
|
264
|
+
gitService.clearCache();
|
|
265
|
+
emitter.emit('git:status:changed');
|
|
266
|
+
}
|
|
267
|
+
lastGitHash = hash;
|
|
268
|
+
} catch { /* ok */ }
|
|
269
|
+
}, 10_000));
|
|
270
|
+
|
|
271
|
+
log.info('Project ready', { id: projectId, scopes: scopeCount });
|
|
272
|
+
|
|
273
|
+
const ctx: ProjectContext = {
|
|
274
|
+
id: projectId,
|
|
275
|
+
config,
|
|
276
|
+
db,
|
|
277
|
+
workflowEngine,
|
|
278
|
+
emitter,
|
|
279
|
+
scopeCache,
|
|
280
|
+
scopeService,
|
|
281
|
+
eventService,
|
|
282
|
+
gateService,
|
|
283
|
+
deployService,
|
|
284
|
+
sprintService,
|
|
285
|
+
sprintOrchestrator,
|
|
286
|
+
batchOrchestrator,
|
|
287
|
+
readinessService,
|
|
288
|
+
workflowService,
|
|
289
|
+
gitService,
|
|
290
|
+
githubService,
|
|
291
|
+
telemetryService,
|
|
292
|
+
scopeWatcher,
|
|
293
|
+
eventWatcher,
|
|
294
|
+
intervals,
|
|
295
|
+
status: 'active',
|
|
296
|
+
|
|
297
|
+
async shutdown() {
|
|
298
|
+
log.info('Shutting down project context', { id: projectId });
|
|
299
|
+
for (const interval of intervals) clearInterval(interval);
|
|
300
|
+
intervals.length = 0;
|
|
301
|
+
try { await scopeWatcher.close(); } catch (e) { log.error('Scope watcher close failed', { id: projectId, error: String(e) }); }
|
|
302
|
+
try { await eventWatcher.close(); } catch (e) { log.error('Event watcher close failed', { id: projectId, error: String(e) }); }
|
|
303
|
+
try { db.close(); } catch (e) { log.error('DB close failed', { id: projectId, error: String(e) }); }
|
|
304
|
+
ctx.status = 'offline';
|
|
305
|
+
},
|
|
306
|
+
};
|
|
307
|
+
|
|
308
|
+
return ctx;
|
|
309
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import type { Server } from 'socket.io';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* A project-scoped Socket.io emitter.
|
|
5
|
+
*
|
|
6
|
+
* Services use this instead of the raw Socket.io Server so that events
|
|
7
|
+
* are automatically scoped to the correct project room. Events are emitted
|
|
8
|
+
* to both the project-specific room (`project:{id}`) and the aggregate
|
|
9
|
+
* room (`all-projects`) so that the All Projects dashboard view receives
|
|
10
|
+
* updates from every project.
|
|
11
|
+
*
|
|
12
|
+
* The emit() signature matches Socket.io's Server.emit() so existing
|
|
13
|
+
* service code requires only a type change (Server → ProjectEmitter).
|
|
14
|
+
*/
|
|
15
|
+
export class ProjectEmitter {
|
|
16
|
+
constructor(
|
|
17
|
+
private io: Server,
|
|
18
|
+
private projectId: string,
|
|
19
|
+
) {}
|
|
20
|
+
|
|
21
|
+
/** Emit an event to this project's room and the all-projects room. */
|
|
22
|
+
emit(event: string, ...args: unknown[]): boolean {
|
|
23
|
+
// Inject project_id into the first data argument if it's an object
|
|
24
|
+
const enrichedArgs = args.map((arg, i) => {
|
|
25
|
+
if (i === 0 && arg !== null && typeof arg === 'object' && !Array.isArray(arg)) {
|
|
26
|
+
return { ...(arg as Record<string, unknown>), project_id: this.projectId };
|
|
27
|
+
}
|
|
28
|
+
return arg;
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
this.io.to(`project:${this.projectId}`).emit(event, ...enrichedArgs);
|
|
32
|
+
this.io.to('all-projects').emit(event, ...enrichedArgs);
|
|
33
|
+
return true;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/** Get the underlying Socket.io server (for operations that need it, e.g., connection handling). */
|
|
37
|
+
getServer(): Server {
|
|
38
|
+
return this.io;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/** Get the project ID this emitter is scoped to. */
|
|
42
|
+
getProjectId(): string {
|
|
43
|
+
return this.projectId;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Type alias used by services and routes for the project-scoped emitter.
|
|
49
|
+
*/
|
|
50
|
+
export type Emitter = ProjectEmitter;
|