@opengsd/gsd-core 1.2.0-rc.1

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 (503) hide show
  1. package/LICENSE +21 -0
  2. package/README.ja-JP.md +870 -0
  3. package/README.ko-KR.md +861 -0
  4. package/README.md +301 -0
  5. package/README.pt-BR.md +492 -0
  6. package/README.zh-CN.md +842 -0
  7. package/agents/gsd-advisor-researcher.md +127 -0
  8. package/agents/gsd-ai-researcher.md +133 -0
  9. package/agents/gsd-assumptions-analyzer.md +105 -0
  10. package/agents/gsd-code-fixer.md +668 -0
  11. package/agents/gsd-code-reviewer.md +387 -0
  12. package/agents/gsd-codebase-mapper.md +853 -0
  13. package/agents/gsd-debug-session-manager.md +314 -0
  14. package/agents/gsd-debugger.md +1452 -0
  15. package/agents/gsd-doc-classifier.md +168 -0
  16. package/agents/gsd-doc-synthesizer.md +204 -0
  17. package/agents/gsd-doc-verifier.md +217 -0
  18. package/agents/gsd-doc-writer.md +615 -0
  19. package/agents/gsd-domain-researcher.md +153 -0
  20. package/agents/gsd-eval-auditor.md +191 -0
  21. package/agents/gsd-eval-planner.md +154 -0
  22. package/agents/gsd-executor.md +772 -0
  23. package/agents/gsd-framework-selector.md +160 -0
  24. package/agents/gsd-integration-checker.md +470 -0
  25. package/agents/gsd-intel-updater.md +342 -0
  26. package/agents/gsd-nyquist-auditor.md +203 -0
  27. package/agents/gsd-pattern-mapper.md +335 -0
  28. package/agents/gsd-phase-researcher.md +928 -0
  29. package/agents/gsd-plan-checker.md +978 -0
  30. package/agents/gsd-planner.md +1218 -0
  31. package/agents/gsd-project-researcher.md +677 -0
  32. package/agents/gsd-research-synthesizer.md +255 -0
  33. package/agents/gsd-roadmapper.md +688 -0
  34. package/agents/gsd-security-auditor.md +155 -0
  35. package/agents/gsd-ui-auditor.md +495 -0
  36. package/agents/gsd-ui-checker.md +309 -0
  37. package/agents/gsd-ui-researcher.md +380 -0
  38. package/agents/gsd-user-profiler.md +171 -0
  39. package/agents/gsd-verifier.md +917 -0
  40. package/bin/install.js +10936 -0
  41. package/bin/lib/ui-safety-gate.cjs +107 -0
  42. package/commands/gsd/add-tests.md +42 -0
  43. package/commands/gsd/ai-integration-phase.md +37 -0
  44. package/commands/gsd/audit-fix.md +34 -0
  45. package/commands/gsd/audit-milestone.md +37 -0
  46. package/commands/gsd/audit-uat.md +24 -0
  47. package/commands/gsd/autonomous.md +46 -0
  48. package/commands/gsd/capture.md +62 -0
  49. package/commands/gsd/cleanup.md +24 -0
  50. package/commands/gsd/code-review.md +59 -0
  51. package/commands/gsd/complete-milestone.md +143 -0
  52. package/commands/gsd/config.md +56 -0
  53. package/commands/gsd/debug.md +52 -0
  54. package/commands/gsd/discuss-phase.md +76 -0
  55. package/commands/gsd/docs-update.md +49 -0
  56. package/commands/gsd/eval-review.md +33 -0
  57. package/commands/gsd/execute-phase.md +64 -0
  58. package/commands/gsd/explore.md +27 -0
  59. package/commands/gsd/extract-learnings.md +23 -0
  60. package/commands/gsd/fast.md +31 -0
  61. package/commands/gsd/forensics.md +57 -0
  62. package/commands/gsd/graphify.md +199 -0
  63. package/commands/gsd/health.md +31 -0
  64. package/commands/gsd/help.md +28 -0
  65. package/commands/gsd/import.md +41 -0
  66. package/commands/gsd/inbox.md +39 -0
  67. package/commands/gsd/ingest-docs.md +42 -0
  68. package/commands/gsd/manager.md +45 -0
  69. package/commands/gsd/map-codebase.md +83 -0
  70. package/commands/gsd/milestone-summary.md +51 -0
  71. package/commands/gsd/mvp-phase.md +45 -0
  72. package/commands/gsd/new-milestone.md +45 -0
  73. package/commands/gsd/new-project.md +47 -0
  74. package/commands/gsd/ns-context.md +23 -0
  75. package/commands/gsd/ns-ideate.md +24 -0
  76. package/commands/gsd/ns-manage.md +29 -0
  77. package/commands/gsd/ns-project.md +22 -0
  78. package/commands/gsd/ns-review.md +26 -0
  79. package/commands/gsd/ns-workflow.md +28 -0
  80. package/commands/gsd/pause-work.md +43 -0
  81. package/commands/gsd/phase.md +56 -0
  82. package/commands/gsd/plan-phase.md +62 -0
  83. package/commands/gsd/plan-review-convergence.md +59 -0
  84. package/commands/gsd/pr-branch.md +26 -0
  85. package/commands/gsd/profile-user.md +46 -0
  86. package/commands/gsd/progress.md +47 -0
  87. package/commands/gsd/quick.md +174 -0
  88. package/commands/gsd/resume-work.md +30 -0
  89. package/commands/gsd/review-backlog.md +63 -0
  90. package/commands/gsd/review.md +41 -0
  91. package/commands/gsd/secure-phase.md +36 -0
  92. package/commands/gsd/settings.md +29 -0
  93. package/commands/gsd/ship.md +24 -0
  94. package/commands/gsd/sketch.md +60 -0
  95. package/commands/gsd/spec-phase.md +63 -0
  96. package/commands/gsd/spike.md +57 -0
  97. package/commands/gsd/stats.md +19 -0
  98. package/commands/gsd/surface.md +155 -0
  99. package/commands/gsd/thread.md +24 -0
  100. package/commands/gsd/ui-phase.md +35 -0
  101. package/commands/gsd/ui-review.md +33 -0
  102. package/commands/gsd/ultraplan-phase.md +34 -0
  103. package/commands/gsd/undo.md +35 -0
  104. package/commands/gsd/update.md +48 -0
  105. package/commands/gsd/validate-phase.md +36 -0
  106. package/commands/gsd/verify-work.md +39 -0
  107. package/commands/gsd/workspace.md +52 -0
  108. package/commands/gsd/workstreams.md +70 -0
  109. package/get-shit-done/bin/check-latest-version.cjs +106 -0
  110. package/get-shit-done/bin/gsd-tools.cjs +1676 -0
  111. package/get-shit-done/bin/lib/active-workstream-store.cjs +302 -0
  112. package/get-shit-done/bin/lib/adr-parser.cjs +394 -0
  113. package/get-shit-done/bin/lib/agent-command-router.cjs +65 -0
  114. package/get-shit-done/bin/lib/artifacts.cjs +53 -0
  115. package/get-shit-done/bin/lib/audit.cjs +755 -0
  116. package/get-shit-done/bin/lib/check-command-router.cjs +333 -0
  117. package/get-shit-done/bin/lib/cjs-command-router-adapter.cjs +118 -0
  118. package/get-shit-done/bin/lib/clock.cjs +96 -0
  119. package/get-shit-done/bin/lib/clusters.cjs +135 -0
  120. package/get-shit-done/bin/lib/code-review-flags.cjs +74 -0
  121. package/get-shit-done/bin/lib/command-aliases.cjs +815 -0
  122. package/get-shit-done/bin/lib/command-arg-projection.cjs +62 -0
  123. package/get-shit-done/bin/lib/command-routing-hub.cjs +388 -0
  124. package/get-shit-done/bin/lib/commands.cjs +1188 -0
  125. package/get-shit-done/bin/lib/config-schema.cjs +31 -0
  126. package/get-shit-done/bin/lib/config.cjs +728 -0
  127. package/get-shit-done/bin/lib/configuration.cjs +248 -0
  128. package/get-shit-done/bin/lib/context-utilization.cjs +47 -0
  129. package/get-shit-done/bin/lib/core.cjs +2121 -0
  130. package/get-shit-done/bin/lib/decisions.cjs +116 -0
  131. package/get-shit-done/bin/lib/docs.cjs +270 -0
  132. package/get-shit-done/bin/lib/drift.cjs +388 -0
  133. package/get-shit-done/bin/lib/fallow-runner.cjs +109 -0
  134. package/get-shit-done/bin/lib/frontmatter.cjs +389 -0
  135. package/get-shit-done/bin/lib/gap-checker.cjs +205 -0
  136. package/get-shit-done/bin/lib/graphify.cjs +592 -0
  137. package/get-shit-done/bin/lib/gsd2-import.cjs +514 -0
  138. package/get-shit-done/bin/lib/init-command-router.cjs +58 -0
  139. package/get-shit-done/bin/lib/init.cjs +2112 -0
  140. package/get-shit-done/bin/lib/install-profiles.cjs +603 -0
  141. package/get-shit-done/bin/lib/installer-migration-authoring.cjs +117 -0
  142. package/get-shit-done/bin/lib/installer-migration-report.cjs +354 -0
  143. package/get-shit-done/bin/lib/installer-migrations/000-first-time-baseline.cjs +220 -0
  144. package/get-shit-done/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +41 -0
  145. package/get-shit-done/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +80 -0
  146. package/get-shit-done/bin/lib/installer-migrations.cjs +778 -0
  147. package/get-shit-done/bin/lib/intel.cjs +708 -0
  148. package/get-shit-done/bin/lib/learnings.cjs +421 -0
  149. package/get-shit-done/bin/lib/milestone.cjs +314 -0
  150. package/get-shit-done/bin/lib/model-catalog.cjs +212 -0
  151. package/get-shit-done/bin/lib/model-profiles.cjs +31 -0
  152. package/get-shit-done/bin/lib/observability/event.cjs +82 -0
  153. package/get-shit-done/bin/lib/observability/logger.cjs +174 -0
  154. package/get-shit-done/bin/lib/observability/redaction.cjs +50 -0
  155. package/get-shit-done/bin/lib/package-identity.cjs +31 -0
  156. package/get-shit-done/bin/lib/phase-command-router.cjs +191 -0
  157. package/get-shit-done/bin/lib/phase-lifecycle.cjs +80 -0
  158. package/get-shit-done/bin/lib/phase.cjs +1607 -0
  159. package/get-shit-done/bin/lib/phases-command-router.cjs +39 -0
  160. package/get-shit-done/bin/lib/plan-scan.cjs +97 -0
  161. package/get-shit-done/bin/lib/planning-workspace.cjs +238 -0
  162. package/get-shit-done/bin/lib/profile-output.cjs +1141 -0
  163. package/get-shit-done/bin/lib/profile-pipeline.cjs +539 -0
  164. package/get-shit-done/bin/lib/project-root.cjs +112 -0
  165. package/get-shit-done/bin/lib/prompt-budget.cjs +399 -0
  166. package/get-shit-done/bin/lib/review-reviewer-selection.cjs +125 -0
  167. package/get-shit-done/bin/lib/roadmap-command-router.cjs +28 -0
  168. package/get-shit-done/bin/lib/roadmap.cjs +650 -0
  169. package/get-shit-done/bin/lib/runtime-artifact-layout.cjs +301 -0
  170. package/get-shit-done/bin/lib/runtime-homes.cjs +222 -0
  171. package/get-shit-done/bin/lib/runtime-name-policy.cjs +83 -0
  172. package/get-shit-done/bin/lib/runtime-slash.cjs +112 -0
  173. package/get-shit-done/bin/lib/schema-detect.cjs +165 -0
  174. package/get-shit-done/bin/lib/secrets.cjs +32 -0
  175. package/get-shit-done/bin/lib/security.cjs +600 -0
  176. package/get-shit-done/bin/lib/semver-compare.cjs +35 -0
  177. package/get-shit-done/bin/lib/shell-command-projection.cjs +500 -0
  178. package/get-shit-done/bin/lib/state-command-router.cjs +252 -0
  179. package/get-shit-done/bin/lib/state-document.cjs +263 -0
  180. package/get-shit-done/bin/lib/state.cjs +2038 -0
  181. package/get-shit-done/bin/lib/surface.cjs +470 -0
  182. package/get-shit-done/bin/lib/task-command-router.cjs +81 -0
  183. package/get-shit-done/bin/lib/template.cjs +228 -0
  184. package/get-shit-done/bin/lib/uat.cjs +289 -0
  185. package/get-shit-done/bin/lib/update-context.cjs +209 -0
  186. package/get-shit-done/bin/lib/validate-command-router.cjs +83 -0
  187. package/get-shit-done/bin/lib/validate.cjs +92 -0
  188. package/get-shit-done/bin/lib/verify-command-router.cjs +40 -0
  189. package/get-shit-done/bin/lib/verify.cjs +1511 -0
  190. package/get-shit-done/bin/lib/workstream-inventory-builder.cjs +74 -0
  191. package/get-shit-done/bin/lib/workstream-inventory.cjs +146 -0
  192. package/get-shit-done/bin/lib/workstream-name-policy.cjs +94 -0
  193. package/get-shit-done/bin/lib/workstream.cjs +389 -0
  194. package/get-shit-done/bin/lib/worktree-safety.cjs +985 -0
  195. package/get-shit-done/bin/shared/config-defaults.manifest.json +97 -0
  196. package/get-shit-done/bin/shared/config-schema.manifest.json +175 -0
  197. package/get-shit-done/bin/shared/model-catalog.json +122 -0
  198. package/get-shit-done/bin/shared/runtime-aliases.manifest.json +75 -0
  199. package/get-shit-done/bin/verify-reapply-patches.cjs +352 -0
  200. package/get-shit-done/contexts/dev.md +21 -0
  201. package/get-shit-done/contexts/research.md +22 -0
  202. package/get-shit-done/contexts/review.md +23 -0
  203. package/get-shit-done/references/agent-contracts.md +79 -0
  204. package/get-shit-done/references/ai-evals.md +156 -0
  205. package/get-shit-done/references/ai-frameworks.md +186 -0
  206. package/get-shit-done/references/artifact-types.md +131 -0
  207. package/get-shit-done/references/autonomous-smart-discuss.md +277 -0
  208. package/get-shit-done/references/checkpoints.md +814 -0
  209. package/get-shit-done/references/common-bug-patterns.md +114 -0
  210. package/get-shit-done/references/context-budget.md +85 -0
  211. package/get-shit-done/references/continuation-format.md +253 -0
  212. package/get-shit-done/references/debugger-philosophy.md +76 -0
  213. package/get-shit-done/references/decimal-phase-calculation.md +64 -0
  214. package/get-shit-done/references/doc-conflict-engine.md +91 -0
  215. package/get-shit-done/references/domain-probes.md +125 -0
  216. package/get-shit-done/references/execute-mvp-tdd.md +81 -0
  217. package/get-shit-done/references/executor-examples.md +110 -0
  218. package/get-shit-done/references/few-shot-examples/plan-checker.md +73 -0
  219. package/get-shit-done/references/few-shot-examples/verifier.md +109 -0
  220. package/get-shit-done/references/gate-prompts.md +100 -0
  221. package/get-shit-done/references/gates.md +70 -0
  222. package/get-shit-done/references/git-integration.md +298 -0
  223. package/get-shit-done/references/git-planning-commit.md +40 -0
  224. package/get-shit-done/references/ios-scaffold.md +123 -0
  225. package/get-shit-done/references/mandatory-initial-read.md +2 -0
  226. package/get-shit-done/references/model-profile-resolution.md +38 -0
  227. package/get-shit-done/references/model-profiles.md +245 -0
  228. package/get-shit-done/references/mvp-concepts.md +49 -0
  229. package/get-shit-done/references/phase-argument-parsing.md +61 -0
  230. package/get-shit-done/references/planner-antipatterns.md +89 -0
  231. package/get-shit-done/references/planner-chunked.md +49 -0
  232. package/get-shit-done/references/planner-gap-closure.md +62 -0
  233. package/get-shit-done/references/planner-graphify-auto-update.md +67 -0
  234. package/get-shit-done/references/planner-human-verify-mode.md +57 -0
  235. package/get-shit-done/references/planner-interface-context.md +62 -0
  236. package/get-shit-done/references/planner-mvp-mode.md +53 -0
  237. package/get-shit-done/references/planner-reviews.md +39 -0
  238. package/get-shit-done/references/planner-revision.md +87 -0
  239. package/get-shit-done/references/planner-source-audit.md +73 -0
  240. package/get-shit-done/references/planning-config.md +471 -0
  241. package/get-shit-done/references/project-skills-discovery.md +19 -0
  242. package/get-shit-done/references/questioning.md +162 -0
  243. package/get-shit-done/references/revision-loop.md +97 -0
  244. package/get-shit-done/references/scout-codebase.md +51 -0
  245. package/get-shit-done/references/skeleton-template.md +48 -0
  246. package/get-shit-done/references/sketch-interactivity.md +41 -0
  247. package/get-shit-done/references/sketch-theme-system.md +94 -0
  248. package/get-shit-done/references/sketch-tooling.md +45 -0
  249. package/get-shit-done/references/sketch-variant-patterns.md +81 -0
  250. package/get-shit-done/references/spidr-splitting.md +69 -0
  251. package/get-shit-done/references/tdd.md +330 -0
  252. package/get-shit-done/references/thinking-models-debug.md +44 -0
  253. package/get-shit-done/references/thinking-models-execution.md +50 -0
  254. package/get-shit-done/references/thinking-models-planning.md +62 -0
  255. package/get-shit-done/references/thinking-models-research.md +50 -0
  256. package/get-shit-done/references/thinking-models-verification.md +55 -0
  257. package/get-shit-done/references/thinking-partner.md +96 -0
  258. package/get-shit-done/references/ui-brand.md +160 -0
  259. package/get-shit-done/references/universal-anti-patterns.md +63 -0
  260. package/get-shit-done/references/user-profiling.md +681 -0
  261. package/get-shit-done/references/user-story-template.md +58 -0
  262. package/get-shit-done/references/verification-overrides.md +227 -0
  263. package/get-shit-done/references/verification-patterns.md +612 -0
  264. package/get-shit-done/references/verify-mvp-mode.md +85 -0
  265. package/get-shit-done/references/workstream-flag.md +111 -0
  266. package/get-shit-done/references/worktree-path-safety.md +89 -0
  267. package/get-shit-done/templates/AI-SPEC.md +246 -0
  268. package/get-shit-done/templates/DEBUG.md +169 -0
  269. package/get-shit-done/templates/README.md +77 -0
  270. package/get-shit-done/templates/SECURITY.md +61 -0
  271. package/get-shit-done/templates/UAT.md +265 -0
  272. package/get-shit-done/templates/UI-SPEC.md +100 -0
  273. package/get-shit-done/templates/VALIDATION.md +76 -0
  274. package/get-shit-done/templates/claude-md.md +145 -0
  275. package/get-shit-done/templates/codebase/architecture.md +255 -0
  276. package/get-shit-done/templates/codebase/concerns.md +310 -0
  277. package/get-shit-done/templates/codebase/conventions.md +307 -0
  278. package/get-shit-done/templates/codebase/integrations.md +280 -0
  279. package/get-shit-done/templates/codebase/stack.md +186 -0
  280. package/get-shit-done/templates/codebase/structure.md +285 -0
  281. package/get-shit-done/templates/codebase/testing.md +480 -0
  282. package/get-shit-done/templates/config.json +62 -0
  283. package/get-shit-done/templates/context.md +352 -0
  284. package/get-shit-done/templates/continue-here.md +78 -0
  285. package/get-shit-done/templates/copilot-instructions.md +7 -0
  286. package/get-shit-done/templates/debug-subagent-prompt.md +91 -0
  287. package/get-shit-done/templates/dev-preferences.md +21 -0
  288. package/get-shit-done/templates/discovery.md +146 -0
  289. package/get-shit-done/templates/discussion-log.md +63 -0
  290. package/get-shit-done/templates/milestone-archive.md +123 -0
  291. package/get-shit-done/templates/milestone.md +115 -0
  292. package/get-shit-done/templates/phase-prompt.md +610 -0
  293. package/get-shit-done/templates/planner-subagent-prompt.md +117 -0
  294. package/get-shit-done/templates/project.md +186 -0
  295. package/get-shit-done/templates/requirements.md +231 -0
  296. package/get-shit-done/templates/research-project/ARCHITECTURE.md +204 -0
  297. package/get-shit-done/templates/research-project/FEATURES.md +147 -0
  298. package/get-shit-done/templates/research-project/PITFALLS.md +200 -0
  299. package/get-shit-done/templates/research-project/STACK.md +120 -0
  300. package/get-shit-done/templates/research-project/SUMMARY.md +170 -0
  301. package/get-shit-done/templates/research.md +592 -0
  302. package/get-shit-done/templates/retrospective.md +54 -0
  303. package/get-shit-done/templates/roadmap.md +202 -0
  304. package/get-shit-done/templates/spec.md +307 -0
  305. package/get-shit-done/templates/state.md +195 -0
  306. package/get-shit-done/templates/summary-complex.md +59 -0
  307. package/get-shit-done/templates/summary-minimal.md +41 -0
  308. package/get-shit-done/templates/summary-standard.md +48 -0
  309. package/get-shit-done/templates/summary.md +248 -0
  310. package/get-shit-done/templates/user-profile.md +146 -0
  311. package/get-shit-done/templates/user-setup.md +311 -0
  312. package/get-shit-done/templates/verification-report.md +322 -0
  313. package/get-shit-done/workflows/_runtime-launcher.snippet.sh +1 -0
  314. package/get-shit-done/workflows/add-backlog.md +91 -0
  315. package/get-shit-done/workflows/add-phase.md +113 -0
  316. package/get-shit-done/workflows/add-tests.md +355 -0
  317. package/get-shit-done/workflows/add-todo.md +161 -0
  318. package/get-shit-done/workflows/ai-integration-phase.md +295 -0
  319. package/get-shit-done/workflows/analyze-dependencies.md +96 -0
  320. package/get-shit-done/workflows/audit-fix.md +178 -0
  321. package/get-shit-done/workflows/audit-milestone.md +358 -0
  322. package/get-shit-done/workflows/audit-uat.md +110 -0
  323. package/get-shit-done/workflows/autonomous.md +795 -0
  324. package/get-shit-done/workflows/check-todos.md +180 -0
  325. package/get-shit-done/workflows/cleanup.md +155 -0
  326. package/get-shit-done/workflows/code-review-fix.md +502 -0
  327. package/get-shit-done/workflows/code-review.md +656 -0
  328. package/get-shit-done/workflows/complete-milestone.md +855 -0
  329. package/get-shit-done/workflows/debug.md +232 -0
  330. package/get-shit-done/workflows/diagnose-issues.md +241 -0
  331. package/get-shit-done/workflows/discovery-phase.md +291 -0
  332. package/get-shit-done/workflows/discuss-phase/modes/advisor.md +176 -0
  333. package/get-shit-done/workflows/discuss-phase/modes/all.md +28 -0
  334. package/get-shit-done/workflows/discuss-phase/modes/analyze.md +44 -0
  335. package/get-shit-done/workflows/discuss-phase/modes/auto.md +57 -0
  336. package/get-shit-done/workflows/discuss-phase/modes/batch.md +52 -0
  337. package/get-shit-done/workflows/discuss-phase/modes/chain.md +98 -0
  338. package/get-shit-done/workflows/discuss-phase/modes/default.md +141 -0
  339. package/get-shit-done/workflows/discuss-phase/modes/power.md +44 -0
  340. package/get-shit-done/workflows/discuss-phase/modes/text.md +55 -0
  341. package/get-shit-done/workflows/discuss-phase/templates/checkpoint.json +18 -0
  342. package/get-shit-done/workflows/discuss-phase/templates/context.md +136 -0
  343. package/get-shit-done/workflows/discuss-phase/templates/discussion-log.md +50 -0
  344. package/get-shit-done/workflows/discuss-phase-assumptions.md +675 -0
  345. package/get-shit-done/workflows/discuss-phase-power.md +291 -0
  346. package/get-shit-done/workflows/discuss-phase.md +499 -0
  347. package/get-shit-done/workflows/do.md +111 -0
  348. package/get-shit-done/workflows/docs-update.md +1162 -0
  349. package/get-shit-done/workflows/edit-phase.md +295 -0
  350. package/get-shit-done/workflows/eval-review.md +156 -0
  351. package/get-shit-done/workflows/execute-phase/steps/codebase-drift-gate.md +82 -0
  352. package/get-shit-done/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  353. package/get-shit-done/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  354. package/get-shit-done/workflows/execute-phase.md +1709 -0
  355. package/get-shit-done/workflows/execute-plan.md +526 -0
  356. package/get-shit-done/workflows/explore.md +144 -0
  357. package/get-shit-done/workflows/extract-learnings.md +243 -0
  358. package/get-shit-done/workflows/fast.md +124 -0
  359. package/get-shit-done/workflows/forensics.md +279 -0
  360. package/get-shit-done/workflows/graduation.md +196 -0
  361. package/get-shit-done/workflows/health.md +224 -0
  362. package/get-shit-done/workflows/help/modes/brief.md +22 -0
  363. package/get-shit-done/workflows/help/modes/default.md +50 -0
  364. package/get-shit-done/workflows/help/modes/full.md +784 -0
  365. package/get-shit-done/workflows/help/modes/topic.md +74 -0
  366. package/get-shit-done/workflows/help.md +24 -0
  367. package/get-shit-done/workflows/import.md +254 -0
  368. package/get-shit-done/workflows/inbox.md +387 -0
  369. package/get-shit-done/workflows/ingest-docs.md +339 -0
  370. package/get-shit-done/workflows/insert-phase.md +152 -0
  371. package/get-shit-done/workflows/list-phase-assumptions.md +178 -0
  372. package/get-shit-done/workflows/list-workspaces.md +57 -0
  373. package/get-shit-done/workflows/manager.md +393 -0
  374. package/get-shit-done/workflows/map-codebase.md +444 -0
  375. package/get-shit-done/workflows/milestone-summary.md +224 -0
  376. package/get-shit-done/workflows/mvp-phase.md +222 -0
  377. package/get-shit-done/workflows/new-milestone.md +635 -0
  378. package/get-shit-done/workflows/new-project.md +1555 -0
  379. package/get-shit-done/workflows/new-workspace.md +240 -0
  380. package/get-shit-done/workflows/next.md +299 -0
  381. package/get-shit-done/workflows/node-repair.md +92 -0
  382. package/get-shit-done/workflows/note.md +158 -0
  383. package/get-shit-done/workflows/pause-work.md +244 -0
  384. package/get-shit-done/workflows/plan-milestone-gaps.md +281 -0
  385. package/get-shit-done/workflows/plan-phase.md +1809 -0
  386. package/get-shit-done/workflows/plan-review-convergence.md +346 -0
  387. package/get-shit-done/workflows/plant-seed.md +230 -0
  388. package/get-shit-done/workflows/pr-branch.md +157 -0
  389. package/get-shit-done/workflows/profile-user.md +453 -0
  390. package/get-shit-done/workflows/progress.md +699 -0
  391. package/get-shit-done/workflows/quick.md +1039 -0
  392. package/get-shit-done/workflows/reapply-patches.md +426 -0
  393. package/get-shit-done/workflows/remove-phase.md +156 -0
  394. package/get-shit-done/workflows/remove-workspace.md +108 -0
  395. package/get-shit-done/workflows/resume-project.md +332 -0
  396. package/get-shit-done/workflows/review.md +623 -0
  397. package/get-shit-done/workflows/scan.md +105 -0
  398. package/get-shit-done/workflows/secure-phase.md +180 -0
  399. package/get-shit-done/workflows/session-report.md +146 -0
  400. package/get-shit-done/workflows/settings-advanced.md +620 -0
  401. package/get-shit-done/workflows/settings-integrations.md +312 -0
  402. package/get-shit-done/workflows/settings.md +552 -0
  403. package/get-shit-done/workflows/ship.md +356 -0
  404. package/get-shit-done/workflows/sketch-wrap-up.md +286 -0
  405. package/get-shit-done/workflows/sketch.md +361 -0
  406. package/get-shit-done/workflows/spec-phase.md +262 -0
  407. package/get-shit-done/workflows/spike-wrap-up.md +307 -0
  408. package/get-shit-done/workflows/spike.md +453 -0
  409. package/get-shit-done/workflows/stats.md +80 -0
  410. package/get-shit-done/workflows/sync-skills.md +182 -0
  411. package/get-shit-done/workflows/thread.md +222 -0
  412. package/get-shit-done/workflows/transition.md +694 -0
  413. package/get-shit-done/workflows/ui-phase.md +328 -0
  414. package/get-shit-done/workflows/ui-review.md +193 -0
  415. package/get-shit-done/workflows/ultraplan-phase.md +199 -0
  416. package/get-shit-done/workflows/undo.md +314 -0
  417. package/get-shit-done/workflows/update.md +443 -0
  418. package/get-shit-done/workflows/validate-phase.md +179 -0
  419. package/get-shit-done/workflows/verify-phase.md +544 -0
  420. package/get-shit-done/workflows/verify-work.md +781 -0
  421. package/hooks/dist/gsd-check-update-worker.js +95 -0
  422. package/hooks/dist/gsd-check-update.js +64 -0
  423. package/hooks/dist/gsd-context-monitor.js +195 -0
  424. package/hooks/dist/gsd-graphify-update.sh +158 -0
  425. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  426. package/hooks/dist/gsd-prompt-guard.js +97 -0
  427. package/hooks/dist/gsd-read-guard.js +101 -0
  428. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  429. package/hooks/dist/gsd-session-state.sh +59 -0
  430. package/hooks/dist/gsd-statusline.js +548 -0
  431. package/hooks/dist/gsd-update-banner.js +134 -0
  432. package/hooks/dist/gsd-validate-commit.sh +57 -0
  433. package/hooks/dist/gsd-workflow-guard.js +166 -0
  434. package/hooks/dist/lib/git-cmd.js +150 -0
  435. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  436. package/hooks/gsd-check-update-worker.js +95 -0
  437. package/hooks/gsd-check-update.js +64 -0
  438. package/hooks/gsd-context-monitor.js +195 -0
  439. package/hooks/gsd-graphify-update.sh +158 -0
  440. package/hooks/gsd-phase-boundary.sh +47 -0
  441. package/hooks/gsd-prompt-guard.js +97 -0
  442. package/hooks/gsd-read-guard.js +101 -0
  443. package/hooks/gsd-read-injection-scanner.js +203 -0
  444. package/hooks/gsd-session-state.sh +59 -0
  445. package/hooks/gsd-statusline.js +548 -0
  446. package/hooks/gsd-update-banner.js +134 -0
  447. package/hooks/gsd-validate-commit.sh +57 -0
  448. package/hooks/gsd-workflow-guard.js +166 -0
  449. package/hooks/lib/git-cmd.js +150 -0
  450. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  451. package/hooks/managed-hooks-registry.cjs +34 -0
  452. package/package.json +102 -0
  453. package/scripts/affected-tests-lib.cjs +541 -0
  454. package/scripts/audit-workflow-script-paths.cjs +73 -0
  455. package/scripts/base64-scan.sh +339 -0
  456. package/scripts/build-hooks.js +236 -0
  457. package/scripts/changeset/README.md +129 -0
  458. package/scripts/changeset/cli.cjs +392 -0
  459. package/scripts/changeset/github-release-notes.cjs +199 -0
  460. package/scripts/changeset/lint.cjs +110 -0
  461. package/scripts/changeset/new.cjs +137 -0
  462. package/scripts/changeset/parse.cjs +114 -0
  463. package/scripts/changeset/render.cjs +34 -0
  464. package/scripts/changeset/serialize.cjs +130 -0
  465. package/scripts/check-alias-drift.cjs +108 -0
  466. package/scripts/check-env.cjs +302 -0
  467. package/scripts/check-npm-integrity.cjs +209 -0
  468. package/scripts/ci-guard-runner.cjs +16 -0
  469. package/scripts/ci-prepare-test-scope.cjs +46 -0
  470. package/scripts/ci-rebase-check.cjs +85 -0
  471. package/scripts/ci-test-scope.cjs +302 -0
  472. package/scripts/command-contract-helpers.cjs +64 -0
  473. package/scripts/diff-touches-shipped-paths.cjs +147 -0
  474. package/scripts/fix-slash-commands.cjs +147 -0
  475. package/scripts/gen-inventory-manifest.cjs +109 -0
  476. package/scripts/generate-package-identity.cjs +104 -0
  477. package/scripts/lint-command-contract.cjs +108 -0
  478. package/scripts/lint-descriptions.cjs +83 -0
  479. package/scripts/lint-docs-required.cjs +222 -0
  480. package/scripts/lint-no-source-grep-extras.cjs +81 -0
  481. package/scripts/lint-no-source-grep.cjs +174 -0
  482. package/scripts/lint-package-identity-drift.cjs +141 -0
  483. package/scripts/lint-pr-check-project-dir.cjs +98 -0
  484. package/scripts/lint-shared-module-handsync.cjs +388 -0
  485. package/scripts/lint-shell-command-projection-drift.cjs +57 -0
  486. package/scripts/lint-skill-deps.cjs +180 -0
  487. package/scripts/lint-test-file-count.allowlist.json +36 -0
  488. package/scripts/lint-test-file-count.cjs +190 -0
  489. package/scripts/pr-template-policy.cjs +268 -0
  490. package/scripts/prompt-injection-scan.sh +203 -0
  491. package/scripts/release-tarball-smoke.cjs +627 -0
  492. package/scripts/run-affected-tests.cjs +6 -0
  493. package/scripts/run-cross-platform-tests.cjs +63 -0
  494. package/scripts/run-tests.cjs +282 -0
  495. package/scripts/secret-scan-lint.sh +231 -0
  496. package/scripts/secret-scan.sh +358 -0
  497. package/scripts/setup-branch-protection.sh +236 -0
  498. package/scripts/shared-module-handsync-allowlist.json +183 -0
  499. package/scripts/strip-prose-atrefs.cjs +106 -0
  500. package/scripts/sync-rulesets.sh +34 -0
  501. package/scripts/sync-runtime-launcher.cjs +402 -0
  502. package/scripts/test-failure-reasons.cjs +34 -0
  503. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,650 @@
