@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,566 @@
1
+ #!/usr/bin/env node
2
+ // gsd-hook-version: {{GSD_VERSION}}
3
+ // Claude Code Statusline - GSD Edition
4
+ // Shows: model | current task (or GSD state) | directory | context usage
5
+
6
+ const fs = require('fs');
7
+ const path = require('path');
8
+ const os = require('os');
9
+ const { isSemverNewer } = require('../gsd-core/bin/lib/semver-compare.cjs');
10
+ const { PACKAGE_NAME, updateCacheFileName } = require('../gsd-core/bin/lib/package-identity.cjs');
11
+
12
+ // --- Config + last-command readers ------------------------------------------
13
+
14
+ /**
15
+ * Walk up from dir looking for .planning/config.json and return its parsed contents.
16
+ * Returns {} if not found or unreadable.
17
+ */
18
+ function readGsdConfig(dir) {
19
+ const home = os.homedir();
20
+ let current = dir;
21
+ for (let i = 0; i < 10; i++) {
22
+ const candidate = path.join(current, '.planning', 'config.json');
23
+ if (fs.existsSync(candidate)) {
24
+ try {
25
+ return JSON.parse(fs.readFileSync(candidate, 'utf8')) || {};
26
+ } catch (e) {
27
+ return {};
28
+ }
29
+ }
30
+ const parent = path.dirname(current);
31
+ if (parent === current || current === home) break;
32
+ current = parent;
33
+ }
34
+ return {};
35
+ }
36
+
37
+ /**
38
+ * Lookup a dotted key path (e.g. 'statusline.show_last_command') in a config
39
+ * object that may use either nested or flat keys.
40
+ */
41
+ function getConfigValue(cfg, keyPath) {
42
+ if (!cfg || typeof cfg !== 'object') return undefined;
43
+ if (keyPath in cfg) return cfg[keyPath];
44
+ const parts = keyPath.split('.');
45
+ let cur = cfg;
46
+ for (const p of parts) {
47
+ if (cur == null || typeof cur !== 'object' || !(p in cur)) return undefined;
48
+ cur = cur[p];
49
+ }
50
+ return cur;
51
+ }
52
+
53
+ /**
54
+ * Extract the most recently invoked slash command from a Claude Code JSONL
55
+ * transcript file. Returns the command name (no leading slash) or null.
56
+ *
57
+ * Claude Code embeds slash invocations in user messages as
58
+ * <command-name>/foo</command-name>
59
+ * We scan lines from the end of the file, stopping at the first match.
60
+ */
61
+ function readLastSlashCommand(transcriptPath) {
62
+ if (!transcriptPath || typeof transcriptPath !== 'string') return null;
63
+ let content;
64
+ try {
65
+ if (!fs.existsSync(transcriptPath)) return null;
66
+ // Read only the tail — typical transcripts grow large. 256 KiB comfortably
67
+ // covers dozens of recent turns while staying cheap per render.
68
+ const stat = fs.statSync(transcriptPath);
69
+ const MAX = 256 * 1024;
70
+ const start = Math.max(0, stat.size - MAX);
71
+ const fd = fs.openSync(transcriptPath, 'r');
72
+ try {
73
+ const buf = Buffer.alloc(stat.size - start);
74
+ fs.readSync(fd, buf, 0, buf.length, start);
75
+ content = buf.toString('utf8');
76
+ } finally {
77
+ fs.closeSync(fd);
78
+ }
79
+ } catch (e) {
80
+ return null;
81
+ }
82
+ // Find the LAST occurrence — scan right-to-left via lastIndexOf on the tag.
83
+ const tagClose = '</command-name>';
84
+ const idx = content.lastIndexOf(tagClose);
85
+ if (idx < 0) return null;
86
+ const openTag = '<command-name>';
87
+ const openIdx = content.lastIndexOf(openTag, idx);
88
+ if (openIdx < 0) return null;
89
+ let name = content.slice(openIdx + openTag.length, idx).trim();
90
+ // Strip a leading slash if present, and any trailing arguments-on-same-line noise.
91
+ if (name.startsWith('/')) name = name.slice(1);
92
+ // Command names in Claude Code transcripts are plain identifiers like "gsd-plan-phase"
93
+ // or namespaced like "plugin:skill". Reject anything with whitespace/newlines/control chars.
94
+ if (!name || /[\s\\"<>]/.test(name) || name.length > 80) return null;
95
+ return name;
96
+ }
97
+
98
+ // --- GSD state reader -------------------------------------------------------
99
+
100
+ /**
101
+ * Walk up from dir looking for .planning/STATE.md.
102
+ * Returns parsed state object or null.
103
+ */
104
+ function readGsdState(dir) {
105
+ const home = os.homedir();
106
+ let current = dir;
107
+ for (let i = 0; i < 10; i++) {
108
+ const candidate = path.join(current, '.planning', 'STATE.md');
109
+ if (fs.existsSync(candidate)) {
110
+ try {
111
+ return parseStateMd(fs.readFileSync(candidate, 'utf8'));
112
+ } catch (e) {
113
+ return null;
114
+ }
115
+ }
116
+ const parent = path.dirname(current);
117
+ if (parent === current || current === home) break;
118
+ current = parent;
119
+ }
120
+ return null;
121
+ }
122
+
123
+ /**
124
+ * Parse STATE.md frontmatter + Phase line from body.
125
+ *
126
+ * Returns:
127
+ * { status, milestone, milestoneName, phaseNum, phaseTotal, phaseName,
128
+ * activePhase, nextAction, nextPhases, completedPhases, totalPhases, percent }
129
+ *
130
+ * Phase-lifecycle fields (issue #2833):
131
+ * - activePhase : phase number ("4.5") when an orchestrator is mid-flight, null otherwise
132
+ * - nextAction : recommended next command ("execute-phase") when idle, null otherwise
133
+ * - nextPhases : array of phase numbers (["4.5"]) for nextAction, null otherwise
134
+ * - completedPhases / totalPhases / percent : milestone progress dimension
135
+ *
136
+ * All new fields default to undefined when absent — formatGsdState() degrades
137
+ * gracefully so existing STATE.md files (without these fields) keep working.
138
+ */
139
+ function parseStateMd(content) {
140
+ const state = {};
141
+
142
+ // YAML frontmatter between --- markers (anchored at file start)
143
+ const fmMatch = content.match(/^---\n([\s\S]*?)\n---/);
144
+ if (fmMatch) {
145
+ const fm = fmMatch[1];
146
+ // Top-level scalar key: value
147
+ for (const line of fm.split('\n')) {
148
+ const m = line.match(/^(\w+):\s*(.+)/);
149
+ if (!m) continue;
150
+ const [, key, val] = m;
151
+ const v = val.trim().replace(/^["']|["']$/g, '');
152
+ // status / milestone-level fields (existing — preserved exactly)
153
+ if (key === 'status') state.status = v === 'null' ? null : v;
154
+ if (key === 'milestone') state.milestone = v === 'null' ? null : v;
155
+ if (key === 'milestone_name') state.milestoneName = v === 'null' ? null : v;
156
+ // Phase-lifecycle fields (new in issue #2833)
157
+ // active_phase: phase number when an orchestrator is in-flight, null when idle
158
+ if (key === 'active_phase') state.activePhase = (v === 'null' || v === '') ? null : v;
159
+ // next_action: recommended command when idle (discuss-phase / plan-phase / execute-phase / verify-phase)
160
+ if (key === 'next_action') state.nextAction = (v === 'null' || v === '') ? null : v;
161
+ }
162
+ // next_phases supports both flow array and block-list YAML forms.
163
+ const npFlowMatch = fm.match(/^next_phases:\s*\[([^\]]*)\]/m);
164
+ if (npFlowMatch) {
165
+ const items = npFlowMatch[1].split(',').map(s => s.trim().replace(/^["']|["']$/g, '')).filter(Boolean);
166
+ state.nextPhases = items.length > 0 ? items : null;
167
+ } else {
168
+ const npBlockMatch = fm.match(/^next_phases:\s*\n((?:[ \t]*-[ \t]*[^\n]+\n?)*)/m);
169
+ if (npBlockMatch) {
170
+ const items = npBlockMatch[1]
171
+ .split('\n')
172
+ .map(line => line.match(/^[ \t]*-[ \t]*(.+)$/))
173
+ .filter(Boolean)
174
+ .map(m => m[1].trim().replace(/^["']|["']$/g, ''))
175
+ .filter(Boolean);
176
+ state.nextPhases = items.length > 0 ? items : null;
177
+ }
178
+ }
179
+ // progress nested block: completed_phases / total_phases / percent (2-space indent)
180
+ const progMatch = fm.match(/^progress:\s*\n((?:[ \t]+\w+:.+\n?)+)/m);
181
+ if (progMatch) {
182
+ const cp = progMatch[1].match(/^[ \t]+completed_phases:\s*(\d+)/m);
183
+ const tp = progMatch[1].match(/^[ \t]+total_phases:\s*(\d+)/m);
184
+ const pc = progMatch[1].match(/^[ \t]+percent:\s*(\d+)/m);
185
+ if (cp) state.completedPhases = cp[1];
186
+ if (tp) state.totalPhases = tp[1];
187
+ if (pc) state.percent = pc[1];
188
+ }
189
+ }
190
+
191
+ // Phase: N of M (name) or Phase: none active (...)
192
+ const phaseMatch = content.match(/^Phase:\s*(\d+)\s+of\s+(\d+)(?:\s+\(([^)]+)\))?/m);
193
+ if (phaseMatch) {
194
+ state.phaseNum = phaseMatch[1];
195
+ state.phaseTotal = phaseMatch[2];
196
+ state.phaseName = phaseMatch[3] || null;
197
+ }
198
+
199
+ // Fallback: parse Status: from body when frontmatter is absent
200
+ if (!state.status) {
201
+ const bodyStatus = content.match(/^Status:\s*(.+)/m);
202
+ if (bodyStatus) {
203
+ const raw = bodyStatus[1].trim().toLowerCase();
204
+ if (raw.includes('ready to plan') || raw.includes('planning')) state.status = 'planning';
205
+ else if (raw.includes('execut')) state.status = 'executing';
206
+ else if (raw.includes('complet') || raw.includes('archived')) state.status = 'complete';
207
+ }
208
+ }
209
+
210
+ return state;
211
+ }
212
+
213
+ /**
214
+ * Render a 10-segment milestone progress bar (matches the context meter style).
215
+ *
216
+ * @param {number|string|null|undefined} percent — 0-100; missing/NaN returns ''
217
+ * @returns {string} '[█████░░░░░] 50%' or '' (so callers can `[bar].filter(Boolean)`)
218
+ */
219
+ function renderProgressBar(percent) {
220
+ if (percent == null || isNaN(percent)) return '';
221
+ const pct = Math.max(0, Math.min(100, parseInt(percent, 10)));
222
+ const filled = Math.floor(pct / 10);
223
+ const bar = '█'.repeat(filled) + '░'.repeat(10 - filled);
224
+ return `[${bar}] ${pct}%`;
225
+ }
226
+
227
+ /**
228
+ * Format GSD state into display string.
229
+ *
230
+ * Backward-compatible default (no new fields populated):
231
+ * "v1.9 Code Quality · executing · fix-graphiti-deployment (1/5)"
232
+ *
233
+ * Phase-lifecycle scenes (issue #2833 — activate when STATE.md frontmatter
234
+ * carries the new fields; otherwise rendering falls through to the default):
235
+ *
236
+ * active_phase set → "v2.0 [██░] X% · Phase 4.5 executing"
237
+ * active_phase null + next_action set → "v2.0 [██░] X% · next execute-phase 4.5"
238
+ * percent=100 (milestone done) → "v2.0 [██████████] 100% · milestone complete"
239
+ * none of the above → existing "<status> · <phase>" path
240
+ *
241
+ * Progress bar is opt-in: appended to the milestone segment only when
242
+ * progress.percent is present in frontmatter; absent → empty string.
243
+ */
244
+ function formatGsdState(s) {
245
+ const parts = [];
246
+
247
+ // Milestone segment: version + name + (opt-in) progress bar
248
+ if (s.milestone || s.milestoneName) {
249
+ const ver = s.milestone || '';
250
+ const name = (s.milestoneName && s.milestoneName !== 'milestone') ? s.milestoneName : '';
251
+ const bar = renderProgressBar(s.percent);
252
+ const pieces = [ver, name, bar].filter(Boolean);
253
+ if (pieces.length > 0) parts.push(pieces.join(' '));
254
+ }
255
+
256
+ // Phase-lifecycle scenes (issue #2833) — first match wins; falls through to
257
+ // the original "<status> · <phase>" path when none of the new fields apply.
258
+ const phasesStr = (s.nextPhases && s.nextPhases.length > 0) ? s.nextPhases.join('/') : null;
259
+
260
+ if (s.activePhase) {
261
+ // Scene 1: an orchestrator is mid-flight on this phase.
262
+ // stage = whichever lifecycle status was written by the orchestrator
263
+ // (discussing / planning / executing / verifying)
264
+ const stage = s.status || '';
265
+ parts.push(stage ? `Phase ${s.activePhase} ${stage}` : `Phase ${s.activePhase}`);
266
+ } else if (s.nextAction && phasesStr) {
267
+ // Scene 2: idle + a recommended next command is visible to the user.
268
+ // Surfaces "what to run next" without the user opening STATE.md.
269
+ parts.push(`next ${s.nextAction} ${phasesStr}`);
270
+ } else if (Number(s.percent) === 100 || (s.completedPhases && s.totalPhases && s.completedPhases === s.totalPhases)) {
271
+ // Scene 3: milestone complete (every phase done).
272
+ parts.push('milestone complete');
273
+ } else {
274
+ // Backward-compatible default — preserved EXACTLY for STATE.md files that
275
+ // don't carry the new lifecycle fields. Identical output to v1.38.x and
276
+ // earlier so no existing project's status-line changes shape.
277
+ if (s.status) parts.push(s.status);
278
+ if (s.phaseNum && s.phaseTotal) {
279
+ const phase = s.phaseName
280
+ ? `${s.phaseName} (${s.phaseNum}/${s.phaseTotal})`
281
+ : `ph ${s.phaseNum}/${s.phaseTotal}`;
282
+ parts.push(phase);
283
+ }
284
+ }
285
+
286
+ return parts.join(' · ');
287
+ }
288
+
289
+ // --- stdin ------------------------------------------------------------------
290
+
291
+ function runStatusline() {
292
+ let input = '';
293
+ // Timeout guard: if stdin doesn't close within 3s (e.g. pipe issues on
294
+ // Windows/Git Bash), exit silently instead of hanging. See #775.
295
+ const stdinTimeout = setTimeout(() => process.exit(0), 3000);
296
+ process.stdin.setEncoding('utf8');
297
+ process.stdin.on('data', chunk => input += chunk);
298
+ process.stdin.on('end', () => {
299
+ clearTimeout(stdinTimeout);
300
+ try {
301
+ const data = JSON.parse(input);
302
+ const model = data.model?.display_name || 'Claude';
303
+ const dir = data.workspace?.current_dir || process.cwd();
304
+ const session = data.session_id || '';
305
+ const remaining = data.context_window?.remaining_percentage;
306
+
307
+ // Context window display (shows USED percentage scaled to usable context)
308
+ // Claude Code reserves a buffer for autocompact. By default this is ~16.5%
309
+ // of the total window, but users can override it via CLAUDE_CODE_AUTO_COMPACT_WINDOW
310
+ // (a token count). When the env var is set, compute the buffer % dynamically so
311
+ // the meter correctly reflects early-compaction configurations (#2219).
312
+ const totalCtx = data.context_window?.total_tokens || 1_000_000;
313
+ const acw = parseInt(process.env.CLAUDE_CODE_AUTO_COMPACT_WINDOW || '0', 10);
314
+ const AUTO_COMPACT_BUFFER_PCT = acw > 0
315
+ ? Math.min(100, (acw / totalCtx) * 100)
316
+ : 16.5;
317
+ let ctx = '';
318
+ if (remaining != null) {
319
+ // Normalize: subtract buffer from remaining, scale to usable range
320
+ const usableRemaining = Math.max(0, ((remaining - AUTO_COMPACT_BUFFER_PCT) / (100 - AUTO_COMPACT_BUFFER_PCT)) * 100);
321
+ const used = Math.max(0, Math.min(100, Math.round(100 - usableRemaining)));
322
+
323
+ // Write context metrics to bridge file for the context-monitor PostToolUse hook.
324
+ // The monitor reads this file to inject agent-facing warnings when context is low.
325
+ // Reject session IDs with path separators or traversal sequences to prevent
326
+ // a malicious session_id from writing files outside the temp directory.
327
+ const sessionSafe = session && !/[/\\]|\.\./.test(session);
328
+ if (sessionSafe) {
329
+ try {
330
+ const bridgePath = path.join(os.tmpdir(), `claude-ctx-${session}.json`);
331
+ // used_pct written to the bridge must match CC's native /context reporting:
332
+ // raw used = 100 - remaining_percentage (no buffer normalization applied).
333
+ // The normalized `used` value is correct for the statusline progress bar but
334
+ // inflates the context monitor warning messages by ~13 points (#2451).
335
+ const rawUsedPct = Math.round(100 - remaining);
336
+ const bridgeData = JSON.stringify({
337
+ session_id: session,
338
+ remaining_percentage: remaining,
339
+ used_pct: rawUsedPct,
340
+ timestamp: Math.floor(Date.now() / 1000)
341
+ });
342
+ fs.writeFileSync(bridgePath, bridgeData);
343
+ } catch (e) {
344
+ // Silent fail -- bridge is best-effort, don't break statusline
345
+ }
346
+ }
347
+
348
+ // Build progress bar (10 segments)
349
+ const filled = Math.floor(used / 10);
350
+ const bar = '█'.repeat(filled) + '░'.repeat(10 - filled);
351
+
352
+ // Color based on usable context thresholds
353
+ if (used < 50) {
354
+ ctx = ` \x1b[32m${bar} ${used}%\x1b[0m`;
355
+ } else if (used < 65) {
356
+ ctx = ` \x1b[33m${bar} ${used}%\x1b[0m`;
357
+ } else if (used < 80) {
358
+ ctx = ` \x1b[38;5;208m${bar} ${used}%\x1b[0m`;
359
+ } else {
360
+ ctx = ` \x1b[5;31m💀 ${bar} ${used}%\x1b[0m`;
361
+ }
362
+ }
363
+
364
+ // Current task from todos
365
+ let task = '';
366
+ const homeDir = os.homedir();
367
+ // Respect CLAUDE_CONFIG_DIR for custom config directory setups (#870)
368
+ const claudeDir = process.env.CLAUDE_CONFIG_DIR || path.join(homeDir, '.claude');
369
+ const todosDir = path.join(claudeDir, 'todos');
370
+ if (session && fs.existsSync(todosDir)) {
371
+ try {
372
+ // Single-pass max-by-mtime scan: only the newest matching todos file
373
+ // is needed, so the O(n log n) sort and the intermediate array from the
374
+ // prior `.filter().map(statSync).sort()` chain are unnecessary. Identical
375
+ // I/O (one statSync per match) and identical result. (#305)
376
+ let latest = null;
377
+ for (const entry of fs.readdirSync(todosDir)) {
378
+ if (!entry.startsWith(session) || !entry.includes('-agent-') || !entry.endsWith('.json')) continue;
379
+ const mtime = fs.statSync(path.join(todosDir, entry)).mtime;
380
+ if (!latest || mtime > latest.mtime) latest = { name: entry, mtime };
381
+ }
382
+
383
+ if (latest) {
384
+ try {
385
+ const todos = JSON.parse(fs.readFileSync(path.join(todosDir, latest.name), 'utf8'));
386
+ const inProgress = todos.find(t => t.status === 'in_progress');
387
+ if (inProgress) task = inProgress.activeForm || '';
388
+ } catch (e) {}
389
+ }
390
+ } catch (e) {
391
+ // Silently fail on file system errors - don't break statusline
392
+ }
393
+ }
394
+
395
+ // GSD state (milestone · status · phase) — shown when no todo task
396
+ const gsdStateStr = task ? '' : formatGsdState(readGsdState(dir) || {});
397
+
398
+ // GSD update available?
399
+ // Read only the per-package shared cache file (#607). The legacy
400
+ // runtime-specific fallback has been removed — the per-package filename
401
+ // carries lineage and avoids multi-runtime resolution mismatches (#1421).
402
+ let gsdUpdate = '';
403
+ const cacheFile = path.join(homeDir, '.cache', 'gsd', updateCacheFileName);
404
+ if (fs.existsSync(cacheFile)) {
405
+ try {
406
+ const cache = JSON.parse(fs.readFileSync(cacheFile, 'utf8'));
407
+ const { showUpdate, staleWarning } = evaluateUpdateCache(cache);
408
+ if (showUpdate) {
409
+ gsdUpdate = '\x1b[33m⬆ /gsd:update\x1b[0m │ ';
410
+ }
411
+ if (staleWarning === 'dev') {
412
+ gsdUpdate += '\x1b[33m⚠ dev install — re-run installer to sync hooks\x1b[0m │ ';
413
+ } else if (staleWarning === 'stale') {
414
+ gsdUpdate += '\x1b[31m⚠ stale hooks — run /gsd:update\x1b[0m │ ';
415
+ }
416
+ } catch (e) {}
417
+ }
418
+
419
+ // Last-slash-command suffix and context_position config (#2538, #2937).
420
+ // Reads the active session transcript for the most recent <command-name> tag.
421
+ // Failure here must never break the statusline — wrap the entire lookup.
422
+ let lastCmdSuffix = '';
423
+ let position = 'end';
424
+ try {
425
+ const cfg = readGsdConfig(dir);
426
+ if (getConfigValue(cfg, 'statusline.show_last_command') === true) {
427
+ const transcriptPath = data.transcript_path;
428
+ const lastCmd = readLastSlashCommand(transcriptPath);
429
+ if (lastCmd) {
430
+ lastCmdSuffix = ` │ \x1b[2mlast: /${lastCmd}\x1b[0m`;
431
+ }
432
+ }
433
+ const cfgPos = getConfigValue(cfg, 'statusline.context_position');
434
+ if (cfgPos != null) position = cfgPos;
435
+ } catch (e) {
436
+ // Never break the statusline on config/transcript errors
437
+ }
438
+
439
+ // Output
440
+ const dirname = path.basename(dir);
441
+ const middle = task
442
+ ? `\x1b[1m${task}\x1b[0m`
443
+ : gsdStateStr
444
+ ? `\x1b[2m${gsdStateStr}\x1b[0m`
445
+ : null;
446
+
447
+ process.stdout.write(composeStatusline({ gsdUpdate, model, ctx, middle, dirname, lastCmdSuffix, position }));
448
+ } catch (e) {
449
+ // Silent fail - don't break statusline on parse errors
450
+ }
451
+ });
452
+ }
453
+
454
+ // --- Layout composer --------------------------------------------------------
455
+
456
+ /**
457
+ * Compose the statusline string from pre-built segments.
458
+ *
459
+ * @param {object} opts
460
+ * @param {string} [opts.gsdUpdate=''] - leading update/stale-hooks warning (already formatted)
461
+ * @param {string} opts.model - model display name (plain text; dim styling applied here)
462
+ * @param {string} [opts.ctx=''] - context-window meter segment (empty string = absent)
463
+ * @param {string|null} [opts.middle=null] - middle segment (todo task or GSD state), null = absent
464
+ * @param {string} opts.dirname - project directory basename (dim styling applied here)
465
+ * @param {string} [opts.lastCmdSuffix=''] - last-command suffix, e.g. ' │ last: /foo'
466
+ * @param {'end'|'front'} [opts.position='end']
467
+ * - 'end' (default): ctx appended after dirname — preserved byte-for-byte
468
+ * - 'front': ctx immediately after model name so the meter stays visible in narrow terminals
469
+ *
470
+ * Invalid position values are silently coerced to 'end' — config-set schema rejects
471
+ * invalid values upfront; runtime fallback defends against stale/corrupt configs
472
+ * without breaking the statusline.
473
+ */
474
+ function composeStatusline({
475
+ gsdUpdate = '',
476
+ model,
477
+ ctx = '',
478
+ middle = null,
479
+ dirname,
480
+ lastCmdSuffix = '',
481
+ position = 'end',
482
+ } = {}) {
483
+ const modelSeg = `\x1b[2m${model}\x1b[0m`;
484
+ const dirSeg = `\x1b[2m${dirname}\x1b[0m`;
485
+ // Coerce invalid values to 'end' (belt-and-suspenders; see JSDoc above)
486
+ const pos = position === 'front' ? 'front' : 'end';
487
+
488
+ if (pos === 'front') {
489
+ if (middle) return `${gsdUpdate}${modelSeg}${ctx} │ ${middle} │ ${dirSeg}${lastCmdSuffix}`;
490
+ return `${gsdUpdate}${modelSeg}${ctx} │ ${dirSeg}${lastCmdSuffix}`;
491
+ }
492
+ // 'end' — preserved byte-for-byte relative to original inline templates
493
+ if (middle) return `${gsdUpdate}${modelSeg} │ ${middle} │ ${dirSeg}${ctx}${lastCmdSuffix}`;
494
+ return `${gsdUpdate}${modelSeg} │ ${dirSeg}${ctx}${lastCmdSuffix}`;
495
+ }
496
+
497
+ function isInstalledAheadOfLatest(installed, latest) {
498
+ return isSemverNewer(installed, latest);
499
+ }
500
+
501
+ /**
502
+ * Pure function: evaluate an update-check cache object and return display flags.
503
+ * Applies lineage guard — if package_name is absent or foreign, treats cache as absent.
504
+ *
505
+ * @param {object|null} cache Parsed cache object, or null.
506
+ * @returns {{ showUpdate: boolean, staleWarning: 'none'|'dev'|'stale' }}
507
+ */
508
+ function evaluateUpdateCache(cache) {
509
+ const none = { showUpdate: false, staleWarning: 'none' };
510
+ if (!cache) return none;
511
+ // Lineage guard: package_name must be present and match this package.
512
+ if (!cache.package_name || cache.package_name !== PACKAGE_NAME) return none;
513
+ const showUpdate = Boolean(cache.update_available);
514
+ let staleWarning = 'none';
515
+ if (cache.stale_hooks && cache.stale_hooks.length > 0) {
516
+ const isDevInstall = (
517
+ cache.installed &&
518
+ cache.latest &&
519
+ cache.latest !== 'unknown' &&
520
+ isInstalledAheadOfLatest(cache.installed, cache.latest)
521
+ );
522
+ staleWarning = isDevInstall ? 'dev' : 'stale';
523
+ }
524
+ return { showUpdate, staleWarning };
525
+ }
526
+
527
+ // Export helpers for unit tests. Harmless when run as a script.
528
+ module.exports = {
529
+ readGsdState, parseStateMd, formatGsdState,
530
+ readGsdConfig, getConfigValue, readLastSlashCommand,
531
+ composeStatusline,
532
+ isInstalledAheadOfLatest,
533
+ evaluateUpdateCache,
534
+ };
535
+
536
+ /**
537
+ * Render the statusline from an already-parsed hook input object. Exported for
538
+ * testing without feeding stdin. Returns the rendered string.
539
+ */
540
+ function renderStatusline(data) {
541
+ const model = data.model?.display_name || 'Claude';
542
+ const dir = data.workspace?.current_dir || process.cwd();
543
+ const dirname = path.basename(dir);
544
+
545
+ let lastCmdSuffix = '';
546
+ let position = 'end';
547
+ try {
548
+ const cfg = readGsdConfig(dir);
549
+ if (getConfigValue(cfg, 'statusline.show_last_command') === true) {
550
+ const lastCmd = readLastSlashCommand(data.transcript_path);
551
+ if (lastCmd) {
552
+ lastCmdSuffix = ` │ \x1b[2mlast: /${lastCmd}\x1b[0m`;
553
+ }
554
+ }
555
+ const cfgPos = getConfigValue(cfg, 'statusline.context_position');
556
+ if (cfgPos != null) position = cfgPos;
557
+ } catch (e) { /* swallow */ }
558
+
559
+ const gsdStateStr = formatGsdState(readGsdState(dir) || {});
560
+ const middle = gsdStateStr ? `\x1b[2m${gsdStateStr}\x1b[0m` : null;
561
+ return composeStatusline({ model, ctx: '', middle, dirname, lastCmdSuffix, position });
562
+ }
563
+
564
+ module.exports.renderStatusline = renderStatusline;
565
+
566
+ if (require.main === module) runStatusline();