@therocketcode/gsd-core 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (568) hide show
  1. package/.claude-plugin/plugin.json +23 -0
  2. package/GEMINI.md +53 -0
  3. package/LICENSE +21 -0
  4. package/README.ja-JP.md +125 -0
  5. package/README.ko-KR.md +125 -0
  6. package/README.md +144 -0
  7. package/README.pt-BR.md +125 -0
  8. package/README.zh-CN.md +125 -0
  9. package/agents/gsd-advisor-researcher.md +108 -0
  10. package/agents/gsd-ai-researcher.md +114 -0
  11. package/agents/gsd-assumptions-analyzer.md +105 -0
  12. package/agents/gsd-code-fixer.md +668 -0
  13. package/agents/gsd-code-reviewer.md +387 -0
  14. package/agents/gsd-codebase-mapper.md +853 -0
  15. package/agents/gsd-debug-session-manager.md +314 -0
  16. package/agents/gsd-debugger.md +1452 -0
  17. package/agents/gsd-doc-classifier.md +168 -0
  18. package/agents/gsd-doc-synthesizer.md +204 -0
  19. package/agents/gsd-doc-verifier.md +217 -0
  20. package/agents/gsd-doc-writer.md +616 -0
  21. package/agents/gsd-domain-researcher.md +147 -0
  22. package/agents/gsd-eval-auditor.md +191 -0
  23. package/agents/gsd-eval-planner.md +154 -0
  24. package/agents/gsd-executor.md +785 -0
  25. package/agents/gsd-framework-selector.md +160 -0
  26. package/agents/gsd-integration-checker.md +470 -0
  27. package/agents/gsd-intel-updater.md +342 -0
  28. package/agents/gsd-nyquist-auditor.md +203 -0
  29. package/agents/gsd-pattern-mapper.md +335 -0
  30. package/agents/gsd-phase-researcher.md +867 -0
  31. package/agents/gsd-plan-checker.md +978 -0
  32. package/agents/gsd-planner.md +1204 -0
  33. package/agents/gsd-project-researcher.md +611 -0
  34. package/agents/gsd-research-synthesizer.md +259 -0
  35. package/agents/gsd-roadmapper.md +688 -0
  36. package/agents/gsd-security-auditor.md +155 -0
  37. package/agents/gsd-ui-auditor.md +495 -0
  38. package/agents/gsd-ui-checker.md +309 -0
  39. package/agents/gsd-ui-researcher.md +374 -0
  40. package/agents/gsd-user-profiler.md +171 -0
  41. package/agents/gsd-verifier.md +923 -0
  42. package/assets/gsd-logo-2000-transparent.png +0 -0
  43. package/assets/gsd-logo-2000-transparent.svg +17 -0
  44. package/assets/gsd-logo-2000.png +0 -0
  45. package/assets/gsd-logo-2000.svg +21 -0
  46. package/assets/terminal.svg +68 -0
  47. package/bin/install.js +12726 -0
  48. package/bin/lib/ui-safety-gate.cjs +107 -0
  49. package/commands/gsd/add-tests.md +42 -0
  50. package/commands/gsd/ai-integration-phase.md +37 -0
  51. package/commands/gsd/audit-fix.md +34 -0
  52. package/commands/gsd/audit-milestone.md +37 -0
  53. package/commands/gsd/audit-uat.md +24 -0
  54. package/commands/gsd/autonomous.md +48 -0
  55. package/commands/gsd/capture.md +62 -0
  56. package/commands/gsd/cleanup.md +24 -0
  57. package/commands/gsd/code-review.md +59 -0
  58. package/commands/gsd/complete-milestone.md +143 -0
  59. package/commands/gsd/config.md +56 -0
  60. package/commands/gsd/debug.md +52 -0
  61. package/commands/gsd/discover-product.md +65 -0
  62. package/commands/gsd/discuss-phase.md +77 -0
  63. package/commands/gsd/docs-update.md +49 -0
  64. package/commands/gsd/eval-review.md +33 -0
  65. package/commands/gsd/execute-phase.md +66 -0
  66. package/commands/gsd/explore.md +27 -0
  67. package/commands/gsd/extract-learnings.md +23 -0
  68. package/commands/gsd/fast.md +31 -0
  69. package/commands/gsd/forensics.md +57 -0
  70. package/commands/gsd/graphify.md +204 -0
  71. package/commands/gsd/health.md +31 -0
  72. package/commands/gsd/help.md +28 -0
  73. package/commands/gsd/import.md +45 -0
  74. package/commands/gsd/inbox.md +39 -0
  75. package/commands/gsd/ingest-docs.md +42 -0
  76. package/commands/gsd/manager.md +45 -0
  77. package/commands/gsd/map-codebase.md +83 -0
  78. package/commands/gsd/milestone-summary.md +51 -0
  79. package/commands/gsd/model-domain.md +65 -0
  80. package/commands/gsd/mvp-phase.md +45 -0
  81. package/commands/gsd/new-milestone.md +45 -0
  82. package/commands/gsd/new-project.md +47 -0
  83. package/commands/gsd/ns-context.md +23 -0
  84. package/commands/gsd/ns-ideate.md +24 -0
  85. package/commands/gsd/ns-manage.md +29 -0
  86. package/commands/gsd/ns-project.md +22 -0
  87. package/commands/gsd/ns-review.md +26 -0
  88. package/commands/gsd/ns-workflow.md +28 -0
  89. package/commands/gsd/pause-work.md +43 -0
  90. package/commands/gsd/phase.md +56 -0
  91. package/commands/gsd/plan-phase.md +64 -0
  92. package/commands/gsd/plan-review-convergence.md +59 -0
  93. package/commands/gsd/pr-branch.md +26 -0
  94. package/commands/gsd/profile-user.md +46 -0
  95. package/commands/gsd/progress.md +48 -0
  96. package/commands/gsd/quick.md +174 -0
  97. package/commands/gsd/recommend-architecture.md +64 -0
  98. package/commands/gsd/resume-work.md +30 -0
  99. package/commands/gsd/review-backlog.md +63 -0
  100. package/commands/gsd/review.md +42 -0
  101. package/commands/gsd/secure-phase.md +36 -0
  102. package/commands/gsd/settings.md +29 -0
  103. package/commands/gsd/ship.md +24 -0
  104. package/commands/gsd/sketch.md +60 -0
  105. package/commands/gsd/spec-phase.md +63 -0
  106. package/commands/gsd/spike.md +57 -0
  107. package/commands/gsd/stats.md +20 -0
  108. package/commands/gsd/surface.md +155 -0
  109. package/commands/gsd/testing-strategy.md +65 -0
  110. package/commands/gsd/thread.md +24 -0
  111. package/commands/gsd/ui-phase.md +35 -0
  112. package/commands/gsd/ui-review.md +33 -0
  113. package/commands/gsd/ultraplan-phase.md +34 -0
  114. package/commands/gsd/undo.md +35 -0
  115. package/commands/gsd/update.md +49 -0
  116. package/commands/gsd/validate-phase.md +36 -0
  117. package/commands/gsd/verify-work.md +39 -0
  118. package/commands/gsd/workspace.md +52 -0
  119. package/commands/gsd/workstreams.md +70 -0
  120. package/gemini-extension.json +6 -0
  121. package/gsd-core/bin/check-latest-version.cjs +161 -0
  122. package/gsd-core/bin/gsd-tools.cjs +1928 -0
  123. package/gsd-core/bin/lib/active-workstream-store.cjs +291 -0
  124. package/gsd-core/bin/lib/adr-parser.cjs +399 -0
  125. package/gsd-core/bin/lib/agent-command-router.cjs +68 -0
  126. package/gsd-core/bin/lib/artifacts.cjs +51 -0
  127. package/gsd-core/bin/lib/audit.cjs +743 -0
  128. package/gsd-core/bin/lib/check-command-router.cjs +343 -0
  129. package/gsd-core/bin/lib/cjs-command-router-adapter.cjs +81 -0
  130. package/gsd-core/bin/lib/cli-exit.cjs +42 -0
  131. package/gsd-core/bin/lib/clock.cjs +95 -0
  132. package/gsd-core/bin/lib/clusters.cjs +132 -0
  133. package/gsd-core/bin/lib/code-review-flags.cjs +59 -0
  134. package/gsd-core/bin/lib/command-aliases.cjs +809 -0
  135. package/gsd-core/bin/lib/command-arg-projection.cjs +55 -0
  136. package/gsd-core/bin/lib/command-routing-hub.cjs +300 -0
  137. package/gsd-core/bin/lib/commands.cjs +1203 -0
  138. package/gsd-core/bin/lib/config-schema.cjs +29 -0
  139. package/gsd-core/bin/lib/config-types.cjs +19 -0
  140. package/gsd-core/bin/lib/config.cjs +738 -0
  141. package/gsd-core/bin/lib/configuration.cjs +239 -0
  142. package/gsd-core/bin/lib/context-utilization.cjs +48 -0
  143. package/gsd-core/bin/lib/core.cjs +2051 -0
  144. package/gsd-core/bin/lib/decisions.cjs +118 -0
  145. package/gsd-core/bin/lib/docs.cjs +252 -0
  146. package/gsd-core/bin/lib/drift.cjs +364 -0
  147. package/gsd-core/bin/lib/fallow-runner.cjs +115 -0
  148. package/gsd-core/bin/lib/frontmatter.cjs +442 -0
  149. package/gsd-core/bin/lib/gap-checker.cjs +257 -0
  150. package/gsd-core/bin/lib/graphify.cjs +496 -0
  151. package/gsd-core/bin/lib/gsd2-import.cjs +456 -0
  152. package/gsd-core/bin/lib/init-command-router.cjs +62 -0
  153. package/gsd-core/bin/lib/init.cjs +1815 -0
  154. package/gsd-core/bin/lib/install-profiles.cjs +584 -0
  155. package/gsd-core/bin/lib/installer-migration-authoring.cjs +122 -0
  156. package/gsd-core/bin/lib/installer-migration-report.cjs +350 -0
  157. package/gsd-core/bin/lib/installer-migrations/000-first-time-baseline.cjs +218 -0
  158. package/gsd-core/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +48 -0
  159. package/gsd-core/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +94 -0
  160. package/gsd-core/bin/lib/installer-migrations/003-rename-get-shit-done-to-gsd-core.cjs +108 -0
  161. package/gsd-core/bin/lib/installer-migrations.cjs +823 -0
  162. package/gsd-core/bin/lib/intel.cjs +590 -0
  163. package/gsd-core/bin/lib/learnings.cjs +270 -0
  164. package/gsd-core/bin/lib/legacy-cleanup.cjs +253 -0
  165. package/gsd-core/bin/lib/milestone.cjs +373 -0
  166. package/gsd-core/bin/lib/model-catalog.cjs +154 -0
  167. package/gsd-core/bin/lib/model-profiles.cjs +24 -0
  168. package/gsd-core/bin/lib/observability/event.cjs +51 -0
  169. package/gsd-core/bin/lib/observability/logger.cjs +146 -0
  170. package/gsd-core/bin/lib/observability/redaction.cjs +48 -0
  171. package/gsd-core/bin/lib/package-identity.cjs +35 -0
  172. package/gsd-core/bin/lib/package-legitimacy.cjs +368 -0
  173. package/gsd-core/bin/lib/phase-command-router.cjs +189 -0
  174. package/gsd-core/bin/lib/phase-lifecycle.cjs +74 -0
  175. package/gsd-core/bin/lib/phase.cjs +1307 -0
  176. package/gsd-core/bin/lib/phases-command-router.cjs +43 -0
  177. package/gsd-core/bin/lib/plan-scan.cjs +91 -0
  178. package/gsd-core/bin/lib/planning-workspace.cjs +245 -0
  179. package/gsd-core/bin/lib/profile-output.cjs +1120 -0
  180. package/gsd-core/bin/lib/profile-pipeline.cjs +517 -0
  181. package/gsd-core/bin/lib/project-root.cjs +119 -0
  182. package/gsd-core/bin/lib/prompt-budget.cjs +305 -0
  183. package/gsd-core/bin/lib/research-provider.cjs +137 -0
  184. package/gsd-core/bin/lib/research-store.cjs +167 -0
  185. package/gsd-core/bin/lib/review-reviewer-selection.cjs +121 -0
  186. package/gsd-core/bin/lib/roadmap-command-router.cjs +166 -0
  187. package/gsd-core/bin/lib/roadmap-upgrade.cjs +476 -0
  188. package/gsd-core/bin/lib/roadmap.cjs +600 -0
  189. package/gsd-core/bin/lib/runtime-artifact-layout.cjs +312 -0
  190. package/gsd-core/bin/lib/runtime-config-adapter-registry.cjs +56 -0
  191. package/gsd-core/bin/lib/runtime-homes.cjs +190 -0
  192. package/gsd-core/bin/lib/runtime-name-policy.cjs +96 -0
  193. package/gsd-core/bin/lib/runtime-slash.cjs +119 -0
  194. package/gsd-core/bin/lib/schema-detect.cjs +159 -0
  195. package/gsd-core/bin/lib/secrets.cjs +34 -0
  196. package/gsd-core/bin/lib/security.cjs +480 -0
  197. package/gsd-core/bin/lib/semver-compare.cjs +42 -0
  198. package/gsd-core/bin/lib/shell-command-projection.cjs +533 -0
  199. package/gsd-core/bin/lib/state-command-router.cjs +160 -0
  200. package/gsd-core/bin/lib/state-document.cjs +259 -0
  201. package/gsd-core/bin/lib/state.cjs +2010 -0
  202. package/gsd-core/bin/lib/surface.cjs +449 -0
  203. package/gsd-core/bin/lib/task-command-router.cjs +85 -0
  204. package/gsd-core/bin/lib/template.cjs +237 -0
  205. package/gsd-core/bin/lib/uat.cjs +297 -0
  206. package/gsd-core/bin/lib/ui-safety-gate.cjs +98 -0
  207. package/gsd-core/bin/lib/update-context.cjs +218 -0
  208. package/gsd-core/bin/lib/validate-command-router.cjs +91 -0
  209. package/gsd-core/bin/lib/validate.cjs +112 -0
  210. package/gsd-core/bin/lib/verification-command-router.cjs +31 -0
  211. package/gsd-core/bin/lib/verification.cjs +193 -0
  212. package/gsd-core/bin/lib/verify-command-router.cjs +44 -0
  213. package/gsd-core/bin/lib/verify.cjs +1451 -0
  214. package/gsd-core/bin/lib/workstream-inventory-builder.cjs +81 -0
  215. package/gsd-core/bin/lib/workstream-inventory.cjs +147 -0
  216. package/gsd-core/bin/lib/workstream-name-policy.cjs +91 -0
  217. package/gsd-core/bin/lib/workstream.cjs +380 -0
  218. package/gsd-core/bin/lib/worktree-base-ref.cjs +325 -0
  219. package/gsd-core/bin/lib/worktree-safety.cjs +943 -0
  220. package/gsd-core/bin/shared/config-defaults.manifest.json +98 -0
  221. package/gsd-core/bin/shared/config-schema.manifest.json +192 -0
  222. package/gsd-core/bin/shared/model-catalog.json +149 -0
  223. package/gsd-core/bin/shared/runtime-aliases.manifest.json +75 -0
  224. package/gsd-core/bin/verify-reapply-patches.cjs +349 -0
  225. package/gsd-core/contexts/dev.md +21 -0
  226. package/gsd-core/contexts/research.md +22 -0
  227. package/gsd-core/contexts/review.md +23 -0
  228. package/gsd-core/references/agent-contracts.md +79 -0
  229. package/gsd-core/references/ai-evals.md +156 -0
  230. package/gsd-core/references/ai-frameworks.md +186 -0
  231. package/gsd-core/references/architecture-decision.md +74 -0
  232. package/gsd-core/references/artifact-types.md +131 -0
  233. package/gsd-core/references/auth-in-tests.md +91 -0
  234. package/gsd-core/references/autonomous-smart-discuss.md +277 -0
  235. package/gsd-core/references/checkpoints.md +814 -0
  236. package/gsd-core/references/common-bug-patterns.md +114 -0
  237. package/gsd-core/references/context-budget.md +85 -0
  238. package/gsd-core/references/continuation-format.md +253 -0
  239. package/gsd-core/references/db-test-isolation.md +54 -0
  240. package/gsd-core/references/debugger-philosophy.md +76 -0
  241. package/gsd-core/references/decimal-phase-calculation.md +64 -0
  242. package/gsd-core/references/doc-conflict-engine.md +91 -0
  243. package/gsd-core/references/domain-modeling.md +80 -0
  244. package/gsd-core/references/domain-probes.md +125 -0
  245. package/gsd-core/references/e2e-tiering.md +35 -0
  246. package/gsd-core/references/execute-mvp-tdd.md +81 -0
  247. package/gsd-core/references/executor-examples.md +110 -0
  248. package/gsd-core/references/few-shot-examples/plan-checker.md +73 -0
  249. package/gsd-core/references/few-shot-examples/verifier.md +109 -0
  250. package/gsd-core/references/flaky-test-checklist.md +22 -0
  251. package/gsd-core/references/gate-prompts.md +100 -0
  252. package/gsd-core/references/gates.md +70 -0
  253. package/gsd-core/references/git-integration.md +298 -0
  254. package/gsd-core/references/git-planning-commit.md +40 -0
  255. package/gsd-core/references/ios-scaffold.md +123 -0
  256. package/gsd-core/references/mandatory-initial-read.md +2 -0
  257. package/gsd-core/references/model-profile-resolution.md +38 -0
  258. package/gsd-core/references/model-profiles.md +245 -0
  259. package/gsd-core/references/mvp-concepts.md +49 -0
  260. package/gsd-core/references/phase-argument-parsing.md +61 -0
  261. package/gsd-core/references/planner-antipatterns.md +89 -0
  262. package/gsd-core/references/planner-chunked.md +49 -0
  263. package/gsd-core/references/planner-gap-closure.md +62 -0
  264. package/gsd-core/references/planner-graphify-auto-update.md +67 -0
  265. package/gsd-core/references/planner-human-verify-mode.md +57 -0
  266. package/gsd-core/references/planner-interface-context.md +62 -0
  267. package/gsd-core/references/planner-load-graph-context.md +36 -0
  268. package/gsd-core/references/planner-mvp-mode.md +53 -0
  269. package/gsd-core/references/planner-reviews.md +39 -0
  270. package/gsd-core/references/planner-revision.md +87 -0
  271. package/gsd-core/references/planner-source-audit.md +73 -0
  272. package/gsd-core/references/planning-config.md +473 -0
  273. package/gsd-core/references/product-discovery.md +49 -0
  274. package/gsd-core/references/project-skills-discovery.md +19 -0
  275. package/gsd-core/references/questioning.md +162 -0
  276. package/gsd-core/references/realistic-test-data.md +44 -0
  277. package/gsd-core/references/research-documentation-lookup.md +29 -0
  278. package/gsd-core/references/research-philosophy.md +29 -0
  279. package/gsd-core/references/research-verification-protocol.md +27 -0
  280. package/gsd-core/references/revision-loop.md +97 -0
  281. package/gsd-core/references/scout-codebase.md +51 -0
  282. package/gsd-core/references/skeleton-template.md +48 -0
  283. package/gsd-core/references/sketch-interactivity.md +41 -0
  284. package/gsd-core/references/sketch-theme-system.md +94 -0
  285. package/gsd-core/references/sketch-tooling.md +45 -0
  286. package/gsd-core/references/sketch-variant-patterns.md +81 -0
  287. package/gsd-core/references/spidr-splitting.md +69 -0
  288. package/gsd-core/references/tdd.md +330 -0
  289. package/gsd-core/references/test-containers.md +55 -0
  290. package/gsd-core/references/test-strategy.md +75 -0
  291. package/gsd-core/references/thinking-models-debug.md +44 -0
  292. package/gsd-core/references/thinking-models-execution.md +50 -0
  293. package/gsd-core/references/thinking-models-planning.md +62 -0
  294. package/gsd-core/references/thinking-models-research.md +50 -0
  295. package/gsd-core/references/thinking-models-verification.md +55 -0
  296. package/gsd-core/references/thinking-partner.md +96 -0
  297. package/gsd-core/references/ui-brand.md +162 -0
  298. package/gsd-core/references/universal-anti-patterns.md +63 -0
  299. package/gsd-core/references/user-profiling.md +681 -0
  300. package/gsd-core/references/user-story-template.md +58 -0
  301. package/gsd-core/references/verification-overrides.md +227 -0
  302. package/gsd-core/references/verification-patterns.md +612 -0
  303. package/gsd-core/references/verify-mvp-mode.md +85 -0
  304. package/gsd-core/references/workstream-flag.md +111 -0
  305. package/gsd-core/references/worktree-branch-check.md +38 -0
  306. package/gsd-core/references/worktree-path-safety.md +67 -0
  307. package/gsd-core/templates/AI-SPEC.md +246 -0
  308. package/gsd-core/templates/DEBUG.md +169 -0
  309. package/gsd-core/templates/README.md +77 -0
  310. package/gsd-core/templates/SECURITY.md +61 -0
  311. package/gsd-core/templates/UAT.md +265 -0
  312. package/gsd-core/templates/UI-SPEC.md +100 -0
  313. package/gsd-core/templates/VALIDATION.md +76 -0
  314. package/gsd-core/templates/adr.md +58 -0
  315. package/gsd-core/templates/claude-md.md +145 -0
  316. package/gsd-core/templates/codebase/architecture.md +255 -0
  317. package/gsd-core/templates/codebase/concerns.md +310 -0
  318. package/gsd-core/templates/codebase/conventions.md +307 -0
  319. package/gsd-core/templates/codebase/integrations.md +280 -0
  320. package/gsd-core/templates/codebase/stack.md +186 -0
  321. package/gsd-core/templates/codebase/structure.md +285 -0
  322. package/gsd-core/templates/codebase/testing.md +480 -0
  323. package/gsd-core/templates/config.json +62 -0
  324. package/gsd-core/templates/context.md +352 -0
  325. package/gsd-core/templates/continue-here.md +78 -0
  326. package/gsd-core/templates/copilot-instructions.md +7 -0
  327. package/gsd-core/templates/debug-subagent-prompt.md +91 -0
  328. package/gsd-core/templates/dev-preferences.md +21 -0
  329. package/gsd-core/templates/discovery.md +146 -0
  330. package/gsd-core/templates/discussion-log.md +63 -0
  331. package/gsd-core/templates/domain-model.md +54 -0
  332. package/gsd-core/templates/milestone-archive.md +123 -0
  333. package/gsd-core/templates/milestone.md +115 -0
  334. package/gsd-core/templates/phase-prompt.md +610 -0
  335. package/gsd-core/templates/planner-subagent-prompt.md +117 -0
  336. package/gsd-core/templates/product-brief.md +55 -0
  337. package/gsd-core/templates/project.md +186 -0
  338. package/gsd-core/templates/requirements.md +231 -0
  339. package/gsd-core/templates/research-project/ARCHITECTURE.md +204 -0
  340. package/gsd-core/templates/research-project/FEATURES.md +147 -0
  341. package/gsd-core/templates/research-project/PITFALLS.md +200 -0
  342. package/gsd-core/templates/research-project/STACK.md +120 -0
  343. package/gsd-core/templates/research-project/SUMMARY.md +170 -0
  344. package/gsd-core/templates/research.md +592 -0
  345. package/gsd-core/templates/retrospective.md +54 -0
  346. package/gsd-core/templates/roadmap.md +202 -0
  347. package/gsd-core/templates/spec.md +307 -0
  348. package/gsd-core/templates/state.md +195 -0
  349. package/gsd-core/templates/summary-complex.md +59 -0
  350. package/gsd-core/templates/summary-minimal.md +41 -0
  351. package/gsd-core/templates/summary-standard.md +48 -0
  352. package/gsd-core/templates/summary.md +248 -0
  353. package/gsd-core/templates/test-strategy.md +50 -0
  354. package/gsd-core/templates/user-profile.md +146 -0
  355. package/gsd-core/templates/user-setup.md +311 -0
  356. package/gsd-core/templates/verification-report.md +322 -0
  357. package/gsd-core/workflows/_runtime-launcher.snippet.sh +1 -0
  358. package/gsd-core/workflows/add-backlog.md +91 -0
  359. package/gsd-core/workflows/add-phase.md +113 -0
  360. package/gsd-core/workflows/add-tests.md +355 -0
  361. package/gsd-core/workflows/add-todo.md +161 -0
  362. package/gsd-core/workflows/ai-integration-phase.md +295 -0
  363. package/gsd-core/workflows/analyze-dependencies.md +96 -0
  364. package/gsd-core/workflows/audit-fix.md +178 -0
  365. package/gsd-core/workflows/audit-milestone.md +360 -0
  366. package/gsd-core/workflows/audit-uat.md +110 -0
  367. package/gsd-core/workflows/autonomous.md +797 -0
  368. package/gsd-core/workflows/check-todos.md +180 -0
  369. package/gsd-core/workflows/cleanup.md +195 -0
  370. package/gsd-core/workflows/code-review-fix.md +502 -0
  371. package/gsd-core/workflows/code-review.md +658 -0
  372. package/gsd-core/workflows/complete-milestone.md +855 -0
  373. package/gsd-core/workflows/debug.md +237 -0
  374. package/gsd-core/workflows/diagnose-issues.md +245 -0
  375. package/gsd-core/workflows/discover-product.md +112 -0
  376. package/gsd-core/workflows/discovery-phase.md +291 -0
  377. package/gsd-core/workflows/discuss-phase/modes/advisor.md +176 -0
  378. package/gsd-core/workflows/discuss-phase/modes/all.md +28 -0
  379. package/gsd-core/workflows/discuss-phase/modes/analyze.md +44 -0
  380. package/gsd-core/workflows/discuss-phase/modes/auto.md +57 -0
  381. package/gsd-core/workflows/discuss-phase/modes/batch.md +52 -0
  382. package/gsd-core/workflows/discuss-phase/modes/chain.md +98 -0
  383. package/gsd-core/workflows/discuss-phase/modes/default.md +141 -0
  384. package/gsd-core/workflows/discuss-phase/modes/power.md +44 -0
  385. package/gsd-core/workflows/discuss-phase/modes/text.md +55 -0
  386. package/gsd-core/workflows/discuss-phase/templates/checkpoint.json +18 -0
  387. package/gsd-core/workflows/discuss-phase/templates/context.md +136 -0
  388. package/gsd-core/workflows/discuss-phase/templates/discussion-log.md +50 -0
  389. package/gsd-core/workflows/discuss-phase-assumptions.md +675 -0
  390. package/gsd-core/workflows/discuss-phase-power.md +291 -0
  391. package/gsd-core/workflows/discuss-phase.md +499 -0
  392. package/gsd-core/workflows/do.md +111 -0
  393. package/gsd-core/workflows/docs-update.md +1176 -0
  394. package/gsd-core/workflows/edit-phase.md +295 -0
  395. package/gsd-core/workflows/eval-review.md +156 -0
  396. package/gsd-core/workflows/execute-phase/steps/codebase-drift-gate.md +95 -0
  397. package/gsd-core/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  398. package/gsd-core/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  399. package/gsd-core/workflows/execute-phase.md +1752 -0
  400. package/gsd-core/workflows/execute-plan.md +526 -0
  401. package/gsd-core/workflows/explore.md +146 -0
  402. package/gsd-core/workflows/extract-learnings.md +243 -0
  403. package/gsd-core/workflows/fast.md +124 -0
  404. package/gsd-core/workflows/forensics.md +279 -0
  405. package/gsd-core/workflows/graduation.md +196 -0
  406. package/gsd-core/workflows/health.md +224 -0
  407. package/gsd-core/workflows/help/modes/brief.md +22 -0
  408. package/gsd-core/workflows/help/modes/default.md +50 -0
  409. package/gsd-core/workflows/help/modes/full.md +789 -0
  410. package/gsd-core/workflows/help/modes/topic.md +74 -0
  411. package/gsd-core/workflows/help.md +24 -0
  412. package/gsd-core/workflows/import.md +256 -0
  413. package/gsd-core/workflows/inbox.md +387 -0
  414. package/gsd-core/workflows/ingest-docs.md +340 -0
  415. package/gsd-core/workflows/insert-phase.md +152 -0
  416. package/gsd-core/workflows/list-phase-assumptions.md +178 -0
  417. package/gsd-core/workflows/list-workspaces.md +57 -0
  418. package/gsd-core/workflows/manager.md +393 -0
  419. package/gsd-core/workflows/map-codebase.md +446 -0
  420. package/gsd-core/workflows/milestone-summary.md +224 -0
  421. package/gsd-core/workflows/model-domain.md +162 -0
  422. package/gsd-core/workflows/mvp-phase.md +222 -0
  423. package/gsd-core/workflows/new-milestone.md +635 -0
  424. package/gsd-core/workflows/new-project.md +1555 -0
  425. package/gsd-core/workflows/new-workspace.md +240 -0
  426. package/gsd-core/workflows/next.md +299 -0
  427. package/gsd-core/workflows/node-repair.md +92 -0
  428. package/gsd-core/workflows/note.md +158 -0
  429. package/gsd-core/workflows/pause-work.md +244 -0
  430. package/gsd-core/workflows/plan-milestone-gaps.md +281 -0
  431. package/gsd-core/workflows/plan-phase.md +1814 -0
  432. package/gsd-core/workflows/plan-review-convergence.md +346 -0
  433. package/gsd-core/workflows/plant-seed.md +230 -0
  434. package/gsd-core/workflows/pr-branch.md +157 -0
  435. package/gsd-core/workflows/profile-user.md +453 -0
  436. package/gsd-core/workflows/progress.md +699 -0
  437. package/gsd-core/workflows/quick.md +1017 -0
  438. package/gsd-core/workflows/reapply-patches.md +426 -0
  439. package/gsd-core/workflows/recommend-architecture.md +135 -0
  440. package/gsd-core/workflows/remove-phase.md +156 -0
  441. package/gsd-core/workflows/remove-workspace.md +108 -0
  442. package/gsd-core/workflows/resume-project.md +332 -0
  443. package/gsd-core/workflows/review.md +748 -0
  444. package/gsd-core/workflows/scan.md +107 -0
  445. package/gsd-core/workflows/secure-phase.md +182 -0
  446. package/gsd-core/workflows/session-report.md +146 -0
  447. package/gsd-core/workflows/settings-advanced.md +810 -0
  448. package/gsd-core/workflows/settings-integrations.md +312 -0
  449. package/gsd-core/workflows/settings.md +566 -0
  450. package/gsd-core/workflows/ship.md +405 -0
  451. package/gsd-core/workflows/sketch-wrap-up.md +286 -0
  452. package/gsd-core/workflows/sketch.md +361 -0
  453. package/gsd-core/workflows/spec-phase.md +263 -0
  454. package/gsd-core/workflows/spike-wrap-up.md +307 -0
  455. package/gsd-core/workflows/spike.md +453 -0
  456. package/gsd-core/workflows/stats.md +80 -0
  457. package/gsd-core/workflows/sync-skills.md +182 -0
  458. package/gsd-core/workflows/testing-strategy.md +122 -0
  459. package/gsd-core/workflows/thread.md +222 -0
  460. package/gsd-core/workflows/transition.md +694 -0
  461. package/gsd-core/workflows/ui-phase.md +328 -0
  462. package/gsd-core/workflows/ui-review.md +193 -0
  463. package/gsd-core/workflows/ultraplan-phase.md +199 -0
  464. package/gsd-core/workflows/undo.md +314 -0
  465. package/gsd-core/workflows/update.md +496 -0
  466. package/gsd-core/workflows/validate-phase.md +181 -0
  467. package/gsd-core/workflows/verify-phase.md +544 -0
  468. package/gsd-core/workflows/verify-work.md +781 -0
  469. package/hooks/dist/gsd-check-update-worker.js +108 -0
  470. package/hooks/dist/gsd-check-update.js +66 -0
  471. package/hooks/dist/gsd-config-reload.js +133 -0
  472. package/hooks/dist/gsd-context-monitor.js +195 -0
  473. package/hooks/dist/gsd-cursor-post-tool.js +75 -0
  474. package/hooks/dist/gsd-cursor-session-start.js +52 -0
  475. package/hooks/dist/gsd-graphify-update.sh +158 -0
  476. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  477. package/hooks/dist/gsd-prompt-guard.js +97 -0
  478. package/hooks/dist/gsd-read-guard.js +101 -0
  479. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  480. package/hooks/dist/gsd-session-state.sh +59 -0
  481. package/hooks/dist/gsd-statusline.js +566 -0
  482. package/hooks/dist/gsd-update-banner.js +138 -0
  483. package/hooks/dist/gsd-validate-commit.sh +57 -0
  484. package/hooks/dist/gsd-workflow-guard.js +167 -0
  485. package/hooks/dist/gsd-worktree-path-guard.js +169 -0
  486. package/hooks/dist/lib/git-cmd.js +150 -0
  487. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  488. package/hooks/dist/managed-hooks-registry.cjs +38 -0
  489. package/hooks/gsd-check-update-worker.js +108 -0
  490. package/hooks/gsd-check-update.js +66 -0
  491. package/hooks/gsd-config-reload.js +133 -0
  492. package/hooks/gsd-context-monitor.js +195 -0
  493. package/hooks/gsd-cursor-post-tool.js +75 -0
  494. package/hooks/gsd-cursor-session-start.js +52 -0
  495. package/hooks/gsd-graphify-update.sh +158 -0
  496. package/hooks/gsd-phase-boundary.sh +47 -0
  497. package/hooks/gsd-prompt-guard.js +97 -0
  498. package/hooks/gsd-read-guard.js +101 -0
  499. package/hooks/gsd-read-injection-scanner.js +203 -0
  500. package/hooks/gsd-session-state.sh +59 -0
  501. package/hooks/gsd-statusline.js +566 -0
  502. package/hooks/gsd-update-banner.js +138 -0
  503. package/hooks/gsd-validate-commit.sh +57 -0
  504. package/hooks/gsd-workflow-guard.js +167 -0
  505. package/hooks/gsd-worktree-path-guard.js +169 -0
  506. package/hooks/hooks.json +69 -0
  507. package/hooks/lib/git-cmd.js +150 -0
  508. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  509. package/hooks/managed-hooks-registry.cjs +38 -0
  510. package/package.json +115 -0
  511. package/scripts/affected-tests-lib.cjs +542 -0
  512. package/scripts/audit-workflow-script-paths.cjs +73 -0
  513. package/scripts/base64-scan.sh +351 -0
  514. package/scripts/build-hooks.js +247 -0
  515. package/scripts/changeset/README.md +129 -0
  516. package/scripts/changeset/cli.cjs +590 -0
  517. package/scripts/changeset/github-release-notes.cjs +199 -0
  518. package/scripts/changeset/lint.cjs +111 -0
  519. package/scripts/changeset/new.cjs +137 -0
  520. package/scripts/changeset/parse.cjs +114 -0
  521. package/scripts/changeset/render.cjs +34 -0
  522. package/scripts/changeset/serialize.cjs +130 -0
  523. package/scripts/check-alias-drift.cjs +114 -0
  524. package/scripts/check-env.cjs +312 -0
  525. package/scripts/check-npm-integrity.cjs +215 -0
  526. package/scripts/ci-guard-runner.cjs +22 -0
  527. package/scripts/ci-prepare-test-scope.cjs +51 -0
  528. package/scripts/ci-rebase-check.cjs +86 -0
  529. package/scripts/ci-test-scope.cjs +431 -0
  530. package/scripts/command-contract-helpers.cjs +64 -0
  531. package/scripts/diff-touches-shipped-paths.cjs +155 -0
  532. package/scripts/fix-slash-commands.cjs +147 -0
  533. package/scripts/gen-inventory-manifest.cjs +115 -0
  534. package/scripts/gen-research-agents.cjs +276 -0
  535. package/scripts/generate-package-identity.cjs +125 -0
  536. package/scripts/issue-dedupe.cjs +278 -0
  537. package/scripts/lib/allowlist-ratchet.cjs +136 -0
  538. package/scripts/lib/cli-exit.cjs +56 -0
  539. package/scripts/lint-command-contract.cjs +114 -0
  540. package/scripts/lint-descriptions.cjs +87 -0
  541. package/scripts/lint-docs-required.cjs +222 -0
  542. package/scripts/lint-legacy-dir-name.cjs +160 -0
  543. package/scripts/lint-package-identity-drift.cjs +141 -0
  544. package/scripts/lint-pr-check-project-dir.cjs +99 -0
  545. package/scripts/lint-shell-command-projection-drift.cjs +62 -0
  546. package/scripts/lint-skill-deps.cjs +185 -0
  547. package/scripts/lint-test-file-count.allowlist.json +135 -0
  548. package/scripts/lint-test-file-count.cjs +246 -0
  549. package/scripts/mutation-matrix.cjs +222 -0
  550. package/scripts/pr-template-policy.cjs +268 -0
  551. package/scripts/prompt-injection-scan.sh +207 -0
  552. package/scripts/release-notes/discord-release-summary.cjs +373 -0
  553. package/scripts/release-notes/format-github-release-notes.cjs +261 -0
  554. package/scripts/release-tarball-smoke.cjs +629 -0
  555. package/scripts/research-profiles.cjs +149 -0
  556. package/scripts/run-affected-tests.cjs +7 -0
  557. package/scripts/run-cross-platform-tests.cjs +67 -0
  558. package/scripts/run-tests.cjs +315 -0
  559. package/scripts/secret-scan-lint.sh +231 -0
  560. package/scripts/secret-scan.sh +358 -0
  561. package/scripts/setup-branch-protection.sh +236 -0
  562. package/scripts/strip-prose-atrefs.cjs +106 -0
  563. package/scripts/sync-manifest-versions.cjs +119 -0
  564. package/scripts/sync-rulesets.sh +34 -0
  565. package/scripts/sync-runtime-launcher.cjs +399 -0
  566. package/scripts/test-failure-reasons.cjs +34 -0
  567. package/scripts/verify-npm-publish.cjs +240 -0
  568. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,943 @@