1
+ /**
2
+ * Roadmap — Roadmap parsing and update operations
3
+ */
4
+
5
+ const fs = require('fs');
6
+ const path = require('path');
7
+ const { escapeRegex, normalizePhaseName, phaseMarkdownRegexSource, phaseMarkdownRegexSourceExact, output, error, findPhaseInternal, stripShippedMilestones, extractCurrentMilestone, replaceInCurrentMilestone, phaseTokenMatches } = require('./core.cjs');
8
+ const { platformWriteSync } = require('./shell-command-projection.cjs');
9
+ const { planningPaths, withPlanningLock, findContextMdIn } = require('./planning-workspace.cjs');
10
+ const scanPhasePlans = require('./plan-scan.cjs');
11
+
12
+ /**
13
+ * Coerce an arbitrary YAML scalar/object into a string for cross-cutting
14
+ * truth aggregation. Handles:
15
+ * - strings (passthrough)
16
+ * - numbers / booleans (String() coercion — issue #2770: bare YAML ints
17
+ * like `- 3` must be surfaced, not silently skipped)
18
+ * - kv-shaped objects from parseMustHavesBlock continuation kv (issue
19
+ * #2757) — extract the first meaningful string field
20
+ *
21
+ * Returns the empty string when no usable text can be derived; callers should
22
+ * skip empty results.
23
+ */
24
+ function coerceTruthToString(t) {
25
+ if (t === null || t === undefined) return '';
26
+ if (typeof t === 'string') return t;
27
+ if (typeof t === 'number' || typeof t === 'boolean' || typeof t === 'bigint') {
28
+ return String(t);
29
+ }
30
+ if (typeof t === 'object') {
31
+ // Prefer common title-bearing keys produced by parseMustHavesBlock
32
+ for (const k of ['title', 'text', 'name', 'rule', 'path', 'provides']) {
33
+ const v = t[k];
34
+ if (typeof v === 'string' && v.trim()) return v;
35
+ if (typeof v === 'number' || typeof v === 'boolean') return String(v);
36
+ }
37
+ }
38
+ return '';
39
+ }
40
+
41
+ function countPhasePlansAndSummaries(phaseDir) {
42
+ const { planCount, summaryCount } = scanPhasePlans(phaseDir);
43
+ // hasContext and hasResearch are not plan-scan concerns — read the directory
44
+ // once and share the listing for all non-plan metadata that cmdRoadmapAnalyze needs.
45
+ let phaseFiles = [];
46
+ try { phaseFiles = fs.readdirSync(phaseDir); } catch { /* empty */ }
47
+ return {
48
+ planCount,
49
+ summaryCount,
50
+ hasContext: findContextMdIn(phaseFiles) !== null,
51
+ hasResearch: phaseFiles.some(f => f.endsWith('-RESEARCH.md') || f === 'RESEARCH.md'),
52
+ };
53
+ }
54
+
55
+ // `phaseMarkdownRegexSource` moved to core.cjs (#3537) so phase.cjs and
56
+ // core.cjs itself can consume it without circular deps. Imported above.
57
+
58
+ /**
59
+ * Search for a phase header (and its section) within the given content string.
60
+ * Returns a result object if found (either a full match or a malformed_roadmap
61
+ * checklist-only match), or null if the phase is not present at all.
62
+ */
63
+ function searchPhaseInContent(content, escapedPhase, phaseNum) {
64
+ // Match "## Phase X:", "### Phase X:", or "#### Phase X:" with optional name
65
+ const phasePattern = new RegExp(
66
+ `#{2,4}\\s*Phase\\s+${escapedPhase}:\\s*([^\\n]+)`,
67
+ 'i'
68
+ );
69
+ const headerMatch = content.match(phasePattern);
70
+
71
+ if (!headerMatch) {
72
+ // Fallback: check if phase exists in summary list but missing detail section
73
+ const checklistPattern = new RegExp(
74
+ `-\\s*\\[[ x]\\]\\s*\\*\\*Phase\\s+${escapedPhase}:\\s*([^*]+)\\*\\*`,
75
+ 'i'
76
+ );
77
+ const checklistMatch = content.match(checklistPattern);
78
+
79
+ if (checklistMatch) {
80
+ return {
81
+ found: false,
82
+ phase_number: phaseNum,
83
+ phase_name: checklistMatch[1].trim(),
84
+ error: 'malformed_roadmap',
85
+ message: `Phase ${phaseNum} exists in summary list but missing "### Phase ${phaseNum}:" detail section. ROADMAP.md needs both formats.`
86
+ };
87
+ }
88
+
89
+ return null;
90
+ }
91
+
92
+ const phaseName = headerMatch[1].trim();
93
+ const headerIndex = headerMatch.index;
94
+
95
+ // Find the end of this section (next ## or ### phase header, or end of file)
96
+ const restOfContent = content.slice(headerIndex);
97
+ const nextHeaderMatch = restOfContent.match(/\n#{2,4}\s+Phase\s+[\w][\w.-]*/i);
98
+ const sectionEnd = nextHeaderMatch
99
+ ? headerIndex + nextHeaderMatch.index
100
+ : content.length;
101
+
102
+ const section = content.slice(headerIndex, sectionEnd).trim();
103
+
104
+ // Extract goal if present (supports both **Goal:** and **Goal**: formats)
105
+ const goalMatch = section.match(/\*\*Goal(?::\*\*|\*\*:)\s*([^\n]+)/i);
106
+ const goal = goalMatch ? goalMatch[1].trim() : null;
107
+
108
+ // Mode: vertical-MVP slice mode flag. Lowercased + trimmed for canonical
109
+ // comparison; unrecognized values are preserved verbatim for forward-compat.
110
+ const modeMatch = section.match(/\*\*Mode(?::\*\*|\*\*:)\s*([^\n]+)/i);
111
+ const mode = modeMatch ? modeMatch[1].trim().toLowerCase() : null;
112
+
113
+ // Extract success criteria as structured array
114
+ const criteriaMatch = section.match(/\*\*Success Criteria\*\*[^\n]*:\s*\n((?:\s*\d+\.\s*[^\n]+\n?)+)/i);
115
+ const success_criteria = criteriaMatch
116
+ ? criteriaMatch[1].trim().split('\n').map(line => line.replace(/^\s*\d+\.\s*/, '').trim()).filter(Boolean)
117
+ : [];
118
+
119
+ return {
120
+ found: true,
121
+ phase_number: phaseNum,
122
+ phase_name: phaseName,
123
+ goal,
124
+ mode,
125
+ success_criteria,
126
+ section,
127
+ };
128
+ }
129
+
130
+ function cmdRoadmapGetPhase(cwd, phaseNum, raw) {
131
+ const roadmapPath = planningPaths(cwd).roadmap;
132
+
133
+ if (!fs.existsSync(roadmapPath)) {
134
+ output({ found: false, error: 'ROADMAP.md not found' }, raw, '');
135
+ return;
136
+ }
137
+
138
+ try {
139
+ const rawContent = fs.readFileSync(roadmapPath, 'utf-8');
140
+ const milestoneContent = extractCurrentMilestone(rawContent, cwd);
141
+
142
+ // #3599 two-pass: when the caller passes a project-code-prefixed ID like
143
+ // `PROJ-42`, try the exact-prefixed heading first (`### Phase PROJ-42:`).
144
+ // If no match, fall back to the #3537 padding-tolerant numeric form so
145
+ // a `CK-01` query still resolves to `### Phase 1:`. Doing this at the
146
+ // call site (instead of inside phaseMarkdownRegexSource) avoids the
147
+ // alternation-order ambiguity where a bare `### Phase 42:` heading in
148
+ // the same document would intercept the match for a `PROJ-42` query.
149
+ const fullContent = stripShippedMilestones(rawContent);
150
+
151
+ const exactSource = phaseMarkdownRegexSourceExact(phaseNum);
152
+ if (exactSource) {
153
+ const exactMilestone = searchPhaseInContent(milestoneContent, exactSource, phaseNum);
154
+ if (exactMilestone && !exactMilestone.error) {
155
+ output(exactMilestone, raw, exactMilestone.section);
156
+ return;
157
+ }
158
+ const exactFull = searchPhaseInContent(fullContent, exactSource, phaseNum);
159
+ if (exactFull && !exactFull.error) {
160
+ output(exactFull, raw, exactFull.section);
161
+ return;
162
+ }
163
+ }
164
+
165
+ // #3537: padding-tolerant fragment so callers passing `02.7` still match
166
+ // un-padded ROADMAP prose (`### Phase 2.7:`).
167
+ const escapedPhase = phaseMarkdownRegexSource(phaseNum);
168
+
169
+ // Search the current milestone slice first, then fall back to full roadmap.
170
+ // A malformed_roadmap result (checklist-only) from the milestone should not
171
+ // block finding a full header match in the wider roadmap content.
172
+ const milestoneResult = searchPhaseInContent(milestoneContent, escapedPhase, phaseNum);
173
+ const result = (milestoneResult && !milestoneResult.error)
174
+ ? milestoneResult
175
+ : searchPhaseInContent(fullContent, escapedPhase, phaseNum) || milestoneResult;
176
+
177
+ if (!result) {
178
+ output({ found: false, phase_number: phaseNum }, raw, '');
179
+ return;
180
+ }
181
+
182
+ if (result.error) {
183
+ output(result, raw, '');
184
+ return;
185
+ }
186
+
187
+ output(result, raw, result.section);
188
+ } catch (e) {
189
+ error('Failed to read ROADMAP.md: ' + e.message);
190
+ }
191
+ }
192
+
193
+ function cmdRoadmapAnalyze(cwd, raw) {
194
+ const roadmapPath = planningPaths(cwd).roadmap;
195
+
196
+ if (!fs.existsSync(roadmapPath)) {
197
+ output({ error: 'ROADMAP.md not found', milestones: [], phases: [], current_phase: null }, raw);
198
+ return;
199
+ }
200
+
201
+ const rawContent = fs.readFileSync(roadmapPath, 'utf-8');
202
+ const content = extractCurrentMilestone(rawContent, cwd);
203
+ const phasesDir = planningPaths(cwd).phases;
204
+
205
+ // Extract all phase headings: ## Phase N: Name or ### Phase N: Name
206
+ const phasePattern = /#{2,4}\s*Phase\s+(\d+[A-Z]?(?:\.\d+)*)\s*:\s*([^\n]+)/gi;
207
+ const phases = [];
208
+ let match;
209
+
210
+ // Build phase directory lookup once (O(1) readdir instead of O(N) per phase)
211
+ const _phaseDirNames = (() => {
212
+ try {
213
+ return fs.readdirSync(phasesDir, { withFileTypes: true })
214
+ .filter(e => e.isDirectory())
215
+ .map(e => e.name);
216
+ } catch { return []; }
217
+ })();
218
+
219
+ while ((match = phasePattern.exec(content)) !== null) {
220
+ const phaseNum = match[1];
221
+ const phaseName = match[2].replace(/\(INSERTED\)/i, '').trim();
222
+
223
+ // Extract goal from the section
224
+ const sectionStart = match.index;
225
+ const restOfContent = content.slice(sectionStart);
226
+ // #3691: `\d` → `\d[\d.]*` so decimal phase headings (e.g. `### Phase 02.3:`) are
227
+ // recognised as section boundaries.
228
+ const nextHeader = restOfContent.match(/\n#{2,4}\s+Phase\s+\d[\d.]*/i);
229
+ const sectionEnd = nextHeader ? sectionStart + nextHeader.index : content.length;
230
+ const section = content.slice(sectionStart, sectionEnd);
231
+
232
+ const goalMatch = section.match(/\*\*Goal(?::\*\*|\*\*:)\s*([^\n]+)/i);
233
+ const goal = goalMatch ? goalMatch[1].trim() : null;
234
+
235
+ const modeMatch = section.match(/\*\*Mode(?::\*\*|\*\*:)\s*([^\n]+)/i);
236
+ const mode = modeMatch ? modeMatch[1].trim().toLowerCase() : null;
237
+
238
+ const dependsMatch = section.match(/\*\*Depends on(?::\*\*|\*\*:)\s*([^\n]+)/i);
239
+ const depends_on = dependsMatch ? dependsMatch[1].trim() : null;
240
+
241
+ // Check completion on disk
242
+ const normalized = normalizePhaseName(phaseNum);
243
+ let diskStatus = 'no_directory';
244
+ let planCount = 0;
245
+ let summaryCount = 0;
246
+ let hasContext = false;
247
+ let hasResearch = false;
248
+
249
+ try {
250
+ const dirMatch = _phaseDirNames.find(d => phaseTokenMatches(d, normalized));
251
+
252
+ if (dirMatch) {
253
+ const counts = countPhasePlansAndSummaries(path.join(phasesDir, dirMatch));
254
+ planCount = counts.planCount;
255
+ summaryCount = counts.summaryCount;
256
+ hasContext = counts.hasContext;
257
+ hasResearch = counts.hasResearch;
258
+
259
+ if (summaryCount >= planCount && planCount > 0) diskStatus = 'complete';
260
+ else if (summaryCount > 0) diskStatus = 'partial';
261
+ else if (planCount > 0) diskStatus = 'planned';
262
+ else if (hasResearch) diskStatus = 'researched';
263
+ else if (hasContext) diskStatus = 'discussed';
264
+ else diskStatus = 'empty';
265
+ }
266
+ } catch { /* intentionally empty */ }
267
+
268
+ // Check ROADMAP checkbox status.
269
+ // #3537: padding-tolerant fragment — the heading discovered above may use
270
+ // a different padding than the summary-bullet checkbox below it (mixed
271
+ // padding inside one ROADMAP is legal and seen in real projects).
272
+ const checkboxPattern = new RegExp(`-\\s*\\[(x| )\\]\\s*.*Phase\\s+${phaseMarkdownRegexSource(phaseNum)}[:\\s]`, 'i');
273
+ const checkboxMatch = content.match(checkboxPattern);
274
+ const roadmapComplete = checkboxMatch ? checkboxMatch[1] === 'x' : false;
275
+
276
+ // If roadmap marks phase complete, trust that over disk file structure.
277
+ // Phases completed before GSD tracking (or via external tools) may lack
278
+ // the standard PLAN/SUMMARY pairs but are still done.
279
+ if (roadmapComplete && diskStatus !== 'complete') {
280
+ diskStatus = 'complete';
281
+ }
282
+
283
+ phases.push({
284
+ number: phaseNum,
285
+ name: phaseName,
286
+ goal,
287
+ mode,
288
+ depends_on,
289
+ plan_count: planCount,
290
+ summary_count: summaryCount,
291
+ has_context: hasContext,
292
+ has_research: hasResearch,
293
+ disk_status: diskStatus,
294
+ roadmap_complete: roadmapComplete,
295
+ });
296
+ }
297
+
298
+ // Extract milestone info
299
+ const milestones = [];
300
+ const milestonePattern = /##\s*(.*v(\d+(?:\.\d+)+)[^(\n]*)/gi;
301
+ let mMatch;
302
+ while ((mMatch = milestonePattern.exec(content)) !== null) {
303
+ milestones.push({
304
+ heading: mMatch[1].trim(),
305
+ version: 'v' + mMatch[2],
306
+ });
307
+ }
308
+
309
+ // Find current and next phase
310
+ const currentPhase = phases.find(p => p.disk_status === 'planned' || p.disk_status === 'partial') || null;
311
+ const nextPhase = phases.find(p => p.disk_status === 'empty' || p.disk_status === 'no_directory' || p.disk_status === 'discussed' || p.disk_status === 'researched') || null;
312
+
313
+ // Aggregated stats
314
+ const totalPlans = phases.reduce((sum, p) => sum + p.plan_count, 0);
315
+ const totalSummaries = phases.reduce((sum, p) => sum + p.summary_count, 0);
316
+ const completedPhases = phases.filter(p => p.disk_status === 'complete').length;
317
+
318
+ // Detect phases in summary list without detail sections (malformed ROADMAP)
319
+ const checklistPattern = /-\s*\[[ x]\]\s*\*\*Phase\s+(\d+[A-Z]?(?:\.\d+)*)/gi;
320
+ const checklistPhases = new Set();
321
+ let checklistMatch;
322
+ while ((checklistMatch = checklistPattern.exec(content)) !== null) {
323
+ checklistPhases.add(checklistMatch[1]);
324
+ }
325
+ const detailPhases = new Set(phases.map(p => p.number));
326
+ const missingDetails = [...checklistPhases].filter(p => !detailPhases.has(p));
327
+
328
+ const result = {
329
+ milestones,
330
+ phases,
331
+ phase_count: phases.length,
332
+ completed_phases: completedPhases,
333
+ total_plans: totalPlans,
334
+ total_summaries: totalSummaries,
335
+ progress_percent: totalPlans > 0 ? Math.min(100, Math.round((totalSummaries / totalPlans) * 100)) : 0,
336
+ current_phase: currentPhase ? currentPhase.number : null,
337
+ next_phase: nextPhase ? nextPhase.number : null,
338
+ missing_phase_details: missingDetails.length > 0 ? missingDetails : null,
339
+ };
340
+
341
+ output(result, raw);
342
+ }
343
+
344
+ function cmdRoadmapUpdatePlanProgress(cwd, phaseNum, raw) {
345
+ if (!phaseNum) {
346
+ error('phase number required for roadmap update-plan-progress');
347
+ }
348
+
349
+ const roadmapPath = planningPaths(cwd).roadmap;
350
+
351
+ const phaseInfo = findPhaseInternal(cwd, phaseNum);
352
+ if (!phaseInfo) {
353
+ error(`Phase ${phaseNum} not found`);
354
+ }
355
+
356
+ const planCount = phaseInfo.plans.length;
357
+ const summaryCount = phaseInfo.summaries.length;
358
+
359
+ if (planCount === 0) {
360
+ output({ updated: false, reason: 'No plans found', plan_count: 0, summary_count: 0 }, raw, 'no plans');
361
+ return;
362
+ }
363
+
364
+ const isComplete = summaryCount >= planCount;
365
+ const status = isComplete ? 'Complete' : summaryCount > 0 ? 'In Progress' : 'Planned';
366
+ const today = new Date().toISOString().split('T')[0];
367
+
368
+ if (!fs.existsSync(roadmapPath)) {
369
+ output({ updated: false, reason: 'ROADMAP.md not found', plan_count: planCount, summary_count: summaryCount }, raw, 'no roadmap');
370
+ return;
371
+ }
372
+
373
+ // Wrap entire read-modify-write in lock to prevent concurrent corruption
374
+ withPlanningLock(cwd, () => {
375
+ let roadmapContent = fs.readFileSync(roadmapPath, 'utf-8');
376
+ const phasePattern = phaseMarkdownRegexSource(phaseNum);
377
+
378
+ // Progress table row: update Plans/Status/Date columns (handles 4 or 5 column tables)
379
+ const tableRowPattern = new RegExp(
380
+ `^(\\|\\s*${phasePattern}\\.?\\s[^|]*(?:\\|[^\\n]*))$`,
381
+ 'im'
382
+ );
383
+ const dateField = isComplete ? ` ${today} ` : ' ';
384
+ roadmapContent = roadmapContent.replace(tableRowPattern, (fullRow) => {
385
+ const cells = fullRow.split('|').slice(1, -1); // drop leading/trailing empty from split
386
+ if (cells.length === 5) {
387
+ // 5-col: Phase | Milestone | Plans | Status | Completed
388
+ cells[2] = ` ${summaryCount}/${planCount} `;
389
+ cells[3] = ` ${status.padEnd(11)}`;
390
+ cells[4] = dateField;
391
+ } else if (cells.length === 4) {
392
+ // 4-col: Phase | Plans | Status | Completed
393
+ cells[1] = ` ${summaryCount}/${planCount} `;
394
+ cells[2] = ` ${status.padEnd(11)}`;
395
+ cells[3] = dateField;
396
+ }
397
+ return '|' + cells.join('|') + '|';
398
+ });
399
+
400
+ // Update plan count in phase detail section
401
+ const planCountPattern = new RegExp(
402
+ `(#{2,4}\\s*Phase\\s+${phasePattern}(?=[:\\s])[\\s\\S]*?\\*\\*Plans:\\*\\*\\s*)[^\\n]+`,
403
+ 'i'
404
+ );
405
+ const planCountText = isComplete
406
+ ? `${summaryCount}/${planCount} plans complete`
407
+ : `${summaryCount}/${planCount} plans executed`;
408
+ roadmapContent = replaceInCurrentMilestone(roadmapContent, planCountPattern, `$1${planCountText}`);
409
+
410
+ // If complete: check checkbox
411
+ if (isComplete) {
412
+ const checkboxPattern = new RegExp(
413
+ `(-\\s*\\[)[ ](\\]\\s*.*Phase\\s+${phasePattern}[:\\s][^\\n]*)`,
414
+ 'i'
415
+ );
416
+ roadmapContent = replaceInCurrentMilestone(roadmapContent, checkboxPattern, `$1x$2 (completed ${today})`);
417
+ }
418
+
419
+ // Mark completed plan checkboxes (e.g. "- [ ] 50-01-PLAN.md", "- [ ] 50-01:", or "- [ ] **50-01**")
420
+ for (const summaryFile of phaseInfo.summaries) {
421
+ const planId = summaryFile.replace('-SUMMARY.md', '').replace('SUMMARY.md', '');
422
+ if (!planId) continue;
423
+ const planEscaped = escapeRegex(planId);
424
+ const planCheckboxPattern = new RegExp(
425
+ `(-\\s*\\[) (\\]\\s*(?:\\*\\*)?${planEscaped}(?:\\*\\*)?)`,
426
+ 'i'
427
+ );
428
+ roadmapContent = roadmapContent.replace(planCheckboxPattern, '$1x$2');
429
+ }
430
+
431
+ platformWriteSync(roadmapPath, roadmapContent);
432
+ });
433
+ output({
434
+ updated: true,
435
+ phase: phaseNum,
436
+ plan_count: planCount,
437
+ summary_count: summaryCount,
438
+ status,
439
+ complete: isComplete,
440
+ }, raw, `${summaryCount}/${planCount} ${status}`);
441
+ }
442
+
443
+ /**
444
+ * Annotate the ROADMAP.md plan list for a phase with wave dependency notes
445
+ * and a cross-cutting constraints subsection derived from PLAN frontmatter.
446
+ *
447
+ * Wave dependency notes: "Wave 2 — blocked on Wave 1 completion" inserted as
448
+ * bold headers before each wave group in the plan checklist.
449
+ *
450
+ * Cross-cutting constraints: must_haves.truths strings that appear in 2+ plans
451
+ * are surfaced in a "Cross-cutting constraints" subsection below the plan list.
452
+ *
453
+ * The operation is idempotent: if wave headers already exist in the section
454
+ * the function returns without modifying the file.
455
+ */
456
+ function cmdRoadmapAnnotateDependencies(cwd, phaseNum, raw) {
457
+ if (!phaseNum) {
458
+ error('phase number required for roadmap annotate-dependencies');
459
+ }
460
+
461
+ const roadmapPath = planningPaths(cwd).roadmap;
462
+ if (!fs.existsSync(roadmapPath)) {
463
+ output({ updated: false, reason: 'ROADMAP.md not found' }, raw, 'no roadmap');
464
+ return;
465
+ }
466
+
467
+ const phaseInfo = findPhaseInternal(cwd, phaseNum);
468
+ if (!phaseInfo || phaseInfo.plans.length === 0) {
469
+ output({ updated: false, reason: 'no plans found for phase', phase: phaseNum }, raw, 'no plans');
470
+ return;
471
+ }
472
+
473
+ const { extractFrontmatter, parseMustHavesBlock } = require('./frontmatter.cjs');
474
+
475
+ // Read each PLAN.md and extract wave + must_haves.truths
476
+ const planData = [];
477
+ for (const planFile of phaseInfo.plans) {
478
+ const planPath = path.join(path.resolve(cwd, phaseInfo.directory), planFile);
479
+ try {
480
+ const content = fs.readFileSync(planPath, 'utf-8');
481
+ const fm = extractFrontmatter(content);
482
+ const wave = parseInt(fm.wave, 10) || 1;
483
+ const planId = planFile.replace(/-PLAN\.md$/i, '').replace(/PLAN\.md$/i, '');
484
+ const truths = parseMustHavesBlock(content, 'truths') || [];
485
+ planData.push({ planFile, planId, wave, truths });
486
+ } catch { /* skip unreadable plans */ }
487
+ }
488
+
489
+ if (planData.length === 0) {
490
+ output({ updated: false, reason: 'could not read plan frontmatter' }, raw, 'no frontmatter');
491
+ return;
492
+ }
493
+
494
+ // Group plans by wave (sorted)
495
+ const waveGroups = new Map();
496
+ for (const p of planData) {
497
+ if (!waveGroups.has(p.wave)) waveGroups.set(p.wave, []);
498
+ waveGroups.get(p.wave).push(p);
499
+ }
500
+ const waves = [...waveGroups.keys()].sort((a, b) => a - b);
501
+
502
+ // Find cross-cutting truths: appear in 2+ plans (de-duplicated, case-insensitive).
503
+ //
504
+ // Issue #2770: must **coerce, not skip**. A previous guard
505
+ // `if (typeof t !== 'string') continue` silently dropped numeric scalars
506
+ // (YAML ints like `- 3`) and kv-shaped truths (`- title: X`), so the
507
+ // cross-cutting analysis lost real constraints rather than crashing on
508
+ // `t.trim()`. We coerce primitives via `String(t)` and extract a sensible
509
+ // string field from object-shaped items produced by parseMustHavesBlock's
510
+ // continuation-kv path (issue #2757 produces those shapes for nested keys).
511
+ const truthCounts = new Map();
512
+ for (const { truths } of planData) {
513
+ const seen = new Set();
514
+ for (const t of truths) {
515
+ const text = coerceTruthToString(t);
516
+ if (!text) continue;
517
+ const trimmed = text.trim();
518
+ const key = trimmed.toLowerCase();
519
+ if (!key || seen.has(key)) continue;
520
+ seen.add(key);
521
+ if (!truthCounts.has(key)) truthCounts.set(key, { count: 0, text: trimmed });
522
+ truthCounts.get(key).count++;
523
+ }
524
+ }
525
+ const crossCuttingTruths = [...truthCounts.values()]
526
+ .filter(v => v.count >= 2)
527
+ .map(v => v.text);
528
+
529
+ // Patch ROADMAP.md
530
+ let updated = false;
531
+ withPlanningLock(cwd, () => {
532
+ const content = fs.readFileSync(roadmapPath, 'utf-8');
533
+
534
+ // Find the phase section.
535
+ // #3537: padding-tolerant fragment so the caller's resolved padded id
536
+ // matches un-padded ROADMAP headings.
537
+ const phaseEscaped = phaseMarkdownRegexSource(phaseNum);
538
+ const phaseHeaderPattern = new RegExp(`(#{2,4}\\s*Phase\\s+${phaseEscaped}:[^\\n]*)`, 'i');
539
+ const phaseMatch = content.match(phaseHeaderPattern);
540
+ if (!phaseMatch) return;
541
+
542
+ const phaseStart = phaseMatch.index;
543
+ const restAfterHeader = content.slice(phaseStart);
544
+ const nextPhaseOffset = restAfterHeader.slice(1).search(/\n#{2,4}\s+Phase\s+\d/i);
545
+ const phaseEnd = nextPhaseOffset >= 0 ? phaseStart + 1 + nextPhaseOffset : content.length;
546
+ const phaseSection = content.slice(phaseStart, phaseEnd);
547
+
548
+ // Idempotency: skip if annotation markers already present
549
+ if (
550
+ /\*\*Wave\s+\d+/i.test(phaseSection) ||
551
+ /\*\*Cross-cutting constraints:\*\*/i.test(phaseSection)
552
+ ) return;
553
+
554
+ // Find the Plans: section within the phase section.
555
+ // #3691 Bug 1: `Plans:\s*\n` required no text after the colon, missing variants like
556
+ // `Plans: 3 plans across 2 waves\n` or `**Plans:** 3 plans\n` (bold-wrapped).
557
+ // `\*{0,2}Plans\*{0,2}:[^\n]*\n` accepts any text (or none) after the colon
558
+ // and tolerates optional `**` markdown bold wrappers on either side.
559
+ // The checklist group uses `+` (not `*`) so that a bold `**Plans:**` description
560
+ // line with no immediately-following checklist items (e.g. a summary line above a
561
+ // separate bare `Plans:` block) does not consume the match and prevent the actual
562
+ // list from being found.
563
+ // Review fix (F2): `(?:^|\n)` anchors the match to start-of-line so mid-line
564
+ // occurrences like `***Plans:***` embedded in a sentence or `OpenPlans: foo`
565
+ // do not trigger a false match. Groups 1 and 2 retain the same semantics.
566
+ const plansBlockMatch = phaseSection.match(/(?:^|\n)(\*{0,2}Plans\*{0,2}:[^\n]*\n)((?:\s*-\s*\[[ x]\][^\n]*\n?)+)/i);
567
+ if (!plansBlockMatch) return;
568
+
569
+ const plansHeader = plansBlockMatch[1];
570
+ const existingList = plansBlockMatch[2];
571
+ const listLines = existingList.split('\n').filter(l => /^\s*-\s*\[/.test(l));
572
+
573
+ if (listLines.length === 0) return;
574
+
575
+ // #314 perf: build a first-wins Map so per-line lookup is O(1) instead of O(plans).
576
+ // First-wins mirrors .find() semantics: if the same planId appears more than once
577
+ // in planData, the earlier entry wins — identical to what .find() returned before.
578
+ const planById = new Map();
579
+ for (const p of planData) {
580
+ if (!planById.has(p.planId)) planById.set(p.planId, p);
581
+ }
582
+
583
+ // Build wave-annotated plan list
584
+ const linesByWave = new Map();
585
+ for (const line of listLines) {
586
+ // Match plan ID from line: "- [ ] 01-01-PLAN.md — ..." or "- [ ] 01-01: ..."
587
+ // #3691 Bug 3: `[\w-]+?` excluded `.`, so decimal IDs like `02.3-01` were captured
588
+ // as `02` only and never matched planData entries. `[\w.-]+?` preserves the
589
+ // terminating alternation (`-PLAN.md|.md|:|\s—`) as the boundary anchor.
590
+ const idMatch = line.match(/\[\s*[x ]\s*\]\s*([\w.-]+?)(?:-PLAN\.md|\.md|:|\s—)/i);
591
+ const planId = idMatch ? idMatch[1] : null;
592
+ // Review fix (F3): reject malformed IDs that start with `.`, contain consecutive
593
+ // dots, or otherwise violate the `^\w[\w.-]*$` contract. A leading-dot ID
594
+ // (e.g. `.invalid-PLAN.md`) would silently default to wave 1 — defensively
595
+ // skip the line instead so corrupted ROADMAP entries don't corrupt wave layout.
596
+ if (planId && !/^\w[\w.-]*$/.test(planId)) continue;
597
+ const planEntry = planId ? (planById.get(planId) || null) : null;
598
+ const wave = planEntry ? planEntry.wave : 1;
599
+ if (!linesByWave.has(wave)) linesByWave.set(wave, []);
600
+ linesByWave.get(wave).push(line);
601
+ }
602
+
603
+ const annotatedLines = [];
604
+ const sortedWaves = [...linesByWave.keys()].sort((a, b) => a - b);
605
+ for (let i = 0; i < sortedWaves.length; i++) {
606
+ const w = sortedWaves[i];
607
+ const waveLines = linesByWave.get(w);
608
+ if (sortedWaves.length > 1) {
609
+ const dep = i > 0 ? ` *(blocked on Wave ${sortedWaves[i - 1]} completion)*` : '';
610
+ annotatedLines.push(`**Wave ${w}**${dep}`);
611
+ }
612
+ annotatedLines.push(...waveLines);
613
+ if (i < sortedWaves.length - 1) annotatedLines.push('');
614
+ }
615
+
616
+ // Append cross-cutting constraints subsection if any found
617
+ if (crossCuttingTruths.length > 0) {
618
+ annotatedLines.push('');
619
+ annotatedLines.push('**Cross-cutting constraints:**');
620
+ for (const t of crossCuttingTruths) {
621
+ annotatedLines.push(`- ${t}`);
622
+ }
623
+ }
624
+
625
+ const newListBlock = annotatedLines.join('\n') + '\n';
626
+ const newPhaseSection = phaseSection.replace(
627
+ plansBlockMatch[0],
628
+ plansHeader + newListBlock
629
+ );
630
+
631
+ const nextContent = content.slice(0, phaseStart) + newPhaseSection + content.slice(phaseEnd);
632
+ if (nextContent === content) return;
633
+ platformWriteSync(roadmapPath, nextContent);
634
+ updated = true;
635
+ });
636
+
637
+ output({
638
+ updated,
639
+ phase: phaseNum,
640
+ waves: waves.length,
641
+ cross_cutting_constraints: crossCuttingTruths.length,
642
+ }, raw, updated ? `annotated ${waves.length} wave(s), ${crossCuttingTruths.length} constraint(s)` : 'skipped (already annotated or no plan list)');
643
+ }
644
+
645
+ module.exports = {
646
+ cmdRoadmapGetPhase,
647
+ cmdRoadmapAnalyze,
648
+ cmdRoadmapUpdatePlanProgress,
649
+ cmdRoadmapAnnotateDependencies,
650
+ };