gsd-pi 2.44.0-dev.d25d507 → 2.45.0-dev.1afbdaa
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/dist/help-text.js +1 -1
- package/dist/loader.js +34 -0
- package/dist/resources/extensions/gsd/activity-log.js +7 -0
- package/dist/resources/extensions/gsd/auto/infra-errors.js +3 -0
- package/dist/resources/extensions/gsd/auto/phases.js +63 -77
- package/dist/resources/extensions/gsd/auto/run-unit.js +6 -3
- package/dist/resources/extensions/gsd/auto/session.js +0 -11
- package/dist/resources/extensions/gsd/auto-artifact-paths.js +112 -0
- package/dist/resources/extensions/gsd/auto-post-unit.js +25 -96
- package/dist/resources/extensions/gsd/auto-prompts.js +24 -1
- package/dist/resources/extensions/gsd/auto-start.js +23 -5
- package/dist/resources/extensions/gsd/auto-timers.js +57 -3
- package/dist/resources/extensions/gsd/auto-worktree-sync.js +4 -0
- package/dist/resources/extensions/gsd/auto-worktree.js +14 -10
- package/dist/resources/extensions/gsd/auto.js +42 -60
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +170 -11
- package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +18 -0
- package/dist/resources/extensions/gsd/commands/catalog.js +7 -1
- package/dist/resources/extensions/gsd/commands/context.js +0 -4
- package/dist/resources/extensions/gsd/commands/handlers/core.js +2 -0
- package/dist/resources/extensions/gsd/commands/handlers/ops.js +10 -0
- package/dist/resources/extensions/gsd/commands/handlers/parallel.js +1 -1
- package/dist/resources/extensions/gsd/commands-mcp-status.js +187 -0
- package/dist/resources/extensions/gsd/crash-recovery.js +2 -4
- package/dist/resources/extensions/gsd/dashboard-overlay.js +0 -44
- package/dist/resources/extensions/gsd/db-writer.js +40 -22
- package/dist/resources/extensions/gsd/doctor-checks.js +167 -2
- package/dist/resources/extensions/gsd/doctor.js +13 -3
- package/dist/resources/extensions/gsd/git-service.js +8 -3
- package/dist/resources/extensions/gsd/gsd-db.js +28 -4
- package/dist/resources/extensions/gsd/guided-flow.js +1 -2
- package/dist/resources/extensions/gsd/markdown-renderer.js +1 -1
- package/dist/resources/extensions/gsd/parallel-merge.js +1 -1
- package/dist/resources/extensions/gsd/parallel-orchestrator.js +5 -18
- package/dist/resources/extensions/gsd/preferences-types.js +2 -2
- package/dist/resources/extensions/gsd/preferences.js +8 -4
- package/dist/resources/extensions/gsd/prompts/complete-milestone.md +21 -10
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +10 -23
- package/dist/resources/extensions/gsd/prompts/discuss.md +2 -2
- package/dist/resources/extensions/gsd/prompts/execute-task.md +5 -15
- package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/guided-plan-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/guided-research-slice.md +1 -1
- package/dist/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/dist/resources/extensions/gsd/prompts/plan-slice.md +5 -3
- package/dist/resources/extensions/gsd/prompts/queue.md +2 -2
- package/dist/resources/extensions/gsd/prompts/quick-task.md +2 -0
- package/dist/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
- package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -6
- package/dist/resources/extensions/gsd/prompts/replan-slice.md +3 -14
- package/dist/resources/extensions/gsd/prompts/research-slice.md +3 -3
- package/dist/resources/extensions/gsd/prompts/rethink.md +83 -0
- package/dist/resources/extensions/gsd/prompts/system.md +1 -1
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +7 -37
- package/dist/resources/extensions/gsd/provider-error-pause.js +7 -0
- package/dist/resources/extensions/gsd/repo-identity.js +45 -7
- package/dist/resources/extensions/gsd/rethink.js +115 -0
- package/dist/resources/extensions/gsd/session-lock.js +1 -3
- package/dist/resources/extensions/gsd/state.js +48 -3
- package/dist/resources/extensions/gsd/sync-lock.js +89 -0
- package/dist/resources/extensions/gsd/tools/complete-milestone.js +61 -11
- package/dist/resources/extensions/gsd/tools/complete-slice.js +56 -11
- package/dist/resources/extensions/gsd/tools/complete-task.js +50 -2
- package/dist/resources/extensions/gsd/tools/plan-milestone.js +37 -1
- package/dist/resources/extensions/gsd/tools/plan-slice.js +31 -1
- package/dist/resources/extensions/gsd/tools/plan-task.js +28 -1
- package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +32 -2
- package/dist/resources/extensions/gsd/tools/reopen-slice.js +86 -0
- package/dist/resources/extensions/gsd/tools/reopen-task.js +90 -0
- package/dist/resources/extensions/gsd/tools/replan-slice.js +34 -2
- package/dist/resources/extensions/gsd/tools/validate-milestone.js +88 -0
- package/dist/resources/extensions/gsd/unit-ownership.js +85 -0
- package/dist/resources/extensions/gsd/workflow-events.js +102 -0
- package/dist/resources/extensions/gsd/workflow-logger.js +193 -0
- package/dist/resources/extensions/gsd/workflow-manifest.js +244 -0
- package/dist/resources/extensions/gsd/workflow-migration.js +280 -0
- package/dist/resources/extensions/gsd/workflow-projections.js +373 -0
- package/dist/resources/extensions/gsd/workflow-reconcile.js +411 -0
- package/dist/resources/extensions/gsd/worktree-manager.js +34 -3
- package/dist/resources/extensions/gsd/worktree-resolver.js +43 -0
- package/dist/resources/extensions/gsd/write-intercept.js +84 -0
- package/dist/resources/extensions/mcp-client/index.js +14 -0
- package/dist/resources/extensions/voice/index.js +11 -16
- package/dist/resources/extensions/voice/linux-ready.js +67 -0
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
- package/dist/web/standalone/.next/build-manifest.json +3 -3
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/react-loadable-manifest.json +2 -2
- package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.html +2 -2
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dist/web/standalone/.next/server/app/page.js +1 -1
- package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +2 -2
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/dist/web/standalone/.next/static/chunks/4024.11ca5c01938e5948.js +9 -0
- package/dist/web/standalone/.next/static/chunks/{3721.bf31263de6d5fa46.js → 485.243af25f0cdf50d6.js} +2 -2
- package/dist/web/standalone/.next/static/chunks/app/{page-b9367c5ae13b99c6.js → page-6654a8cca61a3d1c.js} +1 -1
- package/dist/web/standalone/.next/static/chunks/webpack-0a4cd455ec4197d2.js +1 -0
- package/dist/web/standalone/.next/static/css/dd4ae3f58ac9b600.css +1 -0
- package/package.json +2 -1
- package/packages/native/dist/stream-process/index.js +2 -2
- package/packages/native/src/__tests__/stream-process.test.mjs +34 -0
- package/packages/native/src/stream-process/index.ts +2 -2
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +3 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js +15 -1
- package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js +2 -0
- package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.d.ts +4 -0
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.js +10 -5
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.test.js +185 -0
- package/packages/pi-coding-agent/dist/core/lifecycle-hooks.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.d.ts +15 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.js +41 -0
- package/packages/pi-coding-agent/dist/core/local-model-check.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +239 -10
- package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts +13 -1
- package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/model-registry.js +40 -3
- package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/package-commands.test.js +206 -195
- package/packages/pi-coding-agent/dist/core/package-commands.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +3 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.js +6 -0
- package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/main.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/main.js +17 -0
- package/packages/pi-coding-agent/dist/main.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.js +32 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/timestamp.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +3 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +8 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +12 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.d.ts +15 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.js +40 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/timestamp.js.map +1 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +4 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts +5 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js +13 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/user-message.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +17 -8
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +7 -3
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/auth-storage.ts +15 -1
- package/packages/pi-coding-agent/src/core/compaction-orchestrator.ts +2 -0
- package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -1
- package/packages/pi-coding-agent/src/core/lifecycle-hooks.test.ts +227 -0
- package/packages/pi-coding-agent/src/core/lifecycle-hooks.ts +11 -5
- package/packages/pi-coding-agent/src/core/local-model-check.ts +45 -0
- package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +297 -11
- package/packages/pi-coding-agent/src/core/model-registry.ts +51 -4
- package/packages/pi-coding-agent/src/core/package-commands.test.ts +227 -205
- package/packages/pi-coding-agent/src/core/settings-manager.ts +9 -0
- package/packages/pi-coding-agent/src/main.ts +19 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/timestamp.test.ts +38 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +10 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +15 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/timestamp.ts +48 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +3 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/user-message.ts +18 -3
- package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +16 -7
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +8 -1
- package/pkg/package.json +1 -1
- package/src/resources/extensions/gsd/activity-log.ts +1 -0
- package/src/resources/extensions/gsd/auto/infra-errors.ts +3 -0
- package/src/resources/extensions/gsd/auto/loop-deps.ts +0 -19
- package/src/resources/extensions/gsd/auto/phases.ts +69 -91
- package/src/resources/extensions/gsd/auto/run-unit.ts +6 -3
- package/src/resources/extensions/gsd/auto/session.ts +0 -18
- package/src/resources/extensions/gsd/auto-artifact-paths.ts +131 -0
- package/src/resources/extensions/gsd/auto-dashboard.ts +0 -1
- package/src/resources/extensions/gsd/auto-post-unit.ts +25 -106
- package/src/resources/extensions/gsd/auto-prompts.ts +24 -1
- package/src/resources/extensions/gsd/auto-start.ts +26 -5
- package/src/resources/extensions/gsd/auto-timers.ts +64 -3
- package/src/resources/extensions/gsd/auto-worktree-sync.ts +5 -0
- package/src/resources/extensions/gsd/auto-worktree.ts +17 -11
- package/src/resources/extensions/gsd/auto.ts +44 -86
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +162 -11
- package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +22 -0
- package/src/resources/extensions/gsd/commands/catalog.ts +7 -1
- package/src/resources/extensions/gsd/commands/context.ts +0 -5
- package/src/resources/extensions/gsd/commands/handlers/core.ts +2 -0
- package/src/resources/extensions/gsd/commands/handlers/ops.ts +10 -0
- package/src/resources/extensions/gsd/commands/handlers/parallel.ts +1 -1
- package/src/resources/extensions/gsd/commands-mcp-status.ts +247 -0
- package/src/resources/extensions/gsd/crash-recovery.ts +1 -5
- package/src/resources/extensions/gsd/dashboard-overlay.ts +0 -50
- package/src/resources/extensions/gsd/db-writer.ts +41 -27
- package/src/resources/extensions/gsd/doctor-checks.ts +180 -2
- package/src/resources/extensions/gsd/doctor-types.ts +7 -1
- package/src/resources/extensions/gsd/doctor.ts +13 -4
- package/src/resources/extensions/gsd/git-service.ts +6 -2
- package/src/resources/extensions/gsd/gsd-db.ts +32 -4
- package/src/resources/extensions/gsd/guided-flow.ts +1 -2
- package/src/resources/extensions/gsd/journal.ts +6 -1
- package/src/resources/extensions/gsd/markdown-renderer.ts +1 -1
- package/src/resources/extensions/gsd/parallel-merge.ts +1 -1
- package/src/resources/extensions/gsd/parallel-orchestrator.ts +5 -21
- package/src/resources/extensions/gsd/preferences-types.ts +2 -2
- package/src/resources/extensions/gsd/preferences.ts +7 -3
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +21 -10
- package/src/resources/extensions/gsd/prompts/complete-slice.md +10 -23
- package/src/resources/extensions/gsd/prompts/discuss.md +2 -2
- package/src/resources/extensions/gsd/prompts/execute-task.md +5 -15
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/guided-plan-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/guided-research-slice.md +1 -1
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
- package/src/resources/extensions/gsd/prompts/plan-slice.md +5 -3
- package/src/resources/extensions/gsd/prompts/queue.md +2 -2
- package/src/resources/extensions/gsd/prompts/quick-task.md +2 -0
- package/src/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +6 -6
- package/src/resources/extensions/gsd/prompts/replan-slice.md +3 -14
- package/src/resources/extensions/gsd/prompts/research-slice.md +3 -3
- package/src/resources/extensions/gsd/prompts/rethink.md +83 -0
- package/src/resources/extensions/gsd/prompts/system.md +1 -1
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +7 -37
- package/src/resources/extensions/gsd/provider-error-pause.ts +9 -0
- package/src/resources/extensions/gsd/repo-identity.ts +46 -7
- package/src/resources/extensions/gsd/rethink.ts +154 -0
- package/src/resources/extensions/gsd/session-lock.ts +0 -4
- package/src/resources/extensions/gsd/state.ts +49 -1
- package/src/resources/extensions/gsd/sync-lock.ts +94 -0
- package/src/resources/extensions/gsd/tests/auto-lock-creation.test.ts +5 -13
- package/src/resources/extensions/gsd/tests/auto-loop.test.ts +6 -10
- package/src/resources/extensions/gsd/tests/auto-pr-bugs.test.ts +88 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +96 -0
- package/src/resources/extensions/gsd/tests/complete-slice.test.ts +264 -228
- package/src/resources/extensions/gsd/tests/complete-task.test.ts +317 -250
- package/src/resources/extensions/gsd/tests/completed-units-metrics-sync.test.ts +114 -0
- package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +2 -8
- package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +0 -3
- package/src/resources/extensions/gsd/tests/db-writer.test.ts +79 -0
- package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +121 -0
- package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +60 -0
- package/src/resources/extensions/gsd/tests/est-annotation-timeout.test.ts +120 -0
- package/src/resources/extensions/gsd/tests/gsd-db.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/infra-error.test.ts +20 -2
- package/src/resources/extensions/gsd/tests/inherited-repo-home-dir.test.ts +121 -0
- package/src/resources/extensions/gsd/tests/integration-proof.test.ts +15 -24
- package/src/resources/extensions/gsd/tests/journal-integration.test.ts +0 -3
- package/src/resources/extensions/gsd/tests/mcp-status.test.ts +103 -0
- package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +66 -0
- package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +8 -9
- package/src/resources/extensions/gsd/tests/none-mode-gates.test.ts +42 -3
- package/src/resources/extensions/gsd/tests/parallel-budget-atomicity.test.ts +0 -1
- package/src/resources/extensions/gsd/tests/parallel-crash-recovery.test.ts +0 -7
- package/src/resources/extensions/gsd/tests/parallel-merge.test.ts +7 -8
- package/src/resources/extensions/gsd/tests/parallel-orchestration.test.ts +20 -24
- package/src/resources/extensions/gsd/tests/parallel-worker-monitoring.test.ts +0 -2
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +9 -6
- package/src/resources/extensions/gsd/tests/post-mutation-hook.test.ts +171 -0
- package/src/resources/extensions/gsd/tests/preferences.test.ts +7 -9
- package/src/resources/extensions/gsd/tests/projection-regression.test.ts +174 -0
- package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +26 -21
- package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +176 -0
- package/src/resources/extensions/gsd/tests/reopen-slice.test.ts +155 -0
- package/src/resources/extensions/gsd/tests/reopen-task.test.ts +165 -0
- package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +1 -4
- package/src/resources/extensions/gsd/tests/stop-auto-merge-back.test.ts +67 -0
- package/src/resources/extensions/gsd/tests/stop-auto-remote.test.ts +2 -3
- package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +108 -0
- package/src/resources/extensions/gsd/tests/sync-lock.test.ts +122 -0
- package/src/resources/extensions/gsd/tests/terminated-transient.test.ts +49 -0
- package/src/resources/extensions/gsd/tests/tool-naming.test.ts +2 -1
- package/src/resources/extensions/gsd/tests/unit-ownership.test.ts +175 -0
- package/src/resources/extensions/gsd/tests/workflow-events.test.ts +205 -0
- package/src/resources/extensions/gsd/tests/workflow-logger.test.ts +275 -0
- package/src/resources/extensions/gsd/tests/workflow-manifest.test.ts +186 -0
- package/src/resources/extensions/gsd/tests/workflow-projections.test.ts +171 -0
- package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +220 -0
- package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +65 -0
- package/src/resources/extensions/gsd/tests/write-intercept.test.ts +76 -0
- package/src/resources/extensions/gsd/tools/complete-milestone.ts +74 -11
- package/src/resources/extensions/gsd/tools/complete-slice.ts +68 -11
- package/src/resources/extensions/gsd/tools/complete-task.ts +63 -1
- package/src/resources/extensions/gsd/tools/plan-milestone.ts +45 -0
- package/src/resources/extensions/gsd/tools/plan-slice.ts +40 -0
- package/src/resources/extensions/gsd/tools/plan-task.ts +37 -1
- package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +39 -1
- package/src/resources/extensions/gsd/tools/reopen-slice.ts +125 -0
- package/src/resources/extensions/gsd/tools/reopen-task.ts +129 -0
- package/src/resources/extensions/gsd/tools/replan-slice.ts +41 -1
- package/src/resources/extensions/gsd/tools/validate-milestone.ts +127 -0
- package/src/resources/extensions/gsd/types.ts +8 -0
- package/src/resources/extensions/gsd/unit-ownership.ts +104 -0
- package/src/resources/extensions/gsd/workflow-events.ts +154 -0
- package/src/resources/extensions/gsd/workflow-logger.ts +243 -0
- package/src/resources/extensions/gsd/workflow-manifest.ts +334 -0
- package/src/resources/extensions/gsd/workflow-migration.ts +345 -0
- package/src/resources/extensions/gsd/workflow-projections.ts +425 -0
- package/src/resources/extensions/gsd/workflow-reconcile.ts +503 -0
- package/src/resources/extensions/gsd/worktree-manager.ts +41 -5
- package/src/resources/extensions/gsd/worktree-resolver.ts +44 -0
- package/src/resources/extensions/gsd/write-intercept.ts +90 -0
- package/src/resources/extensions/mcp-client/index.ts +20 -0
- package/src/resources/extensions/voice/index.ts +11 -21
- package/src/resources/extensions/voice/linux-ready.ts +87 -0
- package/src/resources/extensions/voice/tests/linux-ready.test.ts +124 -0
- package/dist/web/standalone/.next/static/chunks/4024.0de81b543b28b9fe.js +0 -9
- package/dist/web/standalone/.next/static/chunks/webpack-9014b5adb127a98a.js +0 -1
- package/dist/web/standalone/.next/static/css/8a727f372cf53002.css +0 -1
- /package/dist/web/standalone/.next/static/{tokoGmfkYfWf1_Yl_Gz7i → j-BskPs0nxxPeYY-bSrab}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{tokoGmfkYfWf1_Yl_Gz7i → j-BskPs0nxxPeYY-bSrab}/_ssgManifest.js +0 -0
|
@@ -11,6 +11,9 @@ import {
|
|
|
11
11
|
} from "../gsd-db.js";
|
|
12
12
|
import { invalidateStateCache } from "../state.js";
|
|
13
13
|
import { renderPlanFromDb, renderReplanFromDb } from "../markdown-renderer.js";
|
|
14
|
+
import { renderAllProjections } from "../workflow-projections.js";
|
|
15
|
+
import { writeManifest } from "../workflow-manifest.js";
|
|
16
|
+
import { appendEvent } from "../workflow-events.js";
|
|
14
17
|
|
|
15
18
|
export interface ReplanSliceTaskInput {
|
|
16
19
|
taskId: string;
|
|
@@ -21,6 +24,7 @@ export interface ReplanSliceTaskInput {
|
|
|
21
24
|
verify: string;
|
|
22
25
|
inputs: string[];
|
|
23
26
|
expectedOutput: string[];
|
|
27
|
+
fullPlanMd?: string;
|
|
24
28
|
}
|
|
25
29
|
|
|
26
30
|
export interface ReplanSliceParams {
|
|
@@ -31,6 +35,10 @@ export interface ReplanSliceParams {
|
|
|
31
35
|
whatChanged: string;
|
|
32
36
|
updatedTasks: ReplanSliceTaskInput[];
|
|
33
37
|
removedTaskIds: string[];
|
|
38
|
+
/** Optional caller-provided identity for audit trail */
|
|
39
|
+
actorName?: string;
|
|
40
|
+
/** Optional caller-provided reason this action was triggered */
|
|
41
|
+
triggerReason?: string;
|
|
34
42
|
}
|
|
35
43
|
|
|
36
44
|
export interface ReplanSliceResult {
|
|
@@ -82,11 +90,23 @@ export async function handleReplanSlice(
|
|
|
82
90
|
return { error: `validation failed: ${(err as Error).message}` };
|
|
83
91
|
}
|
|
84
92
|
|
|
85
|
-
// ── Verify parent slice exists
|
|
93
|
+
// ── Verify parent slice exists and is not closed ─────────────────
|
|
86
94
|
const parentSlice = getSlice(params.milestoneId, params.sliceId);
|
|
87
95
|
if (!parentSlice) {
|
|
88
96
|
return { error: `missing parent slice: ${params.milestoneId}/${params.sliceId}` };
|
|
89
97
|
}
|
|
98
|
+
if (parentSlice.status === "complete" || parentSlice.status === "done") {
|
|
99
|
+
return { error: `cannot replan a closed slice: ${params.sliceId} (status: ${parentSlice.status})` };
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// ── Verify blocker task exists and is complete ────────────────────
|
|
103
|
+
const blockerTask = getTask(params.milestoneId, params.sliceId, params.blockerTaskId);
|
|
104
|
+
if (!blockerTask) {
|
|
105
|
+
return { error: `blockerTaskId not found: ${params.milestoneId}/${params.sliceId}/${params.blockerTaskId}` };
|
|
106
|
+
}
|
|
107
|
+
if (blockerTask.status !== "complete" && blockerTask.status !== "done") {
|
|
108
|
+
return { error: `blockerTaskId ${params.blockerTaskId} is not complete (status: ${blockerTask.status}) — the blocker task must be finished before a replan is triggered` };
|
|
109
|
+
}
|
|
90
110
|
|
|
91
111
|
// ── Structural enforcement ────────────────────────────────────────
|
|
92
112
|
const existingTasks = getSliceTasks(params.milestoneId, params.sliceId);
|
|
@@ -136,6 +156,7 @@ export async function handleReplanSlice(
|
|
|
136
156
|
verify: updatedTask.verify || "",
|
|
137
157
|
inputs: updatedTask.inputs || [],
|
|
138
158
|
expectedOutput: updatedTask.expectedOutput || [],
|
|
159
|
+
fullPlanMd: updatedTask.fullPlanMd,
|
|
139
160
|
});
|
|
140
161
|
} else {
|
|
141
162
|
// Insert new task then set planning fields
|
|
@@ -154,6 +175,7 @@ export async function handleReplanSlice(
|
|
|
154
175
|
verify: updatedTask.verify || "",
|
|
155
176
|
inputs: updatedTask.inputs || [],
|
|
156
177
|
expectedOutput: updatedTask.expectedOutput || [],
|
|
178
|
+
fullPlanMd: updatedTask.fullPlanMd,
|
|
157
179
|
});
|
|
158
180
|
}
|
|
159
181
|
}
|
|
@@ -180,6 +202,24 @@ export async function handleReplanSlice(
|
|
|
180
202
|
invalidateStateCache();
|
|
181
203
|
clearParseCache();
|
|
182
204
|
|
|
205
|
+
// ── Post-mutation hook: projections, manifest, event log ─────
|
|
206
|
+
try {
|
|
207
|
+
await renderAllProjections(basePath, params.milestoneId);
|
|
208
|
+
writeManifest(basePath);
|
|
209
|
+
appendEvent(basePath, {
|
|
210
|
+
cmd: "replan-slice",
|
|
211
|
+
params: { milestoneId: params.milestoneId, sliceId: params.sliceId, blockerTaskId: params.blockerTaskId },
|
|
212
|
+
ts: new Date().toISOString(),
|
|
213
|
+
actor: "agent",
|
|
214
|
+
actor_name: params.actorName,
|
|
215
|
+
trigger_reason: params.triggerReason,
|
|
216
|
+
});
|
|
217
|
+
} catch (hookErr) {
|
|
218
|
+
process.stderr.write(
|
|
219
|
+
`gsd: replan-slice post-mutation hook warning: ${(hookErr as Error).message}\n`,
|
|
220
|
+
);
|
|
221
|
+
}
|
|
222
|
+
|
|
183
223
|
return {
|
|
184
224
|
milestoneId: params.milestoneId,
|
|
185
225
|
sliceId: params.sliceId,
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* validate-milestone handler — the core operation behind gsd_validate_milestone.
|
|
3
|
+
*
|
|
4
|
+
* Persists milestone validation results to the assessments table,
|
|
5
|
+
* renders VALIDATION.md to disk, and invalidates caches.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { join } from "node:path";
|
|
9
|
+
|
|
10
|
+
import {
|
|
11
|
+
transaction,
|
|
12
|
+
_getAdapter,
|
|
13
|
+
} from "../gsd-db.js";
|
|
14
|
+
import { resolveMilestonePath, clearPathCache } from "../paths.js";
|
|
15
|
+
import { saveFile, clearParseCache } from "../files.js";
|
|
16
|
+
import { invalidateStateCache } from "../state.js";
|
|
17
|
+
|
|
18
|
+
export interface ValidateMilestoneParams {
|
|
19
|
+
milestoneId: string;
|
|
20
|
+
verdict: "pass" | "needs-attention" | "needs-remediation";
|
|
21
|
+
remediationRound: number;
|
|
22
|
+
successCriteriaChecklist: string;
|
|
23
|
+
sliceDeliveryAudit: string;
|
|
24
|
+
crossSliceIntegration: string;
|
|
25
|
+
requirementCoverage: string;
|
|
26
|
+
verdictRationale: string;
|
|
27
|
+
remediationPlan?: string;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export interface ValidateMilestoneResult {
|
|
31
|
+
milestoneId: string;
|
|
32
|
+
verdict: string;
|
|
33
|
+
validationPath: string;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function renderValidationMarkdown(params: ValidateMilestoneParams): string {
|
|
37
|
+
let md = `---
|
|
38
|
+
verdict: ${params.verdict}
|
|
39
|
+
remediation_round: ${params.remediationRound}
|
|
40
|
+
---
|
|
41
|
+
|
|
42
|
+
# Milestone Validation: ${params.milestoneId}
|
|
43
|
+
|
|
44
|
+
## Success Criteria Checklist
|
|
45
|
+
${params.successCriteriaChecklist}
|
|
46
|
+
|
|
47
|
+
## Slice Delivery Audit
|
|
48
|
+
${params.sliceDeliveryAudit}
|
|
49
|
+
|
|
50
|
+
## Cross-Slice Integration
|
|
51
|
+
${params.crossSliceIntegration}
|
|
52
|
+
|
|
53
|
+
## Requirement Coverage
|
|
54
|
+
${params.requirementCoverage}
|
|
55
|
+
|
|
56
|
+
## Verdict Rationale
|
|
57
|
+
${params.verdictRationale}
|
|
58
|
+
`;
|
|
59
|
+
|
|
60
|
+
if (params.verdict === "needs-remediation" && params.remediationPlan) {
|
|
61
|
+
md += `\n## Remediation Plan\n${params.remediationPlan}\n`;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return md;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export async function handleValidateMilestone(
|
|
68
|
+
params: ValidateMilestoneParams,
|
|
69
|
+
basePath: string,
|
|
70
|
+
): Promise<ValidateMilestoneResult | { error: string }> {
|
|
71
|
+
if (!params.milestoneId || typeof params.milestoneId !== "string" || params.milestoneId.trim() === "") {
|
|
72
|
+
return { error: "milestoneId is required and must be a non-empty string" };
|
|
73
|
+
}
|
|
74
|
+
const validVerdicts = ["pass", "needs-attention", "needs-remediation"];
|
|
75
|
+
if (!validVerdicts.includes(params.verdict)) {
|
|
76
|
+
return { error: `verdict must be one of: ${validVerdicts.join(", ")}` };
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// ── Filesystem render ──────────────────────────────────────────────────
|
|
80
|
+
const validationMd = renderValidationMarkdown(params);
|
|
81
|
+
|
|
82
|
+
let validationPath: string;
|
|
83
|
+
const milestoneDir = resolveMilestonePath(basePath, params.milestoneId);
|
|
84
|
+
if (milestoneDir) {
|
|
85
|
+
validationPath = join(milestoneDir, `${params.milestoneId}-VALIDATION.md`);
|
|
86
|
+
} else {
|
|
87
|
+
const gsdDir = join(basePath, ".gsd");
|
|
88
|
+
const manualDir = join(gsdDir, "milestones", params.milestoneId);
|
|
89
|
+
validationPath = join(manualDir, `${params.milestoneId}-VALIDATION.md`);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
try {
|
|
93
|
+
await saveFile(validationPath, validationMd);
|
|
94
|
+
} catch (renderErr) {
|
|
95
|
+
process.stderr.write(
|
|
96
|
+
`gsd-db: validate_milestone — disk render failed: ${(renderErr as Error).message}\n`,
|
|
97
|
+
);
|
|
98
|
+
return { error: `disk render failed: ${(renderErr as Error).message}` };
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// ── DB write — store in assessments table ──────────────────────────────
|
|
102
|
+
const validatedAt = new Date().toISOString();
|
|
103
|
+
|
|
104
|
+
transaction(() => {
|
|
105
|
+
const adapter = _getAdapter()!;
|
|
106
|
+
adapter.prepare(
|
|
107
|
+
`INSERT OR REPLACE INTO assessments (path, milestone_id, slice_id, task_id, status, scope, full_content, created_at)
|
|
108
|
+
VALUES (:path, :mid, NULL, NULL, :verdict, 'milestone-validation', :content, :created_at)`,
|
|
109
|
+
).run({
|
|
110
|
+
":path": validationPath,
|
|
111
|
+
":mid": params.milestoneId,
|
|
112
|
+
":verdict": params.verdict,
|
|
113
|
+
":content": validationMd,
|
|
114
|
+
":created_at": validatedAt,
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
invalidateStateCache();
|
|
119
|
+
clearPathCache();
|
|
120
|
+
clearParseCache();
|
|
121
|
+
|
|
122
|
+
return {
|
|
123
|
+
milestoneId: params.milestoneId,
|
|
124
|
+
verdict: params.verdict,
|
|
125
|
+
validationPath,
|
|
126
|
+
};
|
|
127
|
+
}
|
|
@@ -520,6 +520,10 @@ export interface CompleteTaskParams {
|
|
|
520
520
|
verdict: string;
|
|
521
521
|
durationMs: number;
|
|
522
522
|
}>;
|
|
523
|
+
/** Optional caller-provided identity for audit trail */
|
|
524
|
+
actorName?: string;
|
|
525
|
+
/** Optional caller-provided reason this action was triggered */
|
|
526
|
+
triggerReason?: string;
|
|
523
527
|
}
|
|
524
528
|
|
|
525
529
|
// ─── Complete Slice Params (gsd_complete_slice tool input) ───────────────
|
|
@@ -548,4 +552,8 @@ export interface CompleteSliceParams {
|
|
|
548
552
|
requires: Array<{ slice: string; provides: string }>;
|
|
549
553
|
affects: string[];
|
|
550
554
|
drillDownPaths: string[];
|
|
555
|
+
/** Optional caller-provided identity for audit trail */
|
|
556
|
+
actorName?: string;
|
|
557
|
+
/** Optional caller-provided reason this action was triggered */
|
|
558
|
+
triggerReason?: string;
|
|
551
559
|
}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
// GSD Extension — Unit Ownership
|
|
2
|
+
// Opt-in per-unit ownership claims for multi-agent safety.
|
|
3
|
+
//
|
|
4
|
+
// An agent can claim a unit (task, slice) before working on it.
|
|
5
|
+
// complete-task and complete-slice enforce ownership when claims exist.
|
|
6
|
+
// If no claim file is present, ownership is not enforced (backward compatible).
|
|
7
|
+
//
|
|
8
|
+
// Claim file location: .gsd/unit-claims.json
|
|
9
|
+
// Unit key format:
|
|
10
|
+
// task: "<milestoneId>/<sliceId>/<taskId>"
|
|
11
|
+
// slice: "<milestoneId>/<sliceId>"
|
|
12
|
+
//
|
|
13
|
+
// Copyright (c) 2026 Jeremy McSpadden <jeremy@fluxlabs.net>
|
|
14
|
+
|
|
15
|
+
import { existsSync, readFileSync, mkdirSync } from "node:fs";
|
|
16
|
+
import { join } from "node:path";
|
|
17
|
+
import { atomicWriteSync } from "./atomic-write.js";
|
|
18
|
+
|
|
19
|
+
// ─── Types ───────────────────────────────────────────────────────────────
|
|
20
|
+
|
|
21
|
+
export interface UnitClaim {
|
|
22
|
+
agent: string;
|
|
23
|
+
claimed_at: string;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
type ClaimsMap = Record<string, UnitClaim>;
|
|
27
|
+
|
|
28
|
+
// ─── Key Builders ────────────────────────────────────────────────────────
|
|
29
|
+
|
|
30
|
+
export function taskUnitKey(milestoneId: string, sliceId: string, taskId: string): string {
|
|
31
|
+
return `${milestoneId}/${sliceId}/${taskId}`;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export function sliceUnitKey(milestoneId: string, sliceId: string): string {
|
|
35
|
+
return `${milestoneId}/${sliceId}`;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// ─── File Path ───────────────────────────────────────────────────────────
|
|
39
|
+
|
|
40
|
+
function claimsPath(basePath: string): string {
|
|
41
|
+
return join(basePath, ".gsd", "unit-claims.json");
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// ─── Read Claims ─────────────────────────────────────────────────────────
|
|
45
|
+
|
|
46
|
+
function readClaims(basePath: string): ClaimsMap | null {
|
|
47
|
+
const path = claimsPath(basePath);
|
|
48
|
+
if (!existsSync(path)) return null;
|
|
49
|
+
try {
|
|
50
|
+
return JSON.parse(readFileSync(path, "utf-8")) as ClaimsMap;
|
|
51
|
+
} catch {
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// ─── Public API ──────────────────────────────────────────────────────────
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Claim a unit for an agent.
|
|
60
|
+
* Overwrites any existing claim for this unit (last writer wins).
|
|
61
|
+
*/
|
|
62
|
+
export function claimUnit(basePath: string, unitKey: string, agentName: string): void {
|
|
63
|
+
const claims = readClaims(basePath) ?? {};
|
|
64
|
+
claims[unitKey] = { agent: agentName, claimed_at: new Date().toISOString() };
|
|
65
|
+
const dir = join(basePath, ".gsd");
|
|
66
|
+
mkdirSync(dir, { recursive: true });
|
|
67
|
+
atomicWriteSync(claimsPath(basePath), JSON.stringify(claims, null, 2) + "\n");
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Release a unit claim (remove it from the claims map).
|
|
72
|
+
*/
|
|
73
|
+
export function releaseUnit(basePath: string, unitKey: string): void {
|
|
74
|
+
const claims = readClaims(basePath);
|
|
75
|
+
if (!claims || !(unitKey in claims)) return;
|
|
76
|
+
delete claims[unitKey];
|
|
77
|
+
atomicWriteSync(claimsPath(basePath), JSON.stringify(claims, null, 2) + "\n");
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* Get the current owner of a unit, or null if unclaimed / no claims file.
|
|
82
|
+
*/
|
|
83
|
+
export function getOwner(basePath: string, unitKey: string): string | null {
|
|
84
|
+
const claims = readClaims(basePath);
|
|
85
|
+
if (!claims) return null;
|
|
86
|
+
return claims[unitKey]?.agent ?? null;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Check if an actor is authorized to operate on a unit.
|
|
91
|
+
* Returns null if ownership passes (or is unclaimed / no file).
|
|
92
|
+
* Returns an error string if a different agent owns the unit.
|
|
93
|
+
*/
|
|
94
|
+
export function checkOwnership(
|
|
95
|
+
basePath: string,
|
|
96
|
+
unitKey: string,
|
|
97
|
+
actorName: string | undefined,
|
|
98
|
+
): string | null {
|
|
99
|
+
if (!actorName) return null; // no actor identity provided — opt-in, so allow
|
|
100
|
+
const owner = getOwner(basePath, unitKey);
|
|
101
|
+
if (owner === null) return null; // unit unclaimed or no claims file
|
|
102
|
+
if (owner === actorName) return null; // actor is the owner
|
|
103
|
+
return `Unit ${unitKey} is owned by ${owner}, not ${actorName}`;
|
|
104
|
+
}
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
import { createHash, randomUUID } from "node:crypto";
|
|
2
|
+
import { appendFileSync, readFileSync, existsSync, mkdirSync } from "node:fs";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { atomicWriteSync } from "./atomic-write.js";
|
|
5
|
+
|
|
6
|
+
// ─── Session ID ───────────────────────────────────────────────────────────
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Engine-generated session ID — stable for the lifetime of this process.
|
|
10
|
+
* Agents can reference this to correlate all events from one run.
|
|
11
|
+
*/
|
|
12
|
+
const ENGINE_SESSION_ID: string = randomUUID();
|
|
13
|
+
|
|
14
|
+
export function getSessionId(): string {
|
|
15
|
+
return ENGINE_SESSION_ID;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// ─── Event Types ─────────────────────────────────────────────────────────
|
|
19
|
+
|
|
20
|
+
export interface WorkflowEvent {
|
|
21
|
+
cmd: string; // e.g. "complete_task"
|
|
22
|
+
params: Record<string, unknown>;
|
|
23
|
+
ts: string; // ISO 8601
|
|
24
|
+
hash: string; // content hash (hex, 16 chars)
|
|
25
|
+
actor: "agent" | "system";
|
|
26
|
+
actor_name?: string; // e.g. "executor-agent-01" — caller-provided identity
|
|
27
|
+
trigger_reason?: string; // e.g. "plan-phase complete" — caller-provided causation
|
|
28
|
+
session_id: string; // engine-generated UUID, stable per process lifetime
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// ─── appendEvent ─────────────────────────────────────────────────────────
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Append one event to .gsd/event-log.jsonl.
|
|
35
|
+
* Computes a content hash from cmd+params (deterministic, independent of ts/actor/session).
|
|
36
|
+
* Creates .gsd directory if needed.
|
|
37
|
+
*/
|
|
38
|
+
export function appendEvent(
|
|
39
|
+
basePath: string,
|
|
40
|
+
event: Omit<WorkflowEvent, "hash" | "session_id"> & { actor_name?: string; trigger_reason?: string },
|
|
41
|
+
): void {
|
|
42
|
+
const hash = createHash("sha256")
|
|
43
|
+
.update(JSON.stringify({ cmd: event.cmd, params: event.params, ts: event.ts }))
|
|
44
|
+
.digest("hex")
|
|
45
|
+
.slice(0, 16);
|
|
46
|
+
|
|
47
|
+
const fullEvent: WorkflowEvent = {
|
|
48
|
+
...event,
|
|
49
|
+
hash,
|
|
50
|
+
session_id: ENGINE_SESSION_ID,
|
|
51
|
+
};
|
|
52
|
+
const dir = join(basePath, ".gsd");
|
|
53
|
+
mkdirSync(dir, { recursive: true });
|
|
54
|
+
appendFileSync(join(dir, "event-log.jsonl"), JSON.stringify(fullEvent) + "\n", "utf-8");
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// ─── readEvents ──────────────────────────────────────────────────────────
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Read all events from a JSONL file.
|
|
61
|
+
* Returns empty array if file doesn't exist.
|
|
62
|
+
* Corrupted lines are skipped with stderr warning.
|
|
63
|
+
*/
|
|
64
|
+
export function readEvents(logPath: string): WorkflowEvent[] {
|
|
65
|
+
if (!existsSync(logPath)) {
|
|
66
|
+
return [];
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const content = readFileSync(logPath, "utf-8");
|
|
70
|
+
const lines = content.split("\n").filter((l) => l.length > 0);
|
|
71
|
+
const events: WorkflowEvent[] = [];
|
|
72
|
+
|
|
73
|
+
for (const line of lines) {
|
|
74
|
+
try {
|
|
75
|
+
events.push(JSON.parse(line) as WorkflowEvent);
|
|
76
|
+
} catch {
|
|
77
|
+
process.stderr.write(`workflow-events: skipping corrupted event line: ${line.slice(0, 80)}\n`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
return events;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
// ─── findForkPoint ───────────────────────────────────────────────────────
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Find the index of the last common event between two logs by comparing hashes.
|
|
88
|
+
* Returns -1 if the first events differ (completely diverged).
|
|
89
|
+
* If one log is a prefix of the other, returns length of shorter - 1.
|
|
90
|
+
*/
|
|
91
|
+
export function findForkPoint(
|
|
92
|
+
logA: WorkflowEvent[],
|
|
93
|
+
logB: WorkflowEvent[],
|
|
94
|
+
): number {
|
|
95
|
+
const minLen = Math.min(logA.length, logB.length);
|
|
96
|
+
let lastCommon = -1;
|
|
97
|
+
|
|
98
|
+
for (let i = 0; i < minLen; i++) {
|
|
99
|
+
if (logA[i]!.hash === logB[i]!.hash) {
|
|
100
|
+
lastCommon = i;
|
|
101
|
+
} else {
|
|
102
|
+
break;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
return lastCommon;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// ─── compactMilestoneEvents ─────────────────────────────────────────────────
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Archive a milestone's events from the active log to a separate file.
|
|
113
|
+
* Active log retains only events from other milestones.
|
|
114
|
+
* Archived file is kept on disk for forensics.
|
|
115
|
+
*
|
|
116
|
+
* @param basePath - Project root (parent of .gsd/)
|
|
117
|
+
* @param milestoneId - The milestone whose events should be archived
|
|
118
|
+
* @returns { archived: number } — count of events moved to archive
|
|
119
|
+
*/
|
|
120
|
+
export function compactMilestoneEvents(
|
|
121
|
+
basePath: string,
|
|
122
|
+
milestoneId: string,
|
|
123
|
+
): { archived: number } {
|
|
124
|
+
const logPath = join(basePath, ".gsd", "event-log.jsonl");
|
|
125
|
+
const archivePath = join(basePath, ".gsd", `event-log-${milestoneId}.jsonl.archived`);
|
|
126
|
+
|
|
127
|
+
const allEvents = readEvents(logPath);
|
|
128
|
+
const toArchive = allEvents.filter(
|
|
129
|
+
(e) => (e.params as { milestoneId?: string }).milestoneId === milestoneId,
|
|
130
|
+
);
|
|
131
|
+
const remaining = allEvents.filter(
|
|
132
|
+
(e) => (e.params as { milestoneId?: string }).milestoneId !== milestoneId,
|
|
133
|
+
);
|
|
134
|
+
|
|
135
|
+
if (toArchive.length === 0) {
|
|
136
|
+
return { archived: 0 };
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
// Write archived events to .jsonl.archived file (crash-safe)
|
|
140
|
+
atomicWriteSync(
|
|
141
|
+
archivePath,
|
|
142
|
+
toArchive.map((e) => JSON.stringify(e)).join("\n") + "\n",
|
|
143
|
+
);
|
|
144
|
+
|
|
145
|
+
// Truncate active log to remaining events only
|
|
146
|
+
atomicWriteSync(
|
|
147
|
+
logPath,
|
|
148
|
+
remaining.length > 0
|
|
149
|
+
? remaining.map((e) => JSON.stringify(e)).join("\n") + "\n"
|
|
150
|
+
: "",
|
|
151
|
+
);
|
|
152
|
+
|
|
153
|
+
return { archived: toArchive.length };
|
|
154
|
+
}
|