1
+ "use strict";
2
+ /**
3
+ * Worktree Safety Policy Module
4
+ *
5
+ * Owns worktree-root resolution and non-destructive prune policy decisions.
6
+ *
7
+ * ADR-457 build-at-publish: the hand-written bin/lib/worktree-safety.cjs
8
+ * collapsed to a TypeScript source of truth. Behaviour is preserved
9
+ * byte-for-behaviour from the prior hand-written .cjs; only types are added.
10
+ */
11
+ var __importDefault = (this && this.__importDefault) || function (mod) {
12
+ return (mod && mod.__esModule) ? mod : { "default": mod };
13
+ };
14
+ const node_fs_1 = __importDefault(require("node:fs"));
15
+ const node_path_1 = __importDefault(require("node:path"));
16
+ const shell_command_projection_cjs_1 = require("./shell-command-projection.cjs");
17
+ // Default timeout for worktree-related git subprocess calls.
18
+ // 10 s is generous enough for normal git operations on large repos while still
19
+ // providing a deterministic failure path when git stalls (locked index, hung
20
+ // remote, stalled NFS mount, etc.). Callers can override via deps.timeout.
21
+ const DEFAULT_GIT_TIMEOUT_MS = 10000;
22
+ /**
23
+ * Execute a git command via the shell-projection seam, with a derived
24
+ * `timedOut` field. Tests inject mocks via deps.execGit using the new
25
+ * (args, opts) shape — see worktree-safety-policy.test.cjs.
26
+ *
27
+ * Return shape: { exitCode, stdout, stderr, timedOut, error, signal }
28
+ * - timedOut: true when spawnSync reports SIGTERM + ETIMEDOUT
29
+ */
30
+ function execGitDefault(args, opts = {}) {
31
+ const result = (0, shell_command_projection_cjs_1.execGit)(args, { ...opts, timeout: opts.timeout ?? DEFAULT_GIT_TIMEOUT_MS });
32
+ const timedOut = result.signal === 'SIGTERM' && result.error?.code === 'ETIMEDOUT';
33
+ return { ...result, timedOut };
34
+ }
35
+ function parseWorktreePorcelain(porcelain) {
36
+ return parseWorktreeEntries(porcelain).filter((entry) => entry.branch !== null).map((entry) => ({
37
+ path: entry.path,
38
+ branch: entry.branch,
39
+ }));
40
+ }
41
+ function parseWorktreeEntries(porcelain) {
42
+ const entries = [];
43
+ const blocks = String(porcelain || '').split('\n\n').filter(Boolean);
44
+ for (const block of blocks) {
45
+ const lines = block.split('\n');
46
+ const worktreeLine = lines.find((l) => l.startsWith('worktree '));
47
+ if (!worktreeLine)
48
+ continue;
49
+ const worktreePath = worktreeLine.slice('worktree '.length).trim();
50
+ if (!worktreePath)
51
+ continue;
52
+ const branchLine = lines.find((l) => l.startsWith('branch refs/heads/'));
53
+ const branch = branchLine ? branchLine.slice('branch refs/heads/'.length).trim() : null;
54
+ entries.push({ path: worktreePath, branch });
55
+ }
56
+ return entries;
57
+ }
58
+ function parseWorktreeListPaths(porcelain) {
59
+ return parseWorktreeEntries(porcelain).map((entry) => entry.path);
60
+ }
61
+ function readWorktreeList(repoRoot, deps = {}) {
62
+ const execGit = deps.execGit || execGitDefault;
63
+ const listResult = execGit(['worktree', 'list', '--porcelain'], { cwd: repoRoot });
64
+ if (listResult.timedOut) {
65
+ // AC2 / AC4: surface timeout as a distinct reason so callers can emit a
66
+ // structured warning rather than silently treating the failure as a generic
67
+ // list error (PRED.k302 — error-swallowing-empty-sentinel).
68
+ return {
69
+ ok: false,
70
+ reason: 'git_timed_out',
71
+ porcelain: '',
72
+ entries: [],
73
+ };
74
+ }
75
+ if (listResult.exitCode !== 0) {
76
+ const stderr = String(listResult.stderr || '');
77
+ return {
78
+ ok: false,
79
+ reason: /not a git repository|not a git repo/i.test(stderr)
80
+ ? 'not_a_git_repo'
81
+ : 'git_list_failed',
82
+ porcelain: '',
83
+ entries: [],
84
+ };
85
+ }
86
+ return {
87
+ ok: true,
88
+ reason: 'ok',
89
+ porcelain: listResult.stdout,
90
+ entries: parseWorktreeEntries(listResult.stdout),
91
+ };
92
+ }
93
+ function resolveWorktreeContext(cwd, deps = {}) {
94
+ const execGit = deps.execGit || execGitDefault;
95
+ const existsSync = deps.existsSync || node_fs_1.default.existsSync;
96
+ // Local .planning takes precedence over linked-worktree remapping.
97
+ if (existsSync(node_path_1.default.join(cwd, '.planning'))) {
98
+ return {
99
+ effectiveRoot: cwd,
100
+ mode: 'current_directory',
101
+ reason: 'has_local_planning',
102
+ };
103
+ }
104
+ const gitDir = execGit(['rev-parse', '--git-dir'], { cwd });
105
+ const commonDir = execGit(['rev-parse', '--git-common-dir'], { cwd });
106
+ if (gitDir.exitCode !== 0 || commonDir.exitCode !== 0) {
107
+ return {
108
+ effectiveRoot: cwd,
109
+ mode: 'current_directory',
110
+ reason: 'not_git_repo',
111
+ };
112
+ }
113
+ const gitDirResolved = node_path_1.default.resolve(cwd, gitDir.stdout);
114
+ const commonDirResolved = node_path_1.default.resolve(cwd, commonDir.stdout);
115
+ if (gitDirResolved !== commonDirResolved) {
116
+ return {
117
+ effectiveRoot: node_path_1.default.dirname(commonDirResolved),
118
+ mode: 'linked_worktree_root',
119
+ reason: 'linked_worktree',
120
+ };
121
+ }
122
+ return {
123
+ effectiveRoot: cwd,
124
+ mode: 'current_directory',
125
+ reason: 'main_worktree',
126
+ };
127
+ }
128
+ function planWorktreePrune(repoRoot, options = {}, deps = {}) {
129
+ const parsePorcelain = deps.parseWorktreePorcelain || parseWorktreePorcelain;
130
+ const destructiveModeRequested = Boolean(options.allowDestructive);
131
+ const listed = readWorktreeList(repoRoot, deps);
132
+ if (!listed.ok) {
133
+ return {
134
+ repoRoot,
135
+ action: 'skip',
136
+ reason: listed.reason,
137
+ destructiveModeRequested,
138
+ };
139
+ }
140
+ let worktrees = [];
141
+ try {
142
+ worktrees = parsePorcelain(listed.porcelain);
143
+ }
144
+ catch {
145
+ // Keep historical behavior: still run metadata prune when parsing fails.
146
+ worktrees = [];
147
+ }
148
+ return {
149
+ repoRoot,
150
+ action: 'metadata_prune_only',
151
+ reason: worktrees.length === 0 ? 'no_worktrees' : 'worktrees_present',
152
+ destructiveModeRequested,
153
+ };
154
+ }
155
+ function executeWorktreePrunePlan(plan, deps = {}) {
156
+ const execGit = deps.execGit || execGitDefault;
157
+ if (!plan || plan.action === 'skip') {
158
+ return {
159
+ ok: false,
160
+ action: plan ? plan.action : 'skip',
161
+ reason: plan ? plan.reason : 'missing_plan',
162
+ pruned: [],
163
+ };
164
+ }
165
+ if (plan.action !== 'metadata_prune_only') {
166
+ return {
167
+ ok: false,
168
+ action: plan.action,
169
+ reason: 'unsupported_action',
170
+ pruned: [],
171
+ };
172
+ }
173
+ const result = execGit(['worktree', 'prune'], { cwd: plan.repoRoot });
174
+ if (result.timedOut) {
175
+ // AC4: surface timedOut as a first-class field so callers (e.g.
176
+ // pruneOrphanedWorktrees in core.cjs) can log a structured WARNING rather
177
+ // than silently ignoring it (PRED.k302 — error-swallowing-empty-sentinel).
178
+ return {
179
+ ok: false,
180
+ action: plan.action,
181
+ reason: 'git_timed_out',
182
+ timedOut: true,
183
+ pruned: [],
184
+ };
185
+ }
186
+ return {
187
+ ok: result.exitCode === 0,
188
+ action: plan.action,
189
+ reason: plan.reason,
190
+ timedOut: false,
191
+ pruned: [],
192
+ };
193
+ }
194
+ function listLinkedWorktreePaths(repoRoot, deps = {}) {
195
+ const listed = readWorktreeList(repoRoot, deps);
196
+ if (!listed.ok) {
197
+ return {
198
+ ok: false,
199
+ reason: listed.reason,
200
+ paths: [],
201
+ };
202
+ }
203
+ const allPaths = listed.entries.map((entry) => entry.path);
204
+ // git worktree list always includes the current/main worktree first.
205
+ return {
206
+ ok: true,
207
+ reason: 'ok',
208
+ paths: allPaths.slice(1),
209
+ };
210
+ }
211
+ function inspectWorktreeHealth(repoRoot, options = {}, deps = {}) {
212
+ const inventory = snapshotWorktreeInventory(repoRoot, options, deps);
213
+ if (!inventory.ok) {
214
+ return {
215
+ ok: false,
216
+ reason: inventory.reason,
217
+ findings: [],
218
+ };
219
+ }
220
+ const findings = [];
221
+ for (const entry of inventory.entries) {
222
+ if (!entry.exists) {
223
+ findings.push({
224
+ kind: 'orphan',
225
+ path: entry.path,
226
+ });
227
+ continue;
228
+ }
229
+ if (entry.isStale) {
230
+ findings.push({
231
+ kind: 'stale',
232
+ path: entry.path,
233
+ ageMinutes: entry.ageMinutes ?? undefined,
234
+ });
235
+ }
236
+ }
237
+ return {
238
+ ok: true,
239
+ reason: 'ok',
240
+ findings,
241
+ };
242
+ }
243
+ function snapshotWorktreeInventory(repoRoot, options = {}, deps = {}) {
244
+ const existsSync = deps.existsSync || node_fs_1.default.existsSync;
245
+ const statSync = deps.statSync || node_fs_1.default.statSync;
246
+ const staleAfterMs = options.staleAfterMs ?? (60 * 60 * 1000);
247
+ const nowMs = options.nowMs ?? Date.now();
248
+ const listed = listLinkedWorktreePaths(repoRoot, { execGit: deps.execGit || execGitDefault });
249
+ if (!listed.ok) {
250
+ return {
251
+ ok: false,
252
+ reason: listed.reason,
253
+ entries: [],
254
+ };
255
+ }
256
+ const entries = [];
257
+ for (const worktreePath of listed.paths) {
258
+ let exists = false;
259
+ let isStale = false;
260
+ let ageMinutes = null;
261
+ if (!existsSync(worktreePath)) {
262
+ entries.push({
263
+ path: worktreePath,
264
+ exists,
265
+ isStale,
266
+ ageMinutes,
267
+ });
268
+ continue;
269
+ }
270
+ exists = true;
271
+ try {
272
+ const stat = statSync(worktreePath);
273
+ const ageMs = nowMs - stat.mtimeMs;
274
+ ageMinutes = Math.round(ageMs / 60000);
275
+ if (ageMs > staleAfterMs) {
276
+ isStale = true;
277
+ }
278
+ }
279
+ catch {
280
+ // Keep historical behavior: stat failures are ignored.
281
+ }
282
+ entries.push({
283
+ path: worktreePath,
284
+ exists,
285
+ isStale,
286
+ ageMinutes,
287
+ });
288
+ }
289
+ return {
290
+ ok: true,
291
+ reason: 'ok',
292
+ entries,
293
+ };
294
+ }
295
+ function normalizeCleanupManifestEntry(entry) {
296
+ if (!entry || typeof entry !== 'object')
297
+ return null;
298
+ const e = entry;
299
+ const worktreePath = typeof e.worktree_path === 'string'
300
+ ? e.worktree_path
301
+ : (typeof e.path === 'string' ? e.path : '');
302
+ const branch = typeof e.branch === 'string' ? e.branch : '';
303
+ const expectedBase = typeof e.expected_base === 'string' ? e.expected_base : '';
304
+ if (!worktreePath || !branch || !expectedBase)
305
+ return null;
306
+ if (!/^worktree-agent-[A-Za-z0-9._/-]+$/.test(branch))
307
+ return null;
308
+ return {
309
+ agent_id: typeof e.agent_id === 'string' ? e.agent_id : null,
310
+ worktree_path: worktreePath,
311
+ branch,
312
+ expected_base: expectedBase,
313
+ };
314
+ }
315
+ function normalizeCleanupManifest(manifest) {
316
+ let parsed = manifest;
317
+ if (typeof manifest === 'string') {
318
+ try {
319
+ parsed = JSON.parse(manifest);
320
+ }
321
+ catch {
322
+ return { ok: false, reason: 'invalid_manifest_json', entries: [] };
323
+ }
324
+ }
325
+ const p = parsed;
326
+ const rawEntries = Array.isArray(p)
327
+ ? p
328
+ : (Array.isArray(p?.worktrees) ? p.worktrees : []);
329
+ const seen = new Set();
330
+ const entries = [];
331
+ for (const raw of rawEntries) {
332
+ const entry = normalizeCleanupManifestEntry(raw);
333
+ if (!entry)
334
+ continue;
335
+ const key = `${entry.worktree_path}\0${entry.branch}`;
336
+ if (seen.has(key))
337
+ continue;
338
+ seen.add(key);
339
+ entries.push(entry);
340
+ }
341
+ if (entries.length === 0) {
342
+ return { ok: false, reason: 'empty_manifest', entries: [] };
343
+ }
344
+ return { ok: true, reason: 'ok', entries };
345
+ }
346
+ function planWorktreeWaveCleanup(repoRoot, manifest) {
347
+ const normalized = normalizeCleanupManifest(manifest);
348
+ if (!normalized.ok) {
349
+ return {
350
+ ok: false,
351
+ repoRoot,
352
+ action: 'skip',
353
+ discovery: 'manifest',
354
+ reason: normalized.reason,
355
+ entries: [],
356
+ };
357
+ }
358
+ return {
359
+ ok: true,
360
+ repoRoot,
361
+ action: 'cleanup_wave',
362
+ discovery: 'manifest',
363
+ reason: 'manifest_entries_present',
364
+ entries: normalized.entries,
365
+ };
366
+ }
367
+ function gitResultOk(result) {
368
+ return !!(result && result.exitCode === 0 && !result.timedOut);
369
+ }
370
+ /**
371
+ * Walk <worktreePath>/.planning/ recursively and collect absolute paths of
372
+ * all files whose names match *SUMMARY.md. Returns [] when the directory
373
+ * does not exist or cannot be read.
374
+ *
375
+ * Mirrors the shell fallback in quick.md (#2296, #2070, #2838):
376
+ * find "$WT/.planning" -name "*SUMMARY.md"
377
+ */
378
+ function defaultFindSummaryFiles(worktreePath) {
379
+ const planningDir = node_path_1.default.join(worktreePath, '.planning');
380
+ const results = [];
381
+ function walk(dir) {
382
+ let entries;
383
+ try {
384
+ entries = node_fs_1.default.readdirSync(dir, { withFileTypes: true });
385
+ }
386
+ catch {
387
+ return;
388
+ }
389
+ for (const entry of entries) {
390
+ const full = node_path_1.default.join(dir, entry.name);
391
+ if (entry.isDirectory()) {
392
+ walk(full);
393
+ }
394
+ else if (entry.isFile() && entry.name.endsWith('SUMMARY.md')) {
395
+ results.push(full);
396
+ }
397
+ }
398
+ }
399
+ walk(planningDir);
400
+ return results;
401
+ }
402
+ /**
403
+ * Rescue uncommitted SUMMARY.md artifacts from a worktree into the main repo
404
+ * tree before the dirty-state check. Mirrors the shell-fallback rescue block
405
+ * in quick.md (lines 878–891, #2296/#2070/#2838).
406
+ *
407
+ * For each *SUMMARY.md found under <worktreePath>/.planning/:
408
+ * - compute relative path from worktree root → .planning/<id>-SUMMARY.md
409
+ * - if the file is ALREADY COMMITTED on the worktree branch
410
+ * (`git cat-file -e HEAD:<relPath>` returns exit 0), skip the copy entirely:
411
+ * the merge will carry it naturally and copying it as an untracked file would
412
+ * cause a "untracked working tree files would be overwritten by merge" collision.
413
+ * On timeout or fatal exit (128) the rescue is also skipped (fail-closed).
414
+ * (#706 — execute-phase committed-SUMMARY contract)
415
+ * - destination = <repoRoot>/<relPath>
416
+ * - copy when dest is absent or content differs
417
+ *
418
+ * Returns `{ rescuedRelPaths, failures }`:
419
+ * - `rescuedRelPaths`: Set of worktree-relative paths that were successfully rescued
420
+ * (copy not needed because dest already matches, or copy succeeded). Only paths
421
+ * where the rescue genuinely succeeded are included so the dirty-block filter does
422
+ * not suppress paths that were silently lost.
423
+ * - `failures`: array of `{ relPath, error }` for any path where mkdirSync or
424
+ * copyFileSync threw. A read failure during content comparison is NOT a rescue
425
+ * failure — it sets needsCopy=true and the copy is attempted normally.
426
+ */
427
+ function rescueSummaryArtifacts(worktreePath, repoRoot, deps) {
428
+ const execGit = deps.execGit || execGitDefault;
429
+ const findSummaryFiles = deps.findSummaryFiles || defaultFindSummaryFiles;
430
+ const existsSync = deps.existsSync || node_fs_1.default.existsSync;
431
+ const readFileSync = deps.readFileSync || ((p) => node_fs_1.default.readFileSync(p, 'utf8'));
432
+ const mkdirSync = deps.mkdirSync || ((d, o) => node_fs_1.default.mkdirSync(d, o));
433
+ const copyFileSync = deps.copyFileSync || node_fs_1.default.copyFileSync;
434
+ const summaryPaths = findSummaryFiles(worktreePath);
435
+ const rescuedRelPaths = new Set();
436
+ const failures = [];
437
+ for (const absPath of summaryPaths) {
438
+ // relPath is the path relative to the worktree root (e.g. ".planning/q1-SUMMARY.md")
439
+ // Normalize to forward slashes so the Set comparison against `git status --porcelain`
440
+ // output works on Windows too (git always emits forward slashes in porcelain output).
441
+ const relPath = absPath.slice(worktreePath.length).replace(/^[/\\]/, '').replace(/\\/g, '/');
442
+ // #706: skip rescue when the SUMMARY is already committed on the branch.
443
+ // Use `git cat-file -e HEAD:<relPath>` (not `ls-files --error-unmatch`) so
444
+ // the check is against the committed tree, not the index. ls-files also
445
+ // matches staged-but-uncommitted files, which would skip rescue when the
446
+ // file is staged but not yet committed — the merge wouldn't carry it, and
447
+ // the executor's content could be lost. cat-file -e HEAD:<path> returns
448
+ // exit 0 only when the object exists in the committed HEAD tree.
449
+ //
450
+ // Fail-closed on timeout/fatal git errors: if we cannot determine whether
451
+ // the file is committed, do NOT rescue it (rescuing an actually-committed
452
+ // file would re-create the untracked collision; the merge will surface the
453
+ // issue). The cleanup will be blocked by merge_failed in the worst case,
454
+ // which is the observable behaviour before this fix and is recoverable.
455
+ const catFileResult = execGit(['-C', worktreePath, 'cat-file', '-e', `HEAD:${relPath}`], { cwd: repoRoot });
456
+ if (catFileResult.exitCode !== 1) {
457
+ // Rescue only when cat-file definitively reports the object is absent (exit 1).
458
+ // exit 0 → object exists (committed on HEAD) — merge will carry it, skip.
459
+ // exit 128 → fatal git error (corrupt store, unborn HEAD, etc.) — uncertain,
460
+ // fail-closed: do NOT rescue to avoid recreating the #706 collision.
461
+ // timedOut / null / other → unreliable result — same fail-closed policy.
462
+ // In all non-1 cases the merge will either succeed naturally (0) or surface
463
+ // the problem safely (128/timeout), which is the recoverable pre-fix behaviour.
464
+ continue;
465
+ }
466
+ const dest = node_path_1.default.join(repoRoot, relPath);
467
+ let needsCopy = !existsSync(dest);
468
+ if (!needsCopy) {
469
+ try {
470
+ const srcContent = readFileSync(absPath);
471
+ const destContent = readFileSync(dest);
472
+ needsCopy = srcContent !== destContent;
473
+ }
474
+ catch {
475
+ // Read failure during comparison is not a rescue failure — force a copy attempt.
476
+ needsCopy = true;
477
+ }
478
+ }
479
+ if (needsCopy) {
480
+ try {
481
+ mkdirSync(node_path_1.default.dirname(dest), { recursive: true });
482
+ copyFileSync(absPath, dest);
483
+ // Copy succeeded — the SUMMARY is now safe in the main tree.
484
+ rescuedRelPaths.add(relPath);
485
+ }
486
+ catch (err) {
487
+ // Write failure: the SUMMARY was NOT rescued. Record it so the caller can
488
+ // block cleanup instead of silently losing data.
489
+ failures.push({ relPath, error: err.message });
490
+ }
491
+ }
492
+ else {
493
+ // dest already exists with identical content — SUMMARY is already safe.
494
+ rescuedRelPaths.add(relPath);
495
+ }
496
+ }
497
+ return { rescuedRelPaths, failures };
498
+ }
499
+ function executeWorktreeWaveCleanupPlan(plan, deps = {}) {
500
+ const execGit = deps.execGit || execGitDefault;
501
+ const entries = Array.isArray(plan?.entries) ? plan.entries : [];
502
+ if (!plan || plan.action !== 'cleanup_wave' || entries.length === 0) {
503
+ return {
504
+ ok: false,
505
+ action: plan ? plan.action : 'skip',
506
+ reason: plan ? (plan.reason || 'missing_entries') : 'missing_plan',
507
+ entries: [],
508
+ pending: entries,
509
+ };
510
+ }
511
+ const results = [];
512
+ const pending = [];
513
+ let ok = true;
514
+ for (let i = 0; i < entries.length; i += 1) {
515
+ const entry = entries[i];
516
+ const result = {
517
+ ...entry,
518
+ status: 'pending',
519
+ reason: null,
520
+ stderr: '',
521
+ };
522
+ const branchCheck = execGit(['-C', entry.worktree_path, 'rev-parse', '--abbrev-ref', 'HEAD'], { cwd: plan.repoRoot });
523
+ if (!gitResultOk(branchCheck) || branchCheck.stdout.trim() !== entry.branch) {
524
+ result.status = 'blocked';
525
+ result.reason = 'branch_mismatch';
526
+ result.stderr = branchCheck?.stderr || '';
527
+ results.push(result);
528
+ pending.push(...entries.slice(i + 1));
529
+ ok = false;
530
+ break;
531
+ }
532
+ const mergeBase = execGit(['merge-base', 'HEAD', entry.branch], { cwd: plan.repoRoot });
533
+ if (!gitResultOk(mergeBase) || mergeBase.stdout.trim() !== entry.expected_base) {
534
+ result.status = 'blocked';
535
+ result.reason = 'base_mismatch';
536
+ result.stderr = mergeBase?.stderr || '';
537
+ results.push(result);
538
+ pending.push(...entries.slice(i + 1));
539
+ ok = false;
540
+ break;
541
+ }
542
+ const deletions = execGit(['diff', '--diff-filter=D', '--name-only', `HEAD...${entry.branch}`], { cwd: plan.repoRoot });
543
+ if (!gitResultOk(deletions)) {
544
+ result.status = 'blocked';
545
+ result.reason = 'deletion_check_failed';
546
+ result.stderr = deletions?.stderr || '';
547
+ results.push(result);
548
+ pending.push(...entries.slice(i + 1));
549
+ ok = false;
550
+ break;
551
+ }
552
+ if (deletions.stdout) {
553
+ result.status = 'blocked';
554
+ result.reason = 'branch_contains_deletions';
555
+ result.stderr = deletions.stdout;
556
+ results.push(result);
557
+ pending.push(...entries.slice(i + 1));
558
+ ok = false;
559
+ break;
560
+ }
561
+ // Safety net: rescue uncommitted SUMMARY.md artifacts before the dirty check.
562
+ // The executor leaves <quick_id>-SUMMARY.md uncommitted by contract — the
563
+ // orchestrator commits it. Mirrors quick.md shell fallback (#2296, #2070, #2838, #3804).
564
+ const { rescuedRelPaths, failures: rescueFailures } = rescueSummaryArtifacts(entry.worktree_path, plan.repoRoot, deps);
565
+ if (rescueFailures.length > 0) {
566
+ result.status = 'blocked';
567
+ result.reason = 'summary_rescue_failed';
568
+ result.stderr = rescueFailures.map((f) => `${f.relPath}: ${f.error}`).join('; ');
569
+ results.push(result);
570
+ pending.push(...entries.slice(i + 1));
571
+ ok = false;
572
+ break;
573
+ }
574
+ const worktreeStatus = execGit(['-C', entry.worktree_path, 'status', '--porcelain', '--untracked-files=all'], { cwd: plan.repoRoot });
575
+ if (!gitResultOk(worktreeStatus)) {
576
+ result.status = 'blocked';
577
+ result.reason = 'worktree_dirty';
578
+ result.stderr = worktreeStatus?.stderr || '';
579
+ results.push(result);
580
+ pending.push(...entries.slice(i + 1));
581
+ ok = false;
582
+ break;
583
+ }
584
+ // Filter rescued SUMMARY paths out of the porcelain output before deciding dirty.
585
+ // A line like "?? .planning/q1-SUMMARY.md" should not block when the SUMMARY
586
+ // has already been rescued into the main tree.
587
+ const dirtyLines = (worktreeStatus.stdout || '')
588
+ .split('\n')
589
+ .filter((line) => {
590
+ if (!line.trim())
591
+ return false;
592
+ // porcelain v1 format: "XY path" (3-char prefix + space + path)
593
+ const filePath = line.slice(3).trim();
594
+ return !rescuedRelPaths.has(filePath);
595
+ });
596
+ if (dirtyLines.length > 0) {
597
+ result.status = 'blocked';
598
+ result.reason = 'worktree_dirty';
599
+ result.stderr = dirtyLines.join('\n');
600
+ results.push(result);
601
+ pending.push(...entries.slice(i + 1));
602
+ ok = false;
603
+ break;
604
+ }
605
+ const merge = execGit(['merge', entry.branch, '--no-ff', '--no-edit', '-m', `chore: merge executor worktree (${entry.branch})`], { cwd: plan.repoRoot });
606
+ if (!gitResultOk(merge)) {
607
+ result.status = 'blocked';
608
+ result.reason = 'merge_failed';
609
+ result.stderr = merge?.stderr || merge?.stdout || '';
610
+ results.push(result);
611
+ pending.push(...entries.slice(i + 1));
612
+ ok = false;
613
+ break;
614
+ }
615
+ let remove = execGit(['worktree', 'remove', entry.worktree_path, '--force'], { cwd: plan.repoRoot });
616
+ if (!gitResultOk(remove)) {
617
+ // Locked worktrees require unlock before remove (or --force --force).
618
+ // Attempt: git worktree unlock <path> (ignore failure — already unlocked is ok)
619
+ // then retry git worktree remove --force. (#3707)
620
+ execGit(['worktree', 'unlock', entry.worktree_path], { cwd: plan.repoRoot });
621
+ remove = execGit(['worktree', 'remove', entry.worktree_path, '--force'], { cwd: plan.repoRoot });
622
+ }
623
+ if (!gitResultOk(remove)) {
624
+ result.status = 'blocked';
625
+ result.reason = 'worktree_remove_failed';
626
+ result.stderr = remove?.stderr || '';
627
+ results.push(result);
628
+ pending.push(...entries.slice(i + 1));
629
+ ok = false;
630
+ break;
631
+ }
632
+ const branchDelete = execGit(['branch', '-D', entry.branch], { cwd: plan.repoRoot });
633
+ if (!gitResultOk(branchDelete)) {
634
+ result.status = 'warning';
635
+ result.reason = 'branch_delete_failed';
636
+ result.stderr = branchDelete?.stderr || '';
637
+ ok = false;
638
+ }
639
+ else {
640
+ result.status = 'merged_removed';
641
+ result.reason = 'ok';
642
+ }
643
+ results.push(result);
644
+ }
645
+ return {
646
+ ok,
647
+ action: plan.action,
648
+ reason: ok ? 'ok' : 'cleanup_blocked',
649
+ entries: results,
650
+ pending,
651
+ };
652
+ }
653
+ function cmdWorktreeCleanupWave(cwd, args = []) {
654
+ const manifestFlagIndex = args.indexOf('--manifest');
655
+ const manifestPath = manifestFlagIndex >= 0 ? args[manifestFlagIndex + 1] : '';
656
+ if (!manifestPath) {
657
+ process.stderr.write('Usage: worktree cleanup-wave --manifest <path>\n');
658
+ process.exitCode = 2;
659
+ return;
660
+ }
661
+ let manifest;
662
+ try {
663
+ manifest = node_fs_1.default.readFileSync(node_path_1.default.resolve(cwd, manifestPath), 'utf8');
664
+ }
665
+ catch (err) {
666
+ process.stdout.write(`${JSON.stringify({
667
+ ok: false,
668
+ reason: 'manifest_read_failed',
669
+ error: err.message,
670
+ }, null, 2)}\n`);
671
+ process.exitCode = 1;
672
+ return;
673
+ }
674
+ const plan = planWorktreeWaveCleanup(cwd, manifest);
675
+ const result = executeWorktreeWaveCleanupPlan(plan);
676
+ const response = {
677
+ ok: result.ok,
678
+ plan: {
679
+ action: plan.action,
680
+ discovery: plan.discovery,
681
+ reason: plan.reason,
682
+ entries: plan.entries.length,
683
+ },
684
+ result,
685
+ };
686
+ process.stdout.write(`${JSON.stringify(response, null, 2)}\n`);
687
+ if (!result.ok) {
688
+ process.exitCode = 1;
689
+ }
690
+ }
691
+ /**
692
+ * Reap orphaned linked worktrees whose lock owner process is dead, whose
693
+ * branch tip is fully merged into the default branch, and whose lock file
694
+ * mtime is older than REAP_MTIME_GUARD_MS (race guard).
695
+ */
696
+ const REAP_MTIME_GUARD_MS = 5 * 60 * 1000; // 5 minutes
697
+ function reapOrphanWorktrees(repoRoot, deps = {}) {
698
+ const execGit = deps.execGit || execGitDefault;
699
+ const isPidAliveCheck = deps.isPidAlive || defaultIsPidAlive;
700
+ const readDirSafe = deps.readDirSafe || defaultReadDirSafe;
701
+ const readFileSafe = deps.readFileSafe || defaultReadFileSafe;
702
+ const mtimeSafe = deps.mtimeSafe || defaultMtimeSafe;
703
+ const reapMtimeGuardMs = deps.reapMtimeGuardMs !== undefined ? deps.reapMtimeGuardMs : REAP_MTIME_GUARD_MS;
704
+ const results = [];
705
+ // 1. Discover the .git/worktrees/ admin directory.
706
+ const gitDir = execGit(['rev-parse', '--git-dir'], { cwd: repoRoot });
707
+ if (!gitResultOk(gitDir))
708
+ return results;
709
+ const gitDirPath = node_path_1.default.resolve(repoRoot, gitDir.stdout.trim());
710
+ const worktreesAdminDir = node_path_1.default.join(gitDirPath, 'worktrees');
711
+ const entries = readDirSafe(worktreesAdminDir);
712
+ if (!entries)
713
+ return results;
714
+ // 2. Discover the default branch (main/master/etc) tip.
715
+ const defaultBranchResult = execGit(['symbolic-ref', '--quiet', '--short', 'refs/remotes/origin/HEAD'], { cwd: repoRoot });
716
+ let mainTip;
717
+ if (gitResultOk(defaultBranchResult)) {
718
+ // Remote default branch is known — use it exclusively.
719
+ const branchName = defaultBranchResult.stdout.trim().replace(/^origin\//, '');
720
+ const r = execGit(['rev-parse', `refs/remotes/origin/${branchName}`], { cwd: repoRoot });
721
+ if (!gitResultOk(r))
722
+ return results; // remote ref unresolvable — fail closed
723
+ mainTip = r.stdout.trim();
724
+ }
725
+ else {
726
+ // No remote configured (local-only repo, e.g. test fixtures).
727
+ const hasRemote = execGit(['remote'], { cwd: repoRoot });
728
+ if (gitResultOk(hasRemote) && hasRemote.stdout.trim()) {
729
+ // Remote exists but origin/HEAD not set — ambiguous; fail closed.
730
+ return results;
731
+ }
732
+ // Build candidate list: init.defaultBranch config, HEAD symref, then main, master.
733
+ const candidateBranches = [];
734
+ const configResult = execGit(['config', '--get', 'init.defaultBranch'], { cwd: repoRoot });
735
+ if (gitResultOk(configResult) && configResult.stdout.trim()) {
736
+ candidateBranches.push(configResult.stdout.trim());
737
+ }
738
+ const headSymref = execGit(['symbolic-ref', '--quiet', '--short', 'HEAD'], { cwd: repoRoot });
739
+ if (gitResultOk(headSymref) && headSymref.stdout.trim()) {
740
+ const headBranch = headSymref.stdout.trim();
741
+ if (!candidateBranches.includes(headBranch)) {
742
+ candidateBranches.push(headBranch);
743
+ }
744
+ }
745
+ for (const b of ['main', 'master']) {
746
+ if (!candidateBranches.includes(b))
747
+ candidateBranches.push(b);
748
+ }
749
+ for (const candidate of candidateBranches) {
750
+ const r = execGit(['rev-parse', candidate], { cwd: repoRoot });
751
+ if (gitResultOk(r)) {
752
+ mainTip = r.stdout.trim();
753
+ break;
754
+ }
755
+ }
756
+ if (!mainTip)
757
+ return results;
758
+ }
759
+ // 3. Build a canonical-path → listed-path index from git worktree list.
760
+ const listedResult = execGit(['worktree', 'list', '--porcelain'], { cwd: repoRoot });
761
+ const canonicalToListed = new Map();
762
+ if (gitResultOk(listedResult)) {
763
+ const normalizedListed = listedResult.stdout.replace(/\r\n/g, '\n');
764
+ for (const block of normalizedListed.split('\n\n').filter(Boolean)) {
765
+ const wtLine = block.split('\n').find((l) => l.startsWith('worktree '));
766
+ if (!wtLine)
767
+ continue;
768
+ const listed = wtLine.slice('worktree '.length).trim();
769
+ try {
770
+ const canonical = node_fs_1.default.realpathSync.native(listed);
771
+ canonicalToListed.set(canonical, listed);
772
+ }
773
+ catch {
774
+ // If the path doesn't exist (already removed), skip silently.
775
+ }
776
+ }
777
+ }
778
+ // 4. Process each worktree admin entry that has a 'locked' file.
779
+ for (const entryName of entries) {
780
+ const adminDir = node_path_1.default.join(worktreesAdminDir, entryName);
781
+ const lockedFile = node_path_1.default.join(adminDir, 'locked');
782
+ const lockedContent = readFileSafe(lockedFile);
783
+ if (lockedContent === null)
784
+ continue; // no lock file — not our concern
785
+ // Resolve the actual worktree path from the gitdir pointer.
786
+ const gitdirFile = node_path_1.default.join(adminDir, 'gitdir');
787
+ const gitdirContent = readFileSafe(gitdirFile);
788
+ if (!gitdirContent)
789
+ continue;
790
+ const resolvedGitFile = node_path_1.default.resolve(adminDir, gitdirContent.trim());
791
+ const worktreePath = node_path_1.default.basename(resolvedGitFile) === '.git'
792
+ ? node_path_1.default.dirname(resolvedGitFile)
793
+ : resolvedGitFile;
794
+ // Look up the git-list path (the path git knows about) for use in
795
+ // git worktree unlock/remove commands.
796
+ let gitKnownPath = worktreePath;
797
+ try {
798
+ const canonical = node_fs_1.default.realpathSync.native(worktreePath);
799
+ gitKnownPath = canonicalToListed.get(canonical) || worktreePath;
800
+ }
801
+ catch {
802
+ // worktreePath may not exist yet (already removed); use as-is.
803
+ }
804
+ // 4a. Stale-lock guard: skip if lock is too fresh (PID recycling / race).
805
+ const lockMtime = mtimeSafe(lockedFile);
806
+ if (!lockMtime || Date.now() - lockMtime.getTime() < reapMtimeGuardMs) {
807
+ results.push({ path: worktreePath, status: 'skipped', reason: 'lock_too_fresh' });
808
+ continue;
809
+ }
810
+ // 4b. PID liveness check.
811
+ const pidStr = lockedContent.trim().match(/^\d+/)?.[0];
812
+ if (!pidStr) {
813
+ results.push({ path: worktreePath, status: 'skipped', reason: 'lock_owner_unknown' });
814
+ continue;
815
+ }
816
+ const pid = parseInt(pidStr, 10);
817
+ let pidIsAlive;
818
+ try {
819
+ pidIsAlive = Number.isNaN(pid) || isPidAliveCheck(pid);
820
+ }
821
+ catch {
822
+ pidIsAlive = true; // Cannot determine liveness — treat as alive, do not reap.
823
+ }
824
+ if (pidIsAlive) {
825
+ results.push({ path: worktreePath, status: 'skipped', reason: 'pid_alive' });
826
+ continue;
827
+ }
828
+ // 4c. Ancestry guard: branch-tip must be reachable from main (fail closed).
829
+ let branchTip;
830
+ {
831
+ const headContent = readFileSafe(node_path_1.default.join(adminDir, 'HEAD'));
832
+ if (!headContent) {
833
+ results.push({ path: worktreePath, status: 'skipped', reason: 'cannot_resolve_branch_tip' });
834
+ continue;
835
+ }
836
+ const trimmed = headContent.trim();
837
+ if (trimmed.startsWith('ref: refs/heads/')) {
838
+ // Symbolic ref — resolve to commit SHA via git
839
+ const branchName = trimmed.slice('ref: refs/heads/'.length);
840
+ const resolveResult = execGit(['rev-parse', `refs/heads/${branchName}`], { cwd: repoRoot });
841
+ if (!gitResultOk(resolveResult)) {
842
+ results.push({ path: worktreePath, status: 'skipped', reason: 'cannot_resolve_branch_tip' });
843
+ continue;
844
+ }
845
+ branchTip = resolveResult.stdout.trim();
846
+ }
847
+ else if (/^[0-9a-f]{40}$/i.test(trimmed)) {
848
+ // Detached HEAD — bare SHA
849
+ branchTip = trimmed;
850
+ }
851
+ else {
852
+ results.push({ path: worktreePath, status: 'skipped', reason: 'cannot_resolve_branch_tip' });
853
+ continue;
854
+ }
855
+ }
856
+ const ancestorCheck = execGit(['merge-base', '--is-ancestor', branchTip, mainTip], { cwd: repoRoot });
857
+ if (!gitResultOk(ancestorCheck)) {
858
+ results.push({ path: worktreePath, status: 'skipped', reason: 'branch_not_merged' });
859
+ continue;
860
+ }
861
+ // 4d. Reap: unlock → remove --force.
862
+ execGit(['worktree', 'unlock', gitKnownPath], { cwd: repoRoot }); // ignore failure (already unlocked)
863
+ const removeResult = execGit(['worktree', 'remove', gitKnownPath, '--force'], { cwd: repoRoot });
864
+ if (!gitResultOk(removeResult)) {
865
+ results.push({ path: worktreePath, status: 'skipped', reason: 'remove_failed' });
866
+ continue;
867
+ }
868
+ results.push({ path: gitKnownPath, status: 'reaped', reason: 'pid_dead_and_merged' });
869
+ }
870
+ // 5. Always prune stale metadata (handles missing-on-disk entries).
871
+ execGit(['worktree', 'prune'], { cwd: repoRoot });
872
+ return results;
873
+ }
874
+ // ─── reapOrphanWorktrees deps helpers ─────────────────────────────────────────
875
+ function defaultIsPidAlive(pid) {
876
+ try {
877
+ process.kill(pid, 0);
878
+ return true;
879
+ }
880
+ catch (err) {
881
+ if (err && err.code === 'EPERM')
882
+ return true;
883
+ return false;
884
+ }
885
+ }
886
+ function defaultReadDirSafe(dir) {
887
+ try {
888
+ return node_fs_1.default.readdirSync(dir);
889
+ }
890
+ catch {
891
+ return null;
892
+ }
893
+ }
894
+ function defaultReadFileSafe(file) {
895
+ try {
896
+ return node_fs_1.default.readFileSync(file, 'utf8');
897
+ }
898
+ catch {
899
+ return null;
900
+ }
901
+ }
902
+ function defaultMtimeSafe(file) {
903
+ try {
904
+ return node_fs_1.default.statSync(file).mtime;
905
+ }
906
+ catch {
907
+ return null;
908
+ }
909
+ }
910
+ function cmdWorktreeReapOrphans(cwd) {
911
+ let result;
912
+ try {
913
+ result = reapOrphanWorktrees(cwd);
914
+ }
915
+ catch (err) {
916
+ // Surface failure as a one-line warning; keep exit-zero so workflows don't break.
917
+ process.stderr.write(`[gsd] worktree.reap-orphans failed: ${err && err.message ? err.message : String(err)}\n`);
918
+ result = [];
919
+ }
920
+ const skippedCount = result.filter((r) => r.status === 'skipped').length;
921
+ if (skippedCount > 0) {
922
+ // Surface skipped entries so operators are aware of unresolved orphans.
923
+ process.stderr.write(`[gsd] worktree.reap-orphans: ${skippedCount} orphan(s) skipped (run with DEBUG=1 for details)\n`);
924
+ }
925
+ process.stdout.write(`${JSON.stringify({ ok: true, reaped: result.filter((r) => r.status === 'reaped').length, entries: result }, null, 2)}\n`);
926
+ }
927
+ // Unused exports kept for API compatibility
928
+ void parseWorktreeListPaths;
929
+ module.exports = {
930
+ resolveWorktreeContext,
931
+ parseWorktreePorcelain,
932
+ planWorktreePrune,
933
+ executeWorktreePrunePlan,
934
+ listLinkedWorktreePaths,
935
+ inspectWorktreeHealth,
936
+ snapshotWorktreeInventory,
937
+ normalizeCleanupManifest,
938
+ planWorktreeWaveCleanup,
939
+ executeWorktreeWaveCleanupPlan,
940
+ cmdWorktreeCleanupWave,
941
+ reapOrphanWorktrees,
942
+ cmdWorktreeReapOrphans,
943
+ };