@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,600 @@
1
+ "use strict";
2
+ /**
3
+ * Roadmap — Roadmap parsing and update operations
4
+ *
5
+ * ADR-457 build-at-publish: the hand-written bin/lib/roadmap.cjs collapsed
6
+ * to a TypeScript source of truth. Behaviour is preserved byte-for-behaviour
7
+ * from the prior hand-written .cjs; only strict types are added.
8
+ */
9
+ var __importDefault = (this && this.__importDefault) || function (mod) {
10
+ return (mod && mod.__esModule) ? mod : { "default": mod };
11
+ };
12
+ const node_fs_1 = __importDefault(require("node:fs"));
13
+ const node_path_1 = __importDefault(require("node:path"));
14
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
15
+ const core = require("./core.cjs");
16
+ const { escapeRegex, normalizePhaseName, phaseMarkdownRegexSource, phaseMarkdownRegexSourceExact, output, error, findPhaseInternal, stripShippedMilestones, extractCurrentMilestone, replaceInCurrentMilestone, phaseTokenMatches } = core;
17
+ const shell_command_projection_cjs_1 = require("./shell-command-projection.cjs");
18
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
19
+ const planningWorkspace = require("./planning-workspace.cjs");
20
+ const { planningPaths, withPlanningLock, findContextMdIn } = planningWorkspace;
21
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
22
+ const scanPhasePlans = require("./plan-scan.cjs");
23
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
24
+ const frontmatter = require("./frontmatter.cjs");
25
+ const { extractFrontmatter, parseMustHavesBlock } = frontmatter;
26
+ // ─── coerceTruthToString ──────────────────────────────────────────────────────
27
+ /**
28
+ * Coerce an arbitrary YAML scalar/object into a string for cross-cutting
29
+ * truth aggregation. Handles:
30
+ * - strings (passthrough)
31
+ * - numbers / booleans (String() coercion — issue #2770: bare YAML ints
32
+ * like `- 3` must be surfaced, not silently skipped)
33
+ * - kv-shaped objects from parseMustHavesBlock continuation kv (issue
34
+ * #2757) — extract the first meaningful string field
35
+ *
36
+ * Returns the empty string when no usable text can be derived; callers should
37
+ * skip empty results.
38
+ */
39
+ function coerceTruthToString(t) {
40
+ if (t === null || t === undefined)
41
+ return '';
42
+ if (typeof t === 'string')
43
+ return t;
44
+ if (typeof t === 'number' || typeof t === 'boolean' || typeof t === 'bigint') {
45
+ return String(t);
46
+ }
47
+ if (typeof t === 'object') {
48
+ // Prefer common title-bearing keys produced by parseMustHavesBlock
49
+ for (const k of ['title', 'text', 'name', 'rule', 'path', 'provides']) {
50
+ const v = t[k];
51
+ if (typeof v === 'string' && v.trim())
52
+ return v;
53
+ if (typeof v === 'number' || typeof v === 'boolean')
54
+ return String(v);
55
+ }
56
+ }
57
+ return '';
58
+ }
59
+ // ─── countPhasePlansAndSummaries ──────────────────────────────────────────────
60
+ function countPhasePlansAndSummaries(phaseDir) {
61
+ const { planCount, summaryCount } = scanPhasePlans(phaseDir);
62
+ // hasContext and hasResearch are not plan-scan concerns — read the directory
63
+ // once and share the listing for all non-plan metadata that cmdRoadmapAnalyze needs.
64
+ let phaseFiles = [];
65
+ try {
66
+ phaseFiles = node_fs_1.default.readdirSync(phaseDir);
67
+ }
68
+ catch { /* empty */ }
69
+ return {
70
+ planCount,
71
+ summaryCount,
72
+ hasContext: findContextMdIn(phaseFiles) !== null,
73
+ hasResearch: phaseFiles.some(f => f.endsWith('-RESEARCH.md') || f === 'RESEARCH.md'),
74
+ };
75
+ }
76
+ // `phaseMarkdownRegexSource` moved to core.cjs (#3537) so phase.cjs and
77
+ // core.cjs itself can consume it without circular deps. Imported above.
78
+ // ─── searchPhaseInContent ─────────────────────────────────────────────────────
79
+ /**
80
+ * Search for a phase header (and its section) within the given content string.
81
+ * Returns a result object if found (either a full match or a malformed_roadmap
82
+ * checklist-only match), or null if the phase is not present at all.
83
+ */
84
+ function searchPhaseInContent(content, escapedPhase, phaseNum) {
85
+ // Match "## Phase X:", "### Phase X:", or "#### Phase X:" with optional name
86
+ const phasePattern = new RegExp(`#{2,4}\\s*(?:\\[[^\\]]+\\]\\s*)?Phase\\s+${escapedPhase}:\\s*([^\\n]+)`, 'i');
87
+ const headerMatch = content.match(phasePattern);
88
+ if (!headerMatch) {
89
+ // Fallback: check if phase exists in summary list but missing detail section
90
+ const checklistPattern = new RegExp(`-\\s*\\[[ x]\\]\\s*\\*\\*Phase\\s+${escapedPhase}:\\s*([^*]+)\\*\\*`, 'i');
91
+ const checklistMatch = content.match(checklistPattern);
92
+ if (checklistMatch) {
93
+ return {
94
+ found: false,
95
+ phase_number: phaseNum,
96
+ phase_name: checklistMatch[1].trim(),
97
+ error: 'malformed_roadmap',
98
+ message: `Phase ${phaseNum} exists in summary list but missing "### Phase ${phaseNum}:" detail section. ROADMAP.md needs both formats.`
99
+ };
100
+ }
101
+ return null;
102
+ }
103
+ const phaseName = headerMatch[1].trim();
104
+ const headerIndex = headerMatch.index;
105
+ // Find the end of this section (next ## or ### phase header, or end of file).
106
+ // Also matches bracket-prefixed headings like ### [GSD] Phase 2-01:.
107
+ const restOfContent = content.slice(headerIndex);
108
+ const nextHeaderMatch = restOfContent.match(/\n#{2,4}\s+(?:\[[^\]]+\]\s*)?Phase\s+[\w][\w.-]*/i);
109
+ const sectionEnd = nextHeaderMatch
110
+ ? headerIndex + nextHeaderMatch.index
111
+ : content.length;
112
+ const section = content.slice(headerIndex, sectionEnd).trim();
113
+ // Extract goal if present (supports both **Goal:** and **Goal**: formats)
114
+ const goalMatch = section.match(/\*\*Goal(?::\*\*|\*\*:)\s*([^\n]+)/i);
115
+ const goal = goalMatch ? goalMatch[1].trim() : null;
116
+ // Mode: vertical-MVP slice mode flag. Lowercased + trimmed for canonical
117
+ // comparison; unrecognized values are preserved verbatim for forward-compat.
118
+ const modeMatch = section.match(/\*\*Mode(?::\*\*|\*\*:)\s*([^\n]+)/i);
119
+ const mode = modeMatch ? modeMatch[1].trim().toLowerCase() : null;
120
+ // Extract success criteria as structured array
121
+ const criteriaMatch = section.match(/\*\*Success Criteria\*\*[^\n]*:\s*\n((?:\s*\d+\.\s*[^\n]+\n?)+)/i);
122
+ const success_criteria = criteriaMatch
123
+ ? criteriaMatch[1].trim().split('\n').map(line => line.replace(/^\s*\d+\.\s*/, '').trim()).filter(Boolean)
124
+ : [];
125
+ return {
126
+ found: true,
127
+ phase_number: phaseNum,
128
+ phase_name: phaseName,
129
+ goal,
130
+ mode,
131
+ success_criteria,
132
+ section,
133
+ };
134
+ }
135
+ // ─── cmdRoadmapGetPhase ───────────────────────────────────────────────────────
136
+ function cmdRoadmapGetPhase(cwd, phaseNum, raw) {
137
+ const roadmapPath = planningPaths(cwd).roadmap;
138
+ if (!node_fs_1.default.existsSync(roadmapPath)) {
139
+ output({ found: false, error: 'ROADMAP.md not found' }, raw, '');
140
+ return;
141
+ }
142
+ try {
143
+ const rawContent = node_fs_1.default.readFileSync(roadmapPath, 'utf-8');
144
+ const milestoneContent = extractCurrentMilestone(rawContent, cwd);
145
+ // #3599 two-pass: when the caller passes a project-code-prefixed ID like
146
+ // `PROJ-42`, try the exact-prefixed heading first (`### Phase PROJ-42:`).
147
+ // If no match, fall back to the #3537 padding-tolerant numeric form so
148
+ // a `CK-01` query still resolves to `### Phase 1:`. Doing this at the
149
+ // call site (instead of inside phaseMarkdownRegexSource) avoids the
150
+ // alternation-order ambiguity where a bare `### Phase 42:` heading in
151
+ // the same document would intercept the match for a `PROJ-42` query.
152
+ const fullContent = stripShippedMilestones(rawContent);
153
+ const exactSource = phaseMarkdownRegexSourceExact(phaseNum);
154
+ if (exactSource) {
155
+ const exactMilestone = searchPhaseInContent(milestoneContent, exactSource, phaseNum);
156
+ if (exactMilestone && !exactMilestone.error) {
157
+ output(exactMilestone, raw, exactMilestone.section);
158
+ return;
159
+ }
160
+ const exactFull = searchPhaseInContent(fullContent, exactSource, phaseNum);
161
+ if (exactFull && !exactFull.error) {
162
+ output(exactFull, raw, exactFull.section);
163
+ return;
164
+ }
165
+ }
166
+ // #3537: padding-tolerant fragment so callers passing `02.7` still match
167
+ // un-padded ROADMAP prose (`### Phase 2.7:`).
168
+ const escapedPhase = phaseMarkdownRegexSource(phaseNum);
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
+ if (!result) {
177
+ output({ found: false, phase_number: phaseNum }, raw, '');
178
+ return;
179
+ }
180
+ if (result.error) {
181
+ output(result, raw, '');
182
+ return;
183
+ }
184
+ output(result, raw, result.section);
185
+ }
186
+ catch (e) {
187
+ error('Failed to read ROADMAP.md: ' + e.message);
188
+ }
189
+ }
190
+ // ─── cmdRoadmapAnalyze ────────────────────────────────────────────────────────
191
+ function cmdRoadmapAnalyze(cwd, raw) {
192
+ const roadmapPath = planningPaths(cwd).roadmap;
193
+ if (!node_fs_1.default.existsSync(roadmapPath)) {
194
+ output({ error: 'ROADMAP.md not found', milestones: [], phases: [], current_phase: null }, raw, undefined);
195
+ return;
196
+ }
197
+ const rawContent = node_fs_1.default.readFileSync(roadmapPath, 'utf-8');
198
+ const content = extractCurrentMilestone(rawContent, cwd);
199
+ const phasesDir = planningPaths(cwd).phases;
200
+ // Extract all phase headings: ## Phase N: Name or ### Phase N: Name
201
+ const phasePattern = /#{2,4}\s*(?:\[[^\]]+\]\s*)?Phase\s+(\d+[A-Z]?(?:[.-]\d+)*)\s*:\s*([^\n]+)/gi;
202
+ const phases = [];
203
+ let match;
204
+ // Build phase directory lookup once (O(1) readdir instead of O(N) per phase)
205
+ const _phaseDirNames = (() => {
206
+ try {
207
+ return node_fs_1.default.readdirSync(phasesDir, { withFileTypes: true })
208
+ .filter(e => e.isDirectory())
209
+ .map(e => e.name);
210
+ }
211
+ catch {
212
+ return [];
213
+ }
214
+ })();
215
+ while ((match = phasePattern.exec(content)) !== null) {
216
+ const phaseNum = match[1];
217
+ const phaseName = match[2].replace(/\(INSERTED\)/i, '').trim();
218
+ // Extract goal from the section
219
+ const sectionStart = match.index;
220
+ const restOfContent = content.slice(sectionStart);
221
+ // #3691: `\d` → `\d[\d.]*` so decimal phase headings (e.g. `### Phase 02.3:`) are
222
+ // recognised as section boundaries.
223
+ const nextHeader = restOfContent.match(/\n#{2,4}\s+(?:\[[^\]]+\]\s*)?Phase\s+\d[\d.-]*/i);
224
+ const sectionEnd = nextHeader ? sectionStart + nextHeader.index : content.length;
225
+ const section = content.slice(sectionStart, sectionEnd);
226
+ const goalMatch = section.match(/\*\*Goal(?::\*\*|\*\*:)\s*([^\n]+)/i);
227
+ const goal = goalMatch ? goalMatch[1].trim() : null;
228
+ const modeMatch = section.match(/\*\*Mode(?::\*\*|\*\*:)\s*([^\n]+)/i);
229
+ const mode = modeMatch ? modeMatch[1].trim().toLowerCase() : null;
230
+ const dependsMatch = section.match(/\*\*Depends on(?::\*\*|\*\*:)\s*([^\n]+)/i);
231
+ const depends_on = dependsMatch ? dependsMatch[1].trim() : null;
232
+ // Check completion on disk
233
+ const normalized = normalizePhaseName(phaseNum);
234
+ let diskStatus = 'no_directory';
235
+ let planCount = 0;
236
+ let summaryCount = 0;
237
+ let hasContext = false;
238
+ let hasResearch = false;
239
+ try {
240
+ const dirMatch = _phaseDirNames.find(d => phaseTokenMatches(d, normalized));
241
+ if (dirMatch) {
242
+ const counts = countPhasePlansAndSummaries(node_path_1.default.join(phasesDir, dirMatch));
243
+ planCount = counts.planCount;
244
+ summaryCount = counts.summaryCount;
245
+ hasContext = counts.hasContext;
246
+ hasResearch = counts.hasResearch;
247
+ if (summaryCount >= planCount && planCount > 0)
248
+ diskStatus = 'complete';
249
+ else if (summaryCount > 0)
250
+ diskStatus = 'partial';
251
+ else if (planCount > 0)
252
+ diskStatus = 'planned';
253
+ else if (hasResearch)
254
+ diskStatus = 'researched';
255
+ else if (hasContext)
256
+ diskStatus = 'discussed';
257
+ else
258
+ diskStatus = 'empty';
259
+ }
260
+ }
261
+ catch { /* intentionally empty */ }
262
+ // Check ROADMAP checkbox status.
263
+ // #3537: padding-tolerant fragment — the heading discovered above may use
264
+ // a different padding than the summary-bullet checkbox below it (mixed
265
+ // padding inside one ROADMAP is legal and seen in real projects).
266
+ const checkboxPattern = new RegExp(`-\\s*\\[(x| )\\]\\s*.*Phase\\s+${phaseMarkdownRegexSource(phaseNum)}[:\\s]`, 'i');
267
+ const checkboxMatch = content.match(checkboxPattern);
268
+ const roadmapComplete = checkboxMatch ? checkboxMatch[1] === 'x' : false;
269
+ // If roadmap marks phase complete, trust that over disk file structure.
270
+ // Phases completed before GSD tracking (or via external tools) may lack
271
+ // the standard PLAN/SUMMARY pairs but are still done.
272
+ if (roadmapComplete && diskStatus !== 'complete') {
273
+ diskStatus = 'complete';
274
+ }
275
+ phases.push({
276
+ number: phaseNum,
277
+ name: phaseName,
278
+ goal,
279
+ mode,
280
+ depends_on,
281
+ plan_count: planCount,
282
+ summary_count: summaryCount,
283
+ has_context: hasContext,
284
+ has_research: hasResearch,
285
+ disk_status: diskStatus,
286
+ roadmap_complete: roadmapComplete,
287
+ });
288
+ }
289
+ // Extract milestone info
290
+ const milestones = [];
291
+ const milestonePattern = /##\s*(.*v(\d+(?:\.\d+)+)[^(\n]*)/gi;
292
+ let mMatch;
293
+ while ((mMatch = milestonePattern.exec(content)) !== null) {
294
+ milestones.push({
295
+ heading: mMatch[1].trim(),
296
+ version: 'v' + mMatch[2],
297
+ });
298
+ }
299
+ // Find current and next phase
300
+ const currentPhase = phases.find(p => p.disk_status === 'planned' || p.disk_status === 'partial') || null;
301
+ const nextPhase = phases.find(p => p.disk_status === 'empty' || p.disk_status === 'no_directory' || p.disk_status === 'discussed' || p.disk_status === 'researched') || null;
302
+ // Aggregated stats
303
+ const totalPlans = phases.reduce((sum, p) => sum + p.plan_count, 0);
304
+ const totalSummaries = phases.reduce((sum, p) => sum + p.summary_count, 0);
305
+ const completedPhases = phases.filter(p => p.disk_status === 'complete').length;
306
+ // Detect phases in summary list without detail sections (malformed ROADMAP)
307
+ const checklistPattern = /-\s*\[[ x]\]\s*\*\*Phase\s+(\d+[A-Z]?(?:\.\d+)*)/gi;
308
+ const checklistPhases = new Set();
309
+ let checklistMatch;
310
+ while ((checklistMatch = checklistPattern.exec(content)) !== null) {
311
+ checklistPhases.add(checklistMatch[1]);
312
+ }
313
+ const detailPhases = new Set(phases.map(p => p.number));
314
+ const missingDetails = [...checklistPhases].filter(p => !detailPhases.has(p));
315
+ const result = {
316
+ milestones,
317
+ phases,
318
+ phase_count: phases.length,
319
+ completed_phases: completedPhases,
320
+ total_plans: totalPlans,
321
+ total_summaries: totalSummaries,
322
+ progress_percent: totalPlans > 0 ? Math.min(100, Math.round((totalSummaries / totalPlans) * 100)) : 0,
323
+ current_phase: currentPhase ? currentPhase.number : null,
324
+ next_phase: nextPhase ? nextPhase.number : null,
325
+ missing_phase_details: missingDetails.length > 0 ? missingDetails : null,
326
+ };
327
+ output(result, raw, undefined);
328
+ }
329
+ // ─── cmdRoadmapUpdatePlanProgress ─────────────────────────────────────────────
330
+ function cmdRoadmapUpdatePlanProgress(cwd, phaseNum, raw) {
331
+ if (!phaseNum) {
332
+ error('phase number required for roadmap update-plan-progress');
333
+ }
334
+ const roadmapPath = planningPaths(cwd).roadmap;
335
+ const phaseInfo = findPhaseInternal(cwd, phaseNum);
336
+ if (!phaseInfo) {
337
+ error(`Phase ${phaseNum} not found`);
338
+ }
339
+ const planCount = phaseInfo.plans.length;
340
+ const summaryCount = phaseInfo.summaries.length;
341
+ if (planCount === 0) {
342
+ output({ updated: false, reason: 'No plans found', plan_count: 0, summary_count: 0 }, raw, 'no plans');
343
+ return;
344
+ }
345
+ const isComplete = summaryCount >= planCount;
346
+ const status = isComplete ? 'Complete' : summaryCount > 0 ? 'In Progress' : 'Planned';
347
+ const today = new Date().toISOString().split('T')[0];
348
+ if (!node_fs_1.default.existsSync(roadmapPath)) {
349
+ output({ updated: false, reason: 'ROADMAP.md not found', plan_count: planCount, summary_count: summaryCount }, raw, 'no roadmap');
350
+ return;
351
+ }
352
+ // Wrap entire read-modify-write in lock to prevent concurrent corruption
353
+ withPlanningLock(cwd, () => {
354
+ let roadmapContent = node_fs_1.default.readFileSync(roadmapPath, 'utf-8');
355
+ const phasePattern = phaseMarkdownRegexSource(phaseNum);
356
+ // Progress table row: update Plans/Status/Date columns (handles 4 or 5 column tables)
357
+ const tableRowPattern = new RegExp(`^(\\|\\s*${phasePattern}\\.?\\s[^|]*(?:\\|[^\\n]*))$`, 'im');
358
+ const dateField = isComplete ? ` ${today} ` : ' ';
359
+ roadmapContent = roadmapContent.replace(tableRowPattern, (fullRow) => {
360
+ const cells = fullRow.split('|').slice(1, -1); // drop leading/trailing empty from split
361
+ if (cells.length === 5) {
362
+ // 5-col: Phase | Milestone | Plans | Status | Completed
363
+ cells[2] = ` ${summaryCount}/${planCount} `;
364
+ cells[3] = ` ${status.padEnd(11)}`;
365
+ cells[4] = dateField;
366
+ }
367
+ else if (cells.length === 4) {
368
+ // 4-col: Phase | Plans | Status | Completed
369
+ cells[1] = ` ${summaryCount}/${planCount} `;
370
+ cells[2] = ` ${status.padEnd(11)}`;
371
+ cells[3] = dateField;
372
+ }
373
+ return '|' + cells.join('|') + '|';
374
+ });
375
+ // Update plan count in phase detail section
376
+ const planCountPattern = new RegExp(`(#{2,4}\\s*Phase\\s+${phasePattern}(?=[:\\s])[\\s\\S]*?\\*\\*Plans:\\*\\*\\s*)[^\\n]+`, 'i');
377
+ const planCountText = isComplete
378
+ ? `${summaryCount}/${planCount} plans complete`
379
+ : `${summaryCount}/${planCount} plans executed`;
380
+ roadmapContent = replaceInCurrentMilestone(roadmapContent, planCountPattern, `$1${planCountText}`);
381
+ // If complete: check checkbox
382
+ if (isComplete) {
383
+ const checkboxPattern = new RegExp(`(-\\s*\\[)[ ](\\]\\s*.*Phase\\s+${phasePattern}[:\\s][^\\n]*)`, 'i');
384
+ roadmapContent = replaceInCurrentMilestone(roadmapContent, checkboxPattern, `$1x$2 (completed ${today})`);
385
+ }
386
+ // Mark completed plan checkboxes (e.g. "- [ ] 50-01-PLAN.md", "- [ ] 50-01:", or "- [ ] **50-01**")
387
+ for (const summaryFile of phaseInfo.summaries) {
388
+ const planId = summaryFile.replace('-SUMMARY.md', '').replace('SUMMARY.md', '');
389
+ if (!planId)
390
+ continue;
391
+ const planEscaped = escapeRegex(planId);
392
+ const planCheckboxPattern = new RegExp(`(-\\s*\\[) (\\]\\s*(?:\\*\\*)?${planEscaped}(?:\\*\\*)?)`, 'i');
393
+ roadmapContent = roadmapContent.replace(planCheckboxPattern, '$1x$2');
394
+ }
395
+ (0, shell_command_projection_cjs_1.platformWriteSync)(roadmapPath, roadmapContent);
396
+ });
397
+ output({
398
+ updated: true,
399
+ phase: phaseNum,
400
+ plan_count: planCount,
401
+ summary_count: summaryCount,
402
+ status,
403
+ complete: isComplete,
404
+ }, raw, `${summaryCount}/${planCount} ${status}`);
405
+ }
406
+ // ─── cmdRoadmapAnnotateDependencies ───────────────────────────────────────────
407
+ /**
408
+ * Annotate the ROADMAP.md plan list for a phase with wave dependency notes
409
+ * and a cross-cutting constraints subsection derived from PLAN frontmatter.
410
+ *
411
+ * Wave dependency notes: "Wave 2 — blocked on Wave 1 completion" inserted as
412
+ * bold headers before each wave group in the plan checklist.
413
+ *
414
+ * Cross-cutting constraints: must_haves.truths strings that appear in 2+ plans
415
+ * are surfaced in a "Cross-cutting constraints" subsection below the plan list.
416
+ *
417
+ * The operation is idempotent: if wave headers already exist in the section
418
+ * the function returns without modifying the file.
419
+ */
420
+ function cmdRoadmapAnnotateDependencies(cwd, phaseNum, raw) {
421
+ if (!phaseNum) {
422
+ error('phase number required for roadmap annotate-dependencies');
423
+ }
424
+ const roadmapPath = planningPaths(cwd).roadmap;
425
+ if (!node_fs_1.default.existsSync(roadmapPath)) {
426
+ output({ updated: false, reason: 'ROADMAP.md not found' }, raw, 'no roadmap');
427
+ return;
428
+ }
429
+ const phaseInfo = findPhaseInternal(cwd, phaseNum);
430
+ if (!phaseInfo || phaseInfo.plans.length === 0) {
431
+ output({ updated: false, reason: 'no plans found for phase', phase: phaseNum }, raw, 'no plans');
432
+ return;
433
+ }
434
+ // Read each PLAN.md and extract wave + must_haves.truths
435
+ const planData = [];
436
+ for (const planFile of phaseInfo.plans) {
437
+ const planPath = node_path_1.default.join(node_path_1.default.resolve(cwd, phaseInfo.directory), planFile);
438
+ try {
439
+ const content = node_fs_1.default.readFileSync(planPath, 'utf-8');
440
+ const fm = extractFrontmatter(content);
441
+ const wave = parseInt(fm.wave, 10) || 1;
442
+ const planId = planFile.replace(/-PLAN\.md$/i, '').replace(/PLAN\.md$/i, '');
443
+ const truths = parseMustHavesBlock(content, 'truths') || [];
444
+ planData.push({ planFile, planId, wave, truths });
445
+ }
446
+ catch { /* skip unreadable plans */ }
447
+ }
448
+ if (planData.length === 0) {
449
+ output({ updated: false, reason: 'could not read plan frontmatter' }, raw, 'no frontmatter');
450
+ return;
451
+ }
452
+ // Group plans by wave (sorted)
453
+ const waveGroups = new Map();
454
+ for (const p of planData) {
455
+ if (!waveGroups.has(p.wave))
456
+ waveGroups.set(p.wave, []);
457
+ waveGroups.get(p.wave).push(p);
458
+ }
459
+ const waves = [...waveGroups.keys()].sort((a, b) => a - b);
460
+ // Find cross-cutting truths: appear in 2+ plans (de-duplicated, case-insensitive).
461
+ //
462
+ // Issue #2770: must **coerce, not skip**. A previous guard
463
+ // `if (typeof t !== 'string') continue` silently dropped numeric scalars
464
+ // (YAML ints like `- 3`) and kv-shaped truths (`- title: X`), so the
465
+ // cross-cutting analysis lost real constraints rather than crashing on
466
+ // `t.trim()`. We coerce primitives via `String(t)` and extract a sensible
467
+ // string field from object-shaped items produced by parseMustHavesBlock's
468
+ // continuation-kv path (issue #2757 produces those shapes for nested keys).
469
+ const truthCounts = new Map();
470
+ for (const { truths } of planData) {
471
+ const seen = new Set();
472
+ for (const t of truths) {
473
+ const text = coerceTruthToString(t);
474
+ if (!text)
475
+ continue;
476
+ const trimmed = text.trim();
477
+ const key = trimmed.toLowerCase();
478
+ if (!key || seen.has(key))
479
+ continue;
480
+ seen.add(key);
481
+ if (!truthCounts.has(key))
482
+ truthCounts.set(key, { count: 0, text: trimmed });
483
+ truthCounts.get(key).count++;
484
+ }
485
+ }
486
+ const crossCuttingTruths = [...truthCounts.values()]
487
+ .filter(v => v.count >= 2)
488
+ .map(v => v.text);
489
+ // Patch ROADMAP.md
490
+ let updated = false;
491
+ withPlanningLock(cwd, () => {
492
+ const content = node_fs_1.default.readFileSync(roadmapPath, 'utf-8');
493
+ // Find the phase section.
494
+ // #3537: padding-tolerant fragment so the caller's resolved padded id
495
+ // matches un-padded ROADMAP headings.
496
+ const phaseEscaped = phaseMarkdownRegexSource(phaseNum);
497
+ const phaseHeaderPattern = new RegExp(`(#{2,4}\\s*Phase\\s+${phaseEscaped}:[^\\n]*)`, 'i');
498
+ const phaseMatch = content.match(phaseHeaderPattern);
499
+ if (!phaseMatch)
500
+ return;
501
+ const phaseStart = phaseMatch.index;
502
+ const restAfterHeader = content.slice(phaseStart);
503
+ const nextPhaseOffset = restAfterHeader.slice(1).search(/\n#{2,4}\s+Phase\s+\d/i);
504
+ const phaseEnd = nextPhaseOffset >= 0 ? phaseStart + 1 + nextPhaseOffset : content.length;
505
+ const phaseSection = content.slice(phaseStart, phaseEnd);
506
+ // Idempotency: skip if annotation markers already present
507
+ if (/\*\*Wave\s+\d+/i.test(phaseSection) ||
508
+ /\*\*Cross-cutting constraints:\*\*/i.test(phaseSection))
509
+ return;
510
+ // Find the Plans: section within the phase section.
511
+ // #3691 Bug 1: `Plans:\s*\n` required no text after the colon, missing variants like
512
+ // `Plans: 3 plans across 2 waves\n` or `**Plans:** 3 plans\n` (bold-wrapped).
513
+ // `\*{0,2}Plans\*{0,2}:[^\n]*\n` accepts any text (or none) after the colon
514
+ // and tolerates optional `**` markdown bold wrappers on either side.
515
+ // The checklist group uses `+` (not `*`) so that a bold `**Plans:**` description
516
+ // line with no immediately-following checklist items (e.g. a summary line above a
517
+ // separate bare `Plans:` block) does not consume the match and prevent the actual
518
+ // list from being found.
519
+ // Review fix (F2): `(?:^|\n)` anchors the match to start-of-line so mid-line
520
+ // occurrences like `***Plans:***` embedded in a sentence or `OpenPlans: foo`
521
+ // do not trigger a false match. Groups 1 and 2 retain the same semantics.
522
+ const plansBlockMatch = phaseSection.match(/(?:^|\n)(\*{0,2}Plans\*{0,2}:[^\n]*\n)((?:\s*-\s*\[[ x]\][^\n]*\n?)+)/i);
523
+ if (!plansBlockMatch)
524
+ return;
525
+ const plansHeader = plansBlockMatch[1];
526
+ const existingList = plansBlockMatch[2];
527
+ const listLines = existingList.split('\n').filter(l => /^\s*-\s*\[/.test(l));
528
+ if (listLines.length === 0)
529
+ return;
530
+ // #314 perf: build a first-wins Map so per-line lookup is O(1) instead of O(plans).
531
+ // First-wins mirrors .find() semantics: if the same planId appears more than once
532
+ // in planData, the earlier entry wins — identical to what .find() returned before.
533
+ const planById = new Map();
534
+ for (const p of planData) {
535
+ if (!planById.has(p.planId))
536
+ planById.set(p.planId, p);
537
+ }
538
+ // Build wave-annotated plan list
539
+ const linesByWave = new Map();
540
+ for (const line of listLines) {
541
+ // Match plan ID from line: "- [ ] 01-01-PLAN.md — ..." or "- [ ] 01-01: ..."
542
+ // #3691 Bug 3: `[\w-]+?` excluded `.`, so decimal IDs like `02.3-01` were captured
543
+ // as `02` only and never matched planData entries. `[\w.-]+?` preserves the
544
+ // terminating alternation (`-PLAN.md|.md|:|\s—`) as the boundary anchor.
545
+ const idMatch = line.match(/\[\s*[x ]\s*\]\s*([\w.-]+?)(?:-PLAN\.md|\.md|:|\s—)/i);
546
+ const planId = idMatch ? idMatch[1] : null;
547
+ // Review fix (F3): reject malformed IDs that start with `.`, contain consecutive
548
+ // dots, or otherwise violate the `^\w[\w.-]*$` contract. A leading-dot ID
549
+ // (e.g. `.invalid-PLAN.md`) would silently default to wave 1 — defensively
550
+ // skip the line instead so corrupted ROADMAP entries don't corrupt wave layout.
551
+ if (planId && !/^\w[\w.-]*$/.test(planId))
552
+ continue;
553
+ const planEntry = planId ? (planById.get(planId) || null) : null;
554
+ const wave = planEntry ? planEntry.wave : 1;
555
+ if (!linesByWave.has(wave))
556
+ linesByWave.set(wave, []);
557
+ linesByWave.get(wave).push(line);
558
+ }
559
+ const annotatedLines = [];
560
+ const sortedWaves = [...linesByWave.keys()].sort((a, b) => a - b);
561
+ for (let i = 0; i < sortedWaves.length; i++) {
562
+ const w = sortedWaves[i];
563
+ const waveLines = linesByWave.get(w);
564
+ if (sortedWaves.length > 1) {
565
+ const dep = i > 0 ? ` *(blocked on Wave ${sortedWaves[i - 1]} completion)*` : '';
566
+ annotatedLines.push(`**Wave ${w}**${dep}`);
567
+ }
568
+ annotatedLines.push(...waveLines);
569
+ if (i < sortedWaves.length - 1)
570
+ annotatedLines.push('');
571
+ }
572
+ // Append cross-cutting constraints subsection if any found
573
+ if (crossCuttingTruths.length > 0) {
574
+ annotatedLines.push('');
575
+ annotatedLines.push('**Cross-cutting constraints:**');
576
+ for (const t of crossCuttingTruths) {
577
+ annotatedLines.push(`- ${t}`);
578
+ }
579
+ }
580
+ const newListBlock = annotatedLines.join('\n') + '\n';
581
+ const newPhaseSection = phaseSection.replace(plansBlockMatch[0], plansHeader + newListBlock);
582
+ const nextContent = content.slice(0, phaseStart) + newPhaseSection + content.slice(phaseEnd);
583
+ if (nextContent === content)
584
+ return;
585
+ (0, shell_command_projection_cjs_1.platformWriteSync)(roadmapPath, nextContent);
586
+ updated = true;
587
+ });
588
+ output({
589
+ updated,
590
+ phase: phaseNum,
591
+ waves: waves.length,
592
+ cross_cutting_constraints: crossCuttingTruths.length,
593
+ }, raw, updated ? `annotated ${waves.length} wave(s), ${crossCuttingTruths.length} constraint(s)` : 'skipped (already annotated or no plan list)');
594
+ }
595
+ module.exports = {
596
+ cmdRoadmapGetPhase,
597
+ cmdRoadmapAnalyze,
598
+ cmdRoadmapUpdatePlanProgress,
599
+ cmdRoadmapAnnotateDependencies,
600
+ };