@opengsd/gsd-core 1.2.0-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (503) hide show
  1. package/LICENSE +21 -0
  2. package/README.ja-JP.md +870 -0
  3. package/README.ko-KR.md +861 -0
  4. package/README.md +301 -0
  5. package/README.pt-BR.md +492 -0
  6. package/README.zh-CN.md +842 -0
  7. package/agents/gsd-advisor-researcher.md +127 -0
  8. package/agents/gsd-ai-researcher.md +133 -0
  9. package/agents/gsd-assumptions-analyzer.md +105 -0
  10. package/agents/gsd-code-fixer.md +668 -0
  11. package/agents/gsd-code-reviewer.md +387 -0
  12. package/agents/gsd-codebase-mapper.md +853 -0
  13. package/agents/gsd-debug-session-manager.md +314 -0
  14. package/agents/gsd-debugger.md +1452 -0
  15. package/agents/gsd-doc-classifier.md +168 -0
  16. package/agents/gsd-doc-synthesizer.md +204 -0
  17. package/agents/gsd-doc-verifier.md +217 -0
  18. package/agents/gsd-doc-writer.md +615 -0
  19. package/agents/gsd-domain-researcher.md +153 -0
  20. package/agents/gsd-eval-auditor.md +191 -0
  21. package/agents/gsd-eval-planner.md +154 -0
  22. package/agents/gsd-executor.md +772 -0
  23. package/agents/gsd-framework-selector.md +160 -0
  24. package/agents/gsd-integration-checker.md +470 -0
  25. package/agents/gsd-intel-updater.md +342 -0
  26. package/agents/gsd-nyquist-auditor.md +203 -0
  27. package/agents/gsd-pattern-mapper.md +335 -0
  28. package/agents/gsd-phase-researcher.md +928 -0
  29. package/agents/gsd-plan-checker.md +978 -0
  30. package/agents/gsd-planner.md +1218 -0
  31. package/agents/gsd-project-researcher.md +677 -0
  32. package/agents/gsd-research-synthesizer.md +255 -0
  33. package/agents/gsd-roadmapper.md +688 -0
  34. package/agents/gsd-security-auditor.md +155 -0
  35. package/agents/gsd-ui-auditor.md +495 -0
  36. package/agents/gsd-ui-checker.md +309 -0
  37. package/agents/gsd-ui-researcher.md +380 -0
  38. package/agents/gsd-user-profiler.md +171 -0
  39. package/agents/gsd-verifier.md +917 -0
  40. package/bin/install.js +10936 -0
  41. package/bin/lib/ui-safety-gate.cjs +107 -0
  42. package/commands/gsd/add-tests.md +42 -0
  43. package/commands/gsd/ai-integration-phase.md +37 -0
  44. package/commands/gsd/audit-fix.md +34 -0
  45. package/commands/gsd/audit-milestone.md +37 -0
  46. package/commands/gsd/audit-uat.md +24 -0
  47. package/commands/gsd/autonomous.md +46 -0
  48. package/commands/gsd/capture.md +62 -0
  49. package/commands/gsd/cleanup.md +24 -0
  50. package/commands/gsd/code-review.md +59 -0
  51. package/commands/gsd/complete-milestone.md +143 -0
  52. package/commands/gsd/config.md +56 -0
  53. package/commands/gsd/debug.md +52 -0
  54. package/commands/gsd/discuss-phase.md +76 -0
  55. package/commands/gsd/docs-update.md +49 -0
  56. package/commands/gsd/eval-review.md +33 -0
  57. package/commands/gsd/execute-phase.md +64 -0
  58. package/commands/gsd/explore.md +27 -0
  59. package/commands/gsd/extract-learnings.md +23 -0
  60. package/commands/gsd/fast.md +31 -0
  61. package/commands/gsd/forensics.md +57 -0
  62. package/commands/gsd/graphify.md +199 -0
  63. package/commands/gsd/health.md +31 -0
  64. package/commands/gsd/help.md +28 -0
  65. package/commands/gsd/import.md +41 -0
  66. package/commands/gsd/inbox.md +39 -0
  67. package/commands/gsd/ingest-docs.md +42 -0
  68. package/commands/gsd/manager.md +45 -0
  69. package/commands/gsd/map-codebase.md +83 -0
  70. package/commands/gsd/milestone-summary.md +51 -0
  71. package/commands/gsd/mvp-phase.md +45 -0
  72. package/commands/gsd/new-milestone.md +45 -0
  73. package/commands/gsd/new-project.md +47 -0
  74. package/commands/gsd/ns-context.md +23 -0
  75. package/commands/gsd/ns-ideate.md +24 -0
  76. package/commands/gsd/ns-manage.md +29 -0
  77. package/commands/gsd/ns-project.md +22 -0
  78. package/commands/gsd/ns-review.md +26 -0
  79. package/commands/gsd/ns-workflow.md +28 -0
  80. package/commands/gsd/pause-work.md +43 -0
  81. package/commands/gsd/phase.md +56 -0
  82. package/commands/gsd/plan-phase.md +62 -0
  83. package/commands/gsd/plan-review-convergence.md +59 -0
  84. package/commands/gsd/pr-branch.md +26 -0
  85. package/commands/gsd/profile-user.md +46 -0
  86. package/commands/gsd/progress.md +47 -0
  87. package/commands/gsd/quick.md +174 -0
  88. package/commands/gsd/resume-work.md +30 -0
  89. package/commands/gsd/review-backlog.md +63 -0
  90. package/commands/gsd/review.md +41 -0
  91. package/commands/gsd/secure-phase.md +36 -0
  92. package/commands/gsd/settings.md +29 -0
  93. package/commands/gsd/ship.md +24 -0
  94. package/commands/gsd/sketch.md +60 -0
  95. package/commands/gsd/spec-phase.md +63 -0
  96. package/commands/gsd/spike.md +57 -0
  97. package/commands/gsd/stats.md +19 -0
  98. package/commands/gsd/surface.md +155 -0
  99. package/commands/gsd/thread.md +24 -0
  100. package/commands/gsd/ui-phase.md +35 -0
  101. package/commands/gsd/ui-review.md +33 -0
  102. package/commands/gsd/ultraplan-phase.md +34 -0
  103. package/commands/gsd/undo.md +35 -0
  104. package/commands/gsd/update.md +48 -0
  105. package/commands/gsd/validate-phase.md +36 -0
  106. package/commands/gsd/verify-work.md +39 -0
  107. package/commands/gsd/workspace.md +52 -0
  108. package/commands/gsd/workstreams.md +70 -0
  109. package/get-shit-done/bin/check-latest-version.cjs +106 -0
  110. package/get-shit-done/bin/gsd-tools.cjs +1676 -0
  111. package/get-shit-done/bin/lib/active-workstream-store.cjs +302 -0
  112. package/get-shit-done/bin/lib/adr-parser.cjs +394 -0
  113. package/get-shit-done/bin/lib/agent-command-router.cjs +65 -0
  114. package/get-shit-done/bin/lib/artifacts.cjs +53 -0
  115. package/get-shit-done/bin/lib/audit.cjs +755 -0
  116. package/get-shit-done/bin/lib/check-command-router.cjs +333 -0
  117. package/get-shit-done/bin/lib/cjs-command-router-adapter.cjs +118 -0
  118. package/get-shit-done/bin/lib/clock.cjs +96 -0
  119. package/get-shit-done/bin/lib/clusters.cjs +135 -0
  120. package/get-shit-done/bin/lib/code-review-flags.cjs +74 -0
  121. package/get-shit-done/bin/lib/command-aliases.cjs +815 -0
  122. package/get-shit-done/bin/lib/command-arg-projection.cjs +62 -0
  123. package/get-shit-done/bin/lib/command-routing-hub.cjs +388 -0
  124. package/get-shit-done/bin/lib/commands.cjs +1188 -0
  125. package/get-shit-done/bin/lib/config-schema.cjs +31 -0
  126. package/get-shit-done/bin/lib/config.cjs +728 -0
  127. package/get-shit-done/bin/lib/configuration.cjs +248 -0
  128. package/get-shit-done/bin/lib/context-utilization.cjs +47 -0
  129. package/get-shit-done/bin/lib/core.cjs +2121 -0
  130. package/get-shit-done/bin/lib/decisions.cjs +116 -0
  131. package/get-shit-done/bin/lib/docs.cjs +270 -0
  132. package/get-shit-done/bin/lib/drift.cjs +388 -0
  133. package/get-shit-done/bin/lib/fallow-runner.cjs +109 -0
  134. package/get-shit-done/bin/lib/frontmatter.cjs +389 -0
  135. package/get-shit-done/bin/lib/gap-checker.cjs +205 -0
  136. package/get-shit-done/bin/lib/graphify.cjs +592 -0
  137. package/get-shit-done/bin/lib/gsd2-import.cjs +514 -0
  138. package/get-shit-done/bin/lib/init-command-router.cjs +58 -0
  139. package/get-shit-done/bin/lib/init.cjs +2112 -0
  140. package/get-shit-done/bin/lib/install-profiles.cjs +603 -0
  141. package/get-shit-done/bin/lib/installer-migration-authoring.cjs +117 -0
  142. package/get-shit-done/bin/lib/installer-migration-report.cjs +354 -0
  143. package/get-shit-done/bin/lib/installer-migrations/000-first-time-baseline.cjs +220 -0
  144. package/get-shit-done/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +41 -0
  145. package/get-shit-done/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +80 -0
  146. package/get-shit-done/bin/lib/installer-migrations.cjs +778 -0
  147. package/get-shit-done/bin/lib/intel.cjs +708 -0
  148. package/get-shit-done/bin/lib/learnings.cjs +421 -0
  149. package/get-shit-done/bin/lib/milestone.cjs +314 -0
  150. package/get-shit-done/bin/lib/model-catalog.cjs +212 -0
  151. package/get-shit-done/bin/lib/model-profiles.cjs +31 -0
  152. package/get-shit-done/bin/lib/observability/event.cjs +82 -0
  153. package/get-shit-done/bin/lib/observability/logger.cjs +174 -0
  154. package/get-shit-done/bin/lib/observability/redaction.cjs +50 -0
  155. package/get-shit-done/bin/lib/package-identity.cjs +31 -0
  156. package/get-shit-done/bin/lib/phase-command-router.cjs +191 -0
  157. package/get-shit-done/bin/lib/phase-lifecycle.cjs +80 -0
  158. package/get-shit-done/bin/lib/phase.cjs +1607 -0
  159. package/get-shit-done/bin/lib/phases-command-router.cjs +39 -0
  160. package/get-shit-done/bin/lib/plan-scan.cjs +97 -0
  161. package/get-shit-done/bin/lib/planning-workspace.cjs +238 -0
  162. package/get-shit-done/bin/lib/profile-output.cjs +1141 -0
  163. package/get-shit-done/bin/lib/profile-pipeline.cjs +539 -0
  164. package/get-shit-done/bin/lib/project-root.cjs +112 -0
  165. package/get-shit-done/bin/lib/prompt-budget.cjs +399 -0
  166. package/get-shit-done/bin/lib/review-reviewer-selection.cjs +125 -0
  167. package/get-shit-done/bin/lib/roadmap-command-router.cjs +28 -0
  168. package/get-shit-done/bin/lib/roadmap.cjs +650 -0
  169. package/get-shit-done/bin/lib/runtime-artifact-layout.cjs +301 -0
  170. package/get-shit-done/bin/lib/runtime-homes.cjs +222 -0
  171. package/get-shit-done/bin/lib/runtime-name-policy.cjs +83 -0
  172. package/get-shit-done/bin/lib/runtime-slash.cjs +112 -0
  173. package/get-shit-done/bin/lib/schema-detect.cjs +165 -0
  174. package/get-shit-done/bin/lib/secrets.cjs +32 -0
  175. package/get-shit-done/bin/lib/security.cjs +600 -0
  176. package/get-shit-done/bin/lib/semver-compare.cjs +35 -0
  177. package/get-shit-done/bin/lib/shell-command-projection.cjs +500 -0
  178. package/get-shit-done/bin/lib/state-command-router.cjs +252 -0
  179. package/get-shit-done/bin/lib/state-document.cjs +263 -0
  180. package/get-shit-done/bin/lib/state.cjs +2038 -0
  181. package/get-shit-done/bin/lib/surface.cjs +470 -0
  182. package/get-shit-done/bin/lib/task-command-router.cjs +81 -0
  183. package/get-shit-done/bin/lib/template.cjs +228 -0
  184. package/get-shit-done/bin/lib/uat.cjs +289 -0
  185. package/get-shit-done/bin/lib/update-context.cjs +209 -0
  186. package/get-shit-done/bin/lib/validate-command-router.cjs +83 -0
  187. package/get-shit-done/bin/lib/validate.cjs +92 -0
  188. package/get-shit-done/bin/lib/verify-command-router.cjs +40 -0
  189. package/get-shit-done/bin/lib/verify.cjs +1511 -0
  190. package/get-shit-done/bin/lib/workstream-inventory-builder.cjs +74 -0
  191. package/get-shit-done/bin/lib/workstream-inventory.cjs +146 -0
  192. package/get-shit-done/bin/lib/workstream-name-policy.cjs +94 -0
  193. package/get-shit-done/bin/lib/workstream.cjs +389 -0
  194. package/get-shit-done/bin/lib/worktree-safety.cjs +985 -0
  195. package/get-shit-done/bin/shared/config-defaults.manifest.json +97 -0
  196. package/get-shit-done/bin/shared/config-schema.manifest.json +175 -0
  197. package/get-shit-done/bin/shared/model-catalog.json +122 -0
  198. package/get-shit-done/bin/shared/runtime-aliases.manifest.json +75 -0
  199. package/get-shit-done/bin/verify-reapply-patches.cjs +352 -0
  200. package/get-shit-done/contexts/dev.md +21 -0
  201. package/get-shit-done/contexts/research.md +22 -0
  202. package/get-shit-done/contexts/review.md +23 -0
  203. package/get-shit-done/references/agent-contracts.md +79 -0
  204. package/get-shit-done/references/ai-evals.md +156 -0
  205. package/get-shit-done/references/ai-frameworks.md +186 -0
  206. package/get-shit-done/references/artifact-types.md +131 -0
  207. package/get-shit-done/references/autonomous-smart-discuss.md +277 -0
  208. package/get-shit-done/references/checkpoints.md +814 -0
  209. package/get-shit-done/references/common-bug-patterns.md +114 -0
  210. package/get-shit-done/references/context-budget.md +85 -0
  211. package/get-shit-done/references/continuation-format.md +253 -0
  212. package/get-shit-done/references/debugger-philosophy.md +76 -0
  213. package/get-shit-done/references/decimal-phase-calculation.md +64 -0
  214. package/get-shit-done/references/doc-conflict-engine.md +91 -0
  215. package/get-shit-done/references/domain-probes.md +125 -0
  216. package/get-shit-done/references/execute-mvp-tdd.md +81 -0
  217. package/get-shit-done/references/executor-examples.md +110 -0
  218. package/get-shit-done/references/few-shot-examples/plan-checker.md +73 -0
  219. package/get-shit-done/references/few-shot-examples/verifier.md +109 -0
  220. package/get-shit-done/references/gate-prompts.md +100 -0
  221. package/get-shit-done/references/gates.md +70 -0
  222. package/get-shit-done/references/git-integration.md +298 -0
  223. package/get-shit-done/references/git-planning-commit.md +40 -0
  224. package/get-shit-done/references/ios-scaffold.md +123 -0
  225. package/get-shit-done/references/mandatory-initial-read.md +2 -0
  226. package/get-shit-done/references/model-profile-resolution.md +38 -0
  227. package/get-shit-done/references/model-profiles.md +245 -0
  228. package/get-shit-done/references/mvp-concepts.md +49 -0
  229. package/get-shit-done/references/phase-argument-parsing.md +61 -0
  230. package/get-shit-done/references/planner-antipatterns.md +89 -0
  231. package/get-shit-done/references/planner-chunked.md +49 -0
  232. package/get-shit-done/references/planner-gap-closure.md +62 -0
  233. package/get-shit-done/references/planner-graphify-auto-update.md +67 -0
  234. package/get-shit-done/references/planner-human-verify-mode.md +57 -0
  235. package/get-shit-done/references/planner-interface-context.md +62 -0
  236. package/get-shit-done/references/planner-mvp-mode.md +53 -0
  237. package/get-shit-done/references/planner-reviews.md +39 -0
  238. package/get-shit-done/references/planner-revision.md +87 -0
  239. package/get-shit-done/references/planner-source-audit.md +73 -0
  240. package/get-shit-done/references/planning-config.md +471 -0
  241. package/get-shit-done/references/project-skills-discovery.md +19 -0
  242. package/get-shit-done/references/questioning.md +162 -0
  243. package/get-shit-done/references/revision-loop.md +97 -0
  244. package/get-shit-done/references/scout-codebase.md +51 -0
  245. package/get-shit-done/references/skeleton-template.md +48 -0
  246. package/get-shit-done/references/sketch-interactivity.md +41 -0
  247. package/get-shit-done/references/sketch-theme-system.md +94 -0
  248. package/get-shit-done/references/sketch-tooling.md +45 -0
  249. package/get-shit-done/references/sketch-variant-patterns.md +81 -0
  250. package/get-shit-done/references/spidr-splitting.md +69 -0
  251. package/get-shit-done/references/tdd.md +330 -0
  252. package/get-shit-done/references/thinking-models-debug.md +44 -0
  253. package/get-shit-done/references/thinking-models-execution.md +50 -0
  254. package/get-shit-done/references/thinking-models-planning.md +62 -0
  255. package/get-shit-done/references/thinking-models-research.md +50 -0
  256. package/get-shit-done/references/thinking-models-verification.md +55 -0
  257. package/get-shit-done/references/thinking-partner.md +96 -0
  258. package/get-shit-done/references/ui-brand.md +160 -0
  259. package/get-shit-done/references/universal-anti-patterns.md +63 -0
  260. package/get-shit-done/references/user-profiling.md +681 -0
  261. package/get-shit-done/references/user-story-template.md +58 -0
  262. package/get-shit-done/references/verification-overrides.md +227 -0
  263. package/get-shit-done/references/verification-patterns.md +612 -0
  264. package/get-shit-done/references/verify-mvp-mode.md +85 -0
  265. package/get-shit-done/references/workstream-flag.md +111 -0
  266. package/get-shit-done/references/worktree-path-safety.md +89 -0
  267. package/get-shit-done/templates/AI-SPEC.md +246 -0
  268. package/get-shit-done/templates/DEBUG.md +169 -0
  269. package/get-shit-done/templates/README.md +77 -0
  270. package/get-shit-done/templates/SECURITY.md +61 -0
  271. package/get-shit-done/templates/UAT.md +265 -0
  272. package/get-shit-done/templates/UI-SPEC.md +100 -0
  273. package/get-shit-done/templates/VALIDATION.md +76 -0
  274. package/get-shit-done/templates/claude-md.md +145 -0
  275. package/get-shit-done/templates/codebase/architecture.md +255 -0
  276. package/get-shit-done/templates/codebase/concerns.md +310 -0
  277. package/get-shit-done/templates/codebase/conventions.md +307 -0
  278. package/get-shit-done/templates/codebase/integrations.md +280 -0
  279. package/get-shit-done/templates/codebase/stack.md +186 -0
  280. package/get-shit-done/templates/codebase/structure.md +285 -0
  281. package/get-shit-done/templates/codebase/testing.md +480 -0
  282. package/get-shit-done/templates/config.json +62 -0
  283. package/get-shit-done/templates/context.md +352 -0
  284. package/get-shit-done/templates/continue-here.md +78 -0
  285. package/get-shit-done/templates/copilot-instructions.md +7 -0
  286. package/get-shit-done/templates/debug-subagent-prompt.md +91 -0
  287. package/get-shit-done/templates/dev-preferences.md +21 -0
  288. package/get-shit-done/templates/discovery.md +146 -0
  289. package/get-shit-done/templates/discussion-log.md +63 -0
  290. package/get-shit-done/templates/milestone-archive.md +123 -0
  291. package/get-shit-done/templates/milestone.md +115 -0
  292. package/get-shit-done/templates/phase-prompt.md +610 -0
  293. package/get-shit-done/templates/planner-subagent-prompt.md +117 -0
  294. package/get-shit-done/templates/project.md +186 -0
  295. package/get-shit-done/templates/requirements.md +231 -0
  296. package/get-shit-done/templates/research-project/ARCHITECTURE.md +204 -0
  297. package/get-shit-done/templates/research-project/FEATURES.md +147 -0
  298. package/get-shit-done/templates/research-project/PITFALLS.md +200 -0
  299. package/get-shit-done/templates/research-project/STACK.md +120 -0
  300. package/get-shit-done/templates/research-project/SUMMARY.md +170 -0
  301. package/get-shit-done/templates/research.md +592 -0
  302. package/get-shit-done/templates/retrospective.md +54 -0
  303. package/get-shit-done/templates/roadmap.md +202 -0
  304. package/get-shit-done/templates/spec.md +307 -0
  305. package/get-shit-done/templates/state.md +195 -0
  306. package/get-shit-done/templates/summary-complex.md +59 -0
  307. package/get-shit-done/templates/summary-minimal.md +41 -0
  308. package/get-shit-done/templates/summary-standard.md +48 -0
  309. package/get-shit-done/templates/summary.md +248 -0
  310. package/get-shit-done/templates/user-profile.md +146 -0
  311. package/get-shit-done/templates/user-setup.md +311 -0
  312. package/get-shit-done/templates/verification-report.md +322 -0
  313. package/get-shit-done/workflows/_runtime-launcher.snippet.sh +1 -0
  314. package/get-shit-done/workflows/add-backlog.md +91 -0
  315. package/get-shit-done/workflows/add-phase.md +113 -0
  316. package/get-shit-done/workflows/add-tests.md +355 -0
  317. package/get-shit-done/workflows/add-todo.md +161 -0
  318. package/get-shit-done/workflows/ai-integration-phase.md +295 -0
  319. package/get-shit-done/workflows/analyze-dependencies.md +96 -0
  320. package/get-shit-done/workflows/audit-fix.md +178 -0
  321. package/get-shit-done/workflows/audit-milestone.md +358 -0
  322. package/get-shit-done/workflows/audit-uat.md +110 -0
  323. package/get-shit-done/workflows/autonomous.md +795 -0
  324. package/get-shit-done/workflows/check-todos.md +180 -0
  325. package/get-shit-done/workflows/cleanup.md +155 -0
  326. package/get-shit-done/workflows/code-review-fix.md +502 -0
  327. package/get-shit-done/workflows/code-review.md +656 -0
  328. package/get-shit-done/workflows/complete-milestone.md +855 -0
  329. package/get-shit-done/workflows/debug.md +232 -0
  330. package/get-shit-done/workflows/diagnose-issues.md +241 -0
  331. package/get-shit-done/workflows/discovery-phase.md +291 -0
  332. package/get-shit-done/workflows/discuss-phase/modes/advisor.md +176 -0
  333. package/get-shit-done/workflows/discuss-phase/modes/all.md +28 -0
  334. package/get-shit-done/workflows/discuss-phase/modes/analyze.md +44 -0
  335. package/get-shit-done/workflows/discuss-phase/modes/auto.md +57 -0
  336. package/get-shit-done/workflows/discuss-phase/modes/batch.md +52 -0
  337. package/get-shit-done/workflows/discuss-phase/modes/chain.md +98 -0
  338. package/get-shit-done/workflows/discuss-phase/modes/default.md +141 -0
  339. package/get-shit-done/workflows/discuss-phase/modes/power.md +44 -0
  340. package/get-shit-done/workflows/discuss-phase/modes/text.md +55 -0
  341. package/get-shit-done/workflows/discuss-phase/templates/checkpoint.json +18 -0
  342. package/get-shit-done/workflows/discuss-phase/templates/context.md +136 -0
  343. package/get-shit-done/workflows/discuss-phase/templates/discussion-log.md +50 -0
  344. package/get-shit-done/workflows/discuss-phase-assumptions.md +675 -0
  345. package/get-shit-done/workflows/discuss-phase-power.md +291 -0
  346. package/get-shit-done/workflows/discuss-phase.md +499 -0
  347. package/get-shit-done/workflows/do.md +111 -0
  348. package/get-shit-done/workflows/docs-update.md +1162 -0
  349. package/get-shit-done/workflows/edit-phase.md +295 -0
  350. package/get-shit-done/workflows/eval-review.md +156 -0
  351. package/get-shit-done/workflows/execute-phase/steps/codebase-drift-gate.md +82 -0
  352. package/get-shit-done/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  353. package/get-shit-done/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  354. package/get-shit-done/workflows/execute-phase.md +1709 -0
  355. package/get-shit-done/workflows/execute-plan.md +526 -0
  356. package/get-shit-done/workflows/explore.md +144 -0
  357. package/get-shit-done/workflows/extract-learnings.md +243 -0
  358. package/get-shit-done/workflows/fast.md +124 -0
  359. package/get-shit-done/workflows/forensics.md +279 -0
  360. package/get-shit-done/workflows/graduation.md +196 -0
  361. package/get-shit-done/workflows/health.md +224 -0
  362. package/get-shit-done/workflows/help/modes/brief.md +22 -0
  363. package/get-shit-done/workflows/help/modes/default.md +50 -0
  364. package/get-shit-done/workflows/help/modes/full.md +784 -0
  365. package/get-shit-done/workflows/help/modes/topic.md +74 -0
  366. package/get-shit-done/workflows/help.md +24 -0
  367. package/get-shit-done/workflows/import.md +254 -0
  368. package/get-shit-done/workflows/inbox.md +387 -0
  369. package/get-shit-done/workflows/ingest-docs.md +339 -0
  370. package/get-shit-done/workflows/insert-phase.md +152 -0
  371. package/get-shit-done/workflows/list-phase-assumptions.md +178 -0
  372. package/get-shit-done/workflows/list-workspaces.md +57 -0
  373. package/get-shit-done/workflows/manager.md +393 -0
  374. package/get-shit-done/workflows/map-codebase.md +444 -0
  375. package/get-shit-done/workflows/milestone-summary.md +224 -0
  376. package/get-shit-done/workflows/mvp-phase.md +222 -0
  377. package/get-shit-done/workflows/new-milestone.md +635 -0
  378. package/get-shit-done/workflows/new-project.md +1555 -0
  379. package/get-shit-done/workflows/new-workspace.md +240 -0
  380. package/get-shit-done/workflows/next.md +299 -0
  381. package/get-shit-done/workflows/node-repair.md +92 -0
  382. package/get-shit-done/workflows/note.md +158 -0
  383. package/get-shit-done/workflows/pause-work.md +244 -0
  384. package/get-shit-done/workflows/plan-milestone-gaps.md +281 -0
  385. package/get-shit-done/workflows/plan-phase.md +1809 -0
  386. package/get-shit-done/workflows/plan-review-convergence.md +346 -0
  387. package/get-shit-done/workflows/plant-seed.md +230 -0
  388. package/get-shit-done/workflows/pr-branch.md +157 -0
  389. package/get-shit-done/workflows/profile-user.md +453 -0
  390. package/get-shit-done/workflows/progress.md +699 -0
  391. package/get-shit-done/workflows/quick.md +1039 -0
  392. package/get-shit-done/workflows/reapply-patches.md +426 -0
  393. package/get-shit-done/workflows/remove-phase.md +156 -0
  394. package/get-shit-done/workflows/remove-workspace.md +108 -0
  395. package/get-shit-done/workflows/resume-project.md +332 -0
  396. package/get-shit-done/workflows/review.md +623 -0
  397. package/get-shit-done/workflows/scan.md +105 -0
  398. package/get-shit-done/workflows/secure-phase.md +180 -0
  399. package/get-shit-done/workflows/session-report.md +146 -0
  400. package/get-shit-done/workflows/settings-advanced.md +620 -0
  401. package/get-shit-done/workflows/settings-integrations.md +312 -0
  402. package/get-shit-done/workflows/settings.md +552 -0
  403. package/get-shit-done/workflows/ship.md +356 -0
  404. package/get-shit-done/workflows/sketch-wrap-up.md +286 -0
  405. package/get-shit-done/workflows/sketch.md +361 -0
  406. package/get-shit-done/workflows/spec-phase.md +262 -0
  407. package/get-shit-done/workflows/spike-wrap-up.md +307 -0
  408. package/get-shit-done/workflows/spike.md +453 -0
  409. package/get-shit-done/workflows/stats.md +80 -0
  410. package/get-shit-done/workflows/sync-skills.md +182 -0
  411. package/get-shit-done/workflows/thread.md +222 -0
  412. package/get-shit-done/workflows/transition.md +694 -0
  413. package/get-shit-done/workflows/ui-phase.md +328 -0
  414. package/get-shit-done/workflows/ui-review.md +193 -0
  415. package/get-shit-done/workflows/ultraplan-phase.md +199 -0
  416. package/get-shit-done/workflows/undo.md +314 -0
  417. package/get-shit-done/workflows/update.md +443 -0
  418. package/get-shit-done/workflows/validate-phase.md +179 -0
  419. package/get-shit-done/workflows/verify-phase.md +544 -0
  420. package/get-shit-done/workflows/verify-work.md +781 -0
  421. package/hooks/dist/gsd-check-update-worker.js +95 -0
  422. package/hooks/dist/gsd-check-update.js +64 -0
  423. package/hooks/dist/gsd-context-monitor.js +195 -0
  424. package/hooks/dist/gsd-graphify-update.sh +158 -0
  425. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  426. package/hooks/dist/gsd-prompt-guard.js +97 -0
  427. package/hooks/dist/gsd-read-guard.js +101 -0
  428. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  429. package/hooks/dist/gsd-session-state.sh +59 -0
  430. package/hooks/dist/gsd-statusline.js +548 -0
  431. package/hooks/dist/gsd-update-banner.js +134 -0
  432. package/hooks/dist/gsd-validate-commit.sh +57 -0
  433. package/hooks/dist/gsd-workflow-guard.js +166 -0
  434. package/hooks/dist/lib/git-cmd.js +150 -0
  435. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  436. package/hooks/gsd-check-update-worker.js +95 -0
  437. package/hooks/gsd-check-update.js +64 -0
  438. package/hooks/gsd-context-monitor.js +195 -0
  439. package/hooks/gsd-graphify-update.sh +158 -0
  440. package/hooks/gsd-phase-boundary.sh +47 -0
  441. package/hooks/gsd-prompt-guard.js +97 -0
  442. package/hooks/gsd-read-guard.js +101 -0
  443. package/hooks/gsd-read-injection-scanner.js +203 -0
  444. package/hooks/gsd-session-state.sh +59 -0
  445. package/hooks/gsd-statusline.js +548 -0
  446. package/hooks/gsd-update-banner.js +134 -0
  447. package/hooks/gsd-validate-commit.sh +57 -0
  448. package/hooks/gsd-workflow-guard.js +166 -0
  449. package/hooks/lib/git-cmd.js +150 -0
  450. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  451. package/hooks/managed-hooks-registry.cjs +34 -0
  452. package/package.json +102 -0
  453. package/scripts/affected-tests-lib.cjs +541 -0
  454. package/scripts/audit-workflow-script-paths.cjs +73 -0
  455. package/scripts/base64-scan.sh +339 -0
  456. package/scripts/build-hooks.js +236 -0
  457. package/scripts/changeset/README.md +129 -0
  458. package/scripts/changeset/cli.cjs +392 -0
  459. package/scripts/changeset/github-release-notes.cjs +199 -0
  460. package/scripts/changeset/lint.cjs +110 -0
  461. package/scripts/changeset/new.cjs +137 -0
  462. package/scripts/changeset/parse.cjs +114 -0
  463. package/scripts/changeset/render.cjs +34 -0
  464. package/scripts/changeset/serialize.cjs +130 -0
  465. package/scripts/check-alias-drift.cjs +108 -0
  466. package/scripts/check-env.cjs +302 -0
  467. package/scripts/check-npm-integrity.cjs +209 -0
  468. package/scripts/ci-guard-runner.cjs +16 -0
  469. package/scripts/ci-prepare-test-scope.cjs +46 -0
  470. package/scripts/ci-rebase-check.cjs +85 -0
  471. package/scripts/ci-test-scope.cjs +302 -0
  472. package/scripts/command-contract-helpers.cjs +64 -0
  473. package/scripts/diff-touches-shipped-paths.cjs +147 -0
  474. package/scripts/fix-slash-commands.cjs +147 -0
  475. package/scripts/gen-inventory-manifest.cjs +109 -0
  476. package/scripts/generate-package-identity.cjs +104 -0
  477. package/scripts/lint-command-contract.cjs +108 -0
  478. package/scripts/lint-descriptions.cjs +83 -0
  479. package/scripts/lint-docs-required.cjs +222 -0
  480. package/scripts/lint-no-source-grep-extras.cjs +81 -0
  481. package/scripts/lint-no-source-grep.cjs +174 -0
  482. package/scripts/lint-package-identity-drift.cjs +141 -0
  483. package/scripts/lint-pr-check-project-dir.cjs +98 -0
  484. package/scripts/lint-shared-module-handsync.cjs +388 -0
  485. package/scripts/lint-shell-command-projection-drift.cjs +57 -0
  486. package/scripts/lint-skill-deps.cjs +180 -0
  487. package/scripts/lint-test-file-count.allowlist.json +36 -0
  488. package/scripts/lint-test-file-count.cjs +190 -0
  489. package/scripts/pr-template-policy.cjs +268 -0
  490. package/scripts/prompt-injection-scan.sh +203 -0
  491. package/scripts/release-tarball-smoke.cjs +627 -0
  492. package/scripts/run-affected-tests.cjs +6 -0
  493. package/scripts/run-cross-platform-tests.cjs +63 -0
  494. package/scripts/run-tests.cjs +282 -0
  495. package/scripts/secret-scan-lint.sh +231 -0
  496. package/scripts/secret-scan.sh +358 -0
  497. package/scripts/setup-branch-protection.sh +236 -0
  498. package/scripts/shared-module-handsync-allowlist.json +183 -0
  499. package/scripts/strip-prose-atrefs.cjs +106 -0
  500. package/scripts/sync-rulesets.sh +34 -0
  501. package/scripts/sync-runtime-launcher.cjs +402 -0
  502. package/scripts/test-failure-reasons.cjs +34 -0
  503. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,74 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Workstream Inventory Builder — pure projection from pre-collected
5
+ * filesystem data to typed WorkstreamInventory. No I/O. No async.
6
+ */
7
+
8
+ const path = require('path');
9
+ const relative = path.relative;
10
+
11
+ // Internal helpers
12
+ function toPosixPath(p) {
13
+ return p.split('\\').join('/');
14
+ }
15
+
16
+ function isCompletedInventory(status) {
17
+ const s = String(status ?? '').trim().toLowerCase();
18
+ return /\bmilestone\s+complete\b/.test(s) || /\barchived\b/.test(s);
19
+ }
20
+
21
+ function buildWorkstreamInventory(inputs) {
22
+ const { name, projectDir, workstreamDir, phaseDirNames, activeWorkstreamName, phaseFilesCounts, roadmapPhaseCount, stateProjection, filesExist, } = inputs;
23
+ // Index counts by directory for O(1) lookup during sort/iteration
24
+ const countsMap = new Map();
25
+ for (const entry of phaseFilesCounts) {
26
+ countsMap.set(entry.directory, { planCount: entry.planCount, summaryCount: entry.summaryCount });
27
+ }
28
+ const phases = [];
29
+ let completedPhases = 0;
30
+ let totalPlans = 0;
31
+ let completedPlans = 0;
32
+ for (const dir of [...phaseDirNames].sort()) {
33
+ const counts = countsMap.get(dir) ?? { planCount: 0, summaryCount: 0 };
34
+ const status = counts.summaryCount >= counts.planCount && counts.planCount > 0
35
+ ? 'complete'
36
+ : counts.planCount > 0
37
+ ? 'in_progress'
38
+ : 'pending';
39
+ totalPlans += counts.planCount;
40
+ completedPlans += Math.min(counts.summaryCount, counts.planCount);
41
+ if (status === 'complete')
42
+ completedPhases++;
43
+ phases.push({
44
+ directory: dir,
45
+ status,
46
+ plan_count: counts.planCount,
47
+ summary_count: counts.summaryCount,
48
+ });
49
+ }
50
+ return {
51
+ name,
52
+ path: toPosixPath(relative(projectDir, workstreamDir)),
53
+ active: name === activeWorkstreamName,
54
+ files: {
55
+ roadmap: filesExist.roadmap,
56
+ state: filesExist.state,
57
+ requirements: filesExist.requirements,
58
+ },
59
+ status: stateProjection.status,
60
+ current_phase: stateProjection.current_phase,
61
+ last_activity: stateProjection.last_activity,
62
+ phases,
63
+ phase_count: phases.length,
64
+ completed_phases: completedPhases,
65
+ roadmap_phase_count: roadmapPhaseCount,
66
+ total_plans: totalPlans,
67
+ completed_plans: completedPlans,
68
+ progress_percent: roadmapPhaseCount > 0
69
+ ? Math.min(100, Math.round((completedPhases / roadmapPhaseCount) * 100))
70
+ : 0,
71
+ };
72
+ }
73
+
74
+ module.exports = { buildWorkstreamInventory, isCompletedInventory };
@@ -0,0 +1,146 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Workstream Inventory Module
5
+ *
6
+ * Owns discovery and read-only projection of .planning/workstreams/* state.
7
+ * Command handlers should render outputs from this inventory instead of
8
+ * rescanning workstream directories directly.
9
+ *
10
+ * Pure projection logic lives in workstream-inventory-builder.generated.cjs.
11
+ * This module handles I/O orchestration only.
12
+ */
13
+
14
+ const fs = require('fs');
15
+ const path = require('path');
16
+ const { toPosixPath, readSubdirectories } = require('./core.cjs');
17
+ const scanPhasePlans = require('./plan-scan.cjs');
18
+ const { planningPaths, planningRoot, getActiveWorkstream } = require('./planning-workspace.cjs');
19
+ const { stateExtractField } = require('./state-document.cjs');
20
+ const { buildWorkstreamInventory, isCompletedInventory } = require('./workstream-inventory-builder.cjs');
21
+
22
+ function workstreamsRoot(cwd) {
23
+ return path.join(planningRoot(cwd), 'workstreams');
24
+ }
25
+
26
+ function countRoadmapPhases(roadmapPath, fallbackCount) {
27
+ try {
28
+ const roadmapContent = fs.readFileSync(roadmapPath, 'utf-8');
29
+ const matches = roadmapContent.match(/^#{2,4}\s+Phase\s+[\w][\w.-]*/gm);
30
+ return matches ? matches.length : fallbackCount;
31
+ } catch {
32
+ return fallbackCount;
33
+ }
34
+ }
35
+
36
+ function countPhaseFiles(phaseDir) {
37
+ const scan = scanPhasePlans(phaseDir);
38
+ return { planCount: scan.planCount, summaryCount: scan.summaryCount };
39
+ }
40
+
41
+ function readStateProjection(statePath) {
42
+ try {
43
+ const stateContent = fs.readFileSync(statePath, 'utf-8');
44
+ return {
45
+ status: stateExtractField(stateContent, 'Status') || 'unknown',
46
+ current_phase: stateExtractField(stateContent, 'Current Phase'),
47
+ last_activity: stateExtractField(stateContent, 'Last Activity'),
48
+ };
49
+ } catch {
50
+ return {
51
+ status: 'unknown',
52
+ current_phase: null,
53
+ last_activity: null,
54
+ };
55
+ }
56
+ }
57
+
58
+ function sortWorkstreamInventories(inventories, activeWorkstreamName) {
59
+ return [...inventories].sort((a, b) => {
60
+ const aActive = a.name === activeWorkstreamName ? 1 : 0;
61
+ const bActive = b.name === activeWorkstreamName ? 1 : 0;
62
+ if (aActive !== bActive) {
63
+ return bActive - aActive;
64
+ }
65
+ return a.name.localeCompare(b.name);
66
+ });
67
+ }
68
+
69
+ function inspectWorkstream(cwd, name, options = {}) {
70
+ const wsDir = path.join(workstreamsRoot(cwd), name);
71
+ if (!fs.existsSync(wsDir)) return null;
72
+
73
+ const activeWorkstreamName = options.active === undefined ? getActiveWorkstream(cwd) : options.active;
74
+ const p = planningPaths(cwd, name);
75
+ const phaseDirNames = readSubdirectories(p.phases);
76
+
77
+ // Collect per-phase file counts
78
+ const phaseFilesCounts = phaseDirNames.map(dir => {
79
+ const counts = countPhaseFiles(path.join(p.phases, dir));
80
+ return { directory: dir, planCount: counts.planCount, summaryCount: counts.summaryCount };
81
+ });
82
+
83
+ return buildWorkstreamInventory({
84
+ name,
85
+ projectDir: cwd,
86
+ workstreamDir: wsDir,
87
+ phaseDirNames,
88
+ activeWorkstreamName,
89
+ phaseFilesCounts,
90
+ roadmapPhaseCount: countRoadmapPhases(p.roadmap, phaseDirNames.length),
91
+ stateProjection: readStateProjection(p.state),
92
+ filesExist: {
93
+ roadmap: fs.existsSync(p.roadmap),
94
+ state: fs.existsSync(p.state),
95
+ requirements: fs.existsSync(p.requirements),
96
+ },
97
+ });
98
+ }
99
+
100
+ function listWorkstreamInventories(cwd) {
101
+ const wsRoot = workstreamsRoot(cwd);
102
+ if (!fs.existsSync(wsRoot)) {
103
+ return {
104
+ mode: 'flat',
105
+ active: null,
106
+ workstreams: [],
107
+ count: 0,
108
+ message: 'No workstreams — operating in flat mode',
109
+ };
110
+ }
111
+
112
+ const active = getActiveWorkstream(cwd);
113
+ const entries = fs.readdirSync(wsRoot, { withFileTypes: true });
114
+ const workstreams = [];
115
+ for (const entry of entries) {
116
+ if (!entry.isDirectory()) continue;
117
+ const inventory = inspectWorkstream(cwd, entry.name, { active });
118
+ if (inventory) workstreams.push(inventory);
119
+ }
120
+
121
+ const ordered = sortWorkstreamInventories(workstreams, active);
122
+
123
+ return {
124
+ mode: 'workstream',
125
+ active,
126
+ workstreams: ordered,
127
+ count: ordered.length,
128
+ };
129
+ }
130
+
131
+ function getOtherActiveWorkstreamInventories(cwd, excludeWs) {
132
+ return listWorkstreamInventories(cwd).workstreams
133
+ .filter(inventory => inventory.name !== excludeWs)
134
+ .filter(inventory => !isCompletedInventory(inventory.status));
135
+ }
136
+
137
+ module.exports = {
138
+ countPhaseFiles,
139
+ countRoadmapPhases,
140
+ getOtherActiveWorkstreamInventories,
141
+ inspectWorkstream,
142
+ isCompletedInventory,
143
+ listWorkstreamInventories,
144
+ sortWorkstreamInventories,
145
+ workstreamsRoot,
146
+ };
@@ -0,0 +1,94 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Canonical workstream name validation and slug normalization.
5
+ * Used by active-workstream-store.cjs, planning-workspace.cjs, workstream.cjs.
6
+ */
7
+
8
+ const ACTIVE_WORKSTREAM_RE = /^[a-zA-Z0-9][a-zA-Z0-9._-]*$/;
9
+ const INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE = 'Invalid workstream name: must be alphanumeric, hyphens, underscores, or dots';
10
+
11
+ function normalizeWorkstreamNameInput(name) {
12
+ const value = String(name || '').trim();
13
+ return value || null;
14
+ }
15
+
16
+ function validateActiveWorkstreamName(name) {
17
+ const value = normalizeWorkstreamNameInput(name);
18
+ if (!value) {
19
+ return {
20
+ ok: false,
21
+ reason: 'empty',
22
+ value: null,
23
+ };
24
+ }
25
+ if (hasInvalidPathSegment(value) || !ACTIVE_WORKSTREAM_RE.test(value)) {
26
+ return {
27
+ ok: false,
28
+ reason: 'invalid',
29
+ value,
30
+ };
31
+ }
32
+ return {
33
+ ok: true,
34
+ reason: null,
35
+ value,
36
+ };
37
+ }
38
+ /**
39
+ * Validate a workstream name.
40
+ * Allowed: alphanumeric, hyphens, underscores, dots.
41
+ * Disallowed: empty, spaces, slashes, special chars, path traversal.
42
+ *
43
+ * Alias for isValidActiveWorkstreamName; provided for SDK-layer callers.
44
+ */
45
+ function validateWorkstreamName(name) {
46
+ return isValidActiveWorkstreamName(name);
47
+ }
48
+ /**
49
+ * Convert a display name to a URL/filesystem-safe workstream slug.
50
+ * Lowercases, collapses non-alphanumeric runs to hyphens, strips leading/trailing hyphens.
51
+ */
52
+ function toWorkstreamSlug(name) {
53
+ return String(name || '')
54
+ .toLowerCase()
55
+ .replace(/[^a-z0-9]+/g, '-')
56
+ .replace(/^-+|-+$/g, '');
57
+ }
58
+ /**
59
+ * Returns true when `name` contains a path separator, a bare dot, or a
60
+ * dot-dot sequence — any of which would make the name unsafe for use as a
61
+ * filesystem path segment.
62
+ */
63
+ function hasInvalidPathSegment(name) {
64
+ const value = String(name || '');
65
+ return /[/\\]/.test(value) || value === '.' || value === '..' || value.includes('..');
66
+ }
67
+ /**
68
+ * Returns true when `name` is a valid active workstream name:
69
+ * - Must start with alphanumeric
70
+ * - May contain alphanumeric, dots, underscores, hyphens
71
+ * - Must not contain path traversal sequences (..)
72
+ */
73
+ function isValidActiveWorkstreamName(name) {
74
+ return validateActiveWorkstreamName(name).ok;
75
+ }
76
+
77
+ function assertValidActiveWorkstreamName(name, errorMessage = INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE) {
78
+ const validation = validateActiveWorkstreamName(name);
79
+ if (!validation.ok) {
80
+ throw new Error(errorMessage);
81
+ }
82
+ return validation.value;
83
+ }
84
+
85
+ module.exports = {
86
+ INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE,
87
+ normalizeWorkstreamNameInput,
88
+ validateActiveWorkstreamName,
89
+ validateWorkstreamName,
90
+ toWorkstreamSlug,
91
+ hasInvalidPathSegment,
92
+ isValidActiveWorkstreamName,
93
+ assertValidActiveWorkstreamName,
94
+ };
@@ -0,0 +1,389 @@
1
+ /**
2
+ * Workstream — CRUD operations for workstream namespacing
3
+ *
4
+ * Workstreams enable parallel milestones by scoping ROADMAP.md, STATE.md,
5
+ * REQUIREMENTS.md, and phases/ into .planning/workstreams/{name}/ directories.
6
+ *
7
+ * When no workstreams/ directory exists, GSD operates in "flat mode" with
8
+ * everything at .planning/ — backward compatible with pre-workstream installs.
9
+ */
10
+
11
+ const fs = require('fs');
12
+ const path = require('path');
13
+ const { output, error, toPosixPath, getMilestoneInfo, generateSlugInternal } = require('./core.cjs');
14
+ const { platformWriteSync, platformEnsureDir } = require('./shell-command-projection.cjs');
15
+ const { planningRoot, setActiveWorkstream, getActiveWorkstream } = require('./planning-workspace.cjs');
16
+ const {
17
+ toWorkstreamSlug,
18
+ assertValidActiveWorkstreamName,
19
+ isValidActiveWorkstreamName,
20
+ INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE,
21
+ } = require('./workstream-name-policy.cjs');
22
+ const { formatGsdSlash, resolveRuntime } = require('./runtime-slash.cjs');
23
+ const {
24
+ getOtherActiveWorkstreamInventories,
25
+ inspectWorkstream,
26
+ listWorkstreamInventories,
27
+ } = require('./workstream-inventory.cjs');
28
+
29
+ // ─── Migration ──────────────────────────────────────────────────────────────
30
+
31
+ /**
32
+ * Migrate flat .planning/ layout to workstream mode.
33
+ * Moves per-workstream files (ROADMAP.md, STATE.md, REQUIREMENTS.md, phases/)
34
+ * into .planning/workstreams/{name}/. Shared files (PROJECT.md, config.json,
35
+ * milestones/, research/, codebase/, todos/) stay in place.
36
+ */
37
+ function migrateToWorkstreams(cwd, workstreamName) {
38
+ try {
39
+ assertValidActiveWorkstreamName(workstreamName, 'Invalid workstream name for migration');
40
+ } catch (err) {
41
+ throw new Error('Invalid workstream name for migration');
42
+ }
43
+
44
+ const baseDir = planningRoot(cwd);
45
+ const wsDir = path.join(baseDir, 'workstreams', workstreamName);
46
+
47
+ if (fs.existsSync(path.join(baseDir, 'workstreams'))) {
48
+ throw new Error('Already in workstream mode — .planning/workstreams/ exists');
49
+ }
50
+
51
+ const toMove = [
52
+ { name: 'ROADMAP.md', type: 'file' },
53
+ { name: 'STATE.md', type: 'file' },
54
+ { name: 'REQUIREMENTS.md', type: 'file' },
55
+ { name: 'phases', type: 'dir' },
56
+ ];
57
+
58
+ platformEnsureDir(wsDir);
59
+
60
+ const filesMoved = [];
61
+ try {
62
+ for (const item of toMove) {
63
+ const src = path.join(baseDir, item.name);
64
+ if (fs.existsSync(src)) {
65
+ const dest = path.join(wsDir, item.name);
66
+ fs.renameSync(src, dest);
67
+ filesMoved.push(item.name);
68
+ }
69
+ }
70
+ } catch (err) {
71
+ for (const name of filesMoved) {
72
+ try { fs.renameSync(path.join(wsDir, name), path.join(baseDir, name)); } catch {}
73
+ }
74
+ try { fs.rmSync(wsDir, { recursive: true }); } catch {}
75
+ try { fs.rmdirSync(path.join(baseDir, 'workstreams')); } catch {}
76
+ throw err;
77
+ }
78
+
79
+ return { migrated: true, workstream: workstreamName, files_moved: filesMoved };
80
+ }
81
+
82
+ // ─── CRUD Commands ──────────────────────────────────────────────────────────
83
+
84
+ function cmdWorkstreamCreate(cwd, name, options, raw) {
85
+ if (!name) {
86
+ error('workstream name required. Usage: workstream create <name>');
87
+ }
88
+
89
+ const slug = toWorkstreamSlug(name);
90
+ if (!slug) {
91
+ error('Invalid workstream name — must contain at least one alphanumeric character');
92
+ }
93
+
94
+ const baseDir = planningRoot(cwd);
95
+ if (!fs.existsSync(baseDir)) {
96
+ error(`.planning/ directory not found — run ${formatGsdSlash('new-project', resolveRuntime(cwd))} first`);
97
+ }
98
+
99
+ const wsRoot = path.join(baseDir, 'workstreams');
100
+ const wsDir = path.join(wsRoot, slug);
101
+
102
+ if (fs.existsSync(wsDir) && fs.existsSync(path.join(wsDir, 'STATE.md'))) {
103
+ output({ created: false, error: 'already_exists', workstream: slug, path: toPosixPath(path.relative(cwd, wsDir)) }, raw);
104
+ return;
105
+ }
106
+
107
+ const isFlatMode = !fs.existsSync(wsRoot);
108
+ let migration = null;
109
+ if (isFlatMode && options.migrate !== false) {
110
+ const hasExistingWork = fs.existsSync(path.join(baseDir, 'ROADMAP.md')) ||
111
+ fs.existsSync(path.join(baseDir, 'STATE.md')) ||
112
+ fs.existsSync(path.join(baseDir, 'phases'));
113
+
114
+ if (hasExistingWork) {
115
+ const migrateName = options.migrateName || null;
116
+ let existingWsName;
117
+ if (migrateName) {
118
+ existingWsName = toWorkstreamSlug(migrateName);
119
+ if (!existingWsName) {
120
+ output({
121
+ created: false,
122
+ error: 'migration_failed',
123
+ message: 'Invalid migrate-name — must contain at least one alphanumeric character',
124
+ }, raw);
125
+ return;
126
+ }
127
+ } else {
128
+ try {
129
+ const milestone = getMilestoneInfo(cwd);
130
+ existingWsName = generateSlugInternal(milestone.name) || 'default';
131
+ } catch {
132
+ existingWsName = 'default';
133
+ }
134
+ }
135
+
136
+ try {
137
+ migration = migrateToWorkstreams(cwd, existingWsName);
138
+ } catch (e) {
139
+ output({ created: false, error: 'migration_failed', message: e.message }, raw);
140
+ return;
141
+ }
142
+ } else {
143
+ platformEnsureDir(wsRoot);
144
+ }
145
+ }
146
+
147
+ platformEnsureDir(wsDir);
148
+ platformEnsureDir(path.join(wsDir, 'phases'));
149
+
150
+ const today = new Date().toISOString().split('T')[0];
151
+ const stateContent = [
152
+ '---',
153
+ `workstream: ${slug}`,
154
+ `created: ${today}`,
155
+ '---',
156
+ '',
157
+ '# Project State',
158
+ '',
159
+ '## Current Position',
160
+ '**Status:** Not started',
161
+ '**Current Phase:** None',
162
+ `**Last Activity:** ${today}`,
163
+ '**Last Activity Description:** Workstream created',
164
+ '',
165
+ '## Progress',
166
+ '**Phases Complete:** 0',
167
+ '**Current Plan:** N/A',
168
+ '',
169
+ '## Session Continuity',
170
+ '**Stopped At:** N/A',
171
+ '**Resume File:** None',
172
+ '',
173
+ ].join('\n');
174
+
175
+ const statePath = path.join(wsDir, 'STATE.md');
176
+ if (!fs.existsSync(statePath)) {
177
+ platformWriteSync(statePath, stateContent);
178
+ }
179
+
180
+ setActiveWorkstream(cwd, slug);
181
+
182
+ const relPath = toPosixPath(path.relative(cwd, wsDir));
183
+ output({
184
+ created: true,
185
+ workstream: slug,
186
+ path: relPath,
187
+ state_path: relPath + '/STATE.md',
188
+ phases_path: relPath + '/phases',
189
+ migration: migration || null,
190
+ active: true,
191
+ }, raw);
192
+ }
193
+
194
+ function cmdWorkstreamList(cwd, raw) {
195
+ const inventory = listWorkstreamInventories(cwd);
196
+ if (inventory.mode === 'flat') {
197
+ output({ mode: 'flat', workstreams: [], message: inventory.message }, raw);
198
+ return;
199
+ }
200
+
201
+ const workstreams = inventory.workstreams.map(ws => ({
202
+ name: ws.name,
203
+ path: ws.path,
204
+ has_roadmap: ws.files.roadmap,
205
+ has_state: ws.files.state,
206
+ status: ws.status,
207
+ current_phase: ws.current_phase,
208
+ phase_count: ws.phase_count,
209
+ completed_phases: ws.completed_phases,
210
+ }));
211
+
212
+ output({ mode: 'workstream', workstreams, count: workstreams.length }, raw);
213
+ }
214
+
215
+ function cmdWorkstreamStatus(cwd, name, raw) {
216
+ if (!name) error('workstream name required. Usage: workstream status <name>');
217
+ try {
218
+ assertValidActiveWorkstreamName(name, INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE);
219
+ } catch {
220
+ error(INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE);
221
+ }
222
+
223
+ const wsDir = path.join(planningRoot(cwd), 'workstreams', name);
224
+ if (!fs.existsSync(wsDir)) {
225
+ output({ found: false, workstream: name }, raw);
226
+ return;
227
+ }
228
+
229
+ const inventory = inspectWorkstream(cwd, name);
230
+
231
+ output({
232
+ found: true,
233
+ workstream: name,
234
+ path: inventory.path,
235
+ files: inventory.files,
236
+ phases: inventory.phases,
237
+ phase_count: inventory.phase_count,
238
+ completed_phases: inventory.completed_phases,
239
+ status: inventory.status,
240
+ current_phase: inventory.current_phase,
241
+ last_activity: inventory.last_activity,
242
+ }, raw);
243
+ }
244
+
245
+ function cmdWorkstreamComplete(cwd, name, options, raw) {
246
+ if (!name) error('workstream name required. Usage: workstream complete <name>');
247
+ try {
248
+ assertValidActiveWorkstreamName(name, INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE);
249
+ } catch {
250
+ error(INVALID_ACTIVE_WORKSTREAM_NAME_MESSAGE);
251
+ }
252
+
253
+ const root = planningRoot(cwd);
254
+ const wsRoot = path.join(root, 'workstreams');
255
+ const wsDir = path.join(wsRoot, name);
256
+
257
+ if (!fs.existsSync(wsDir)) {
258
+ output({ completed: false, error: 'not_found', workstream: name }, raw);
259
+ return;
260
+ }
261
+
262
+ const active = getActiveWorkstream(cwd);
263
+ if (active === name) setActiveWorkstream(cwd, null);
264
+
265
+ const archiveDir = path.join(root, 'milestones');
266
+ const today = new Date().toISOString().split('T')[0];
267
+ let archivePath = path.join(archiveDir, `ws-${name}-${today}`);
268
+ let suffix = 1;
269
+ while (fs.existsSync(archivePath)) {
270
+ archivePath = path.join(archiveDir, `ws-${name}-${today}-${suffix++}`);
271
+ }
272
+
273
+ platformEnsureDir(archivePath);
274
+
275
+ const filesMoved = [];
276
+ try {
277
+ const entries = fs.readdirSync(wsDir, { withFileTypes: true });
278
+ for (const entry of entries) {
279
+ fs.renameSync(path.join(wsDir, entry.name), path.join(archivePath, entry.name));
280
+ filesMoved.push(entry.name);
281
+ }
282
+ } catch (err) {
283
+ for (const fname of filesMoved) {
284
+ try { fs.renameSync(path.join(archivePath, fname), path.join(wsDir, fname)); } catch {}
285
+ }
286
+ try { fs.rmSync(archivePath, { recursive: true }); } catch {}
287
+ if (active === name) setActiveWorkstream(cwd, name);
288
+ output({ completed: false, error: 'archive_failed', message: err.message, workstream: name }, raw);
289
+ return;
290
+ }
291
+
292
+ try { fs.rmdirSync(wsDir); } catch {}
293
+
294
+ let remainingWs = 0;
295
+ try {
296
+ remainingWs = fs.readdirSync(wsRoot, { withFileTypes: true }).filter(e => e.isDirectory()).length;
297
+ if (remainingWs === 0) fs.rmdirSync(wsRoot);
298
+ } catch {}
299
+
300
+ output({
301
+ completed: true,
302
+ workstream: name,
303
+ archived_to: toPosixPath(path.relative(cwd, archivePath)),
304
+ remaining_workstreams: remainingWs,
305
+ reverted_to_flat: remainingWs === 0,
306
+ }, raw);
307
+ }
308
+
309
+ // ─── Active Workstream Commands ──────────────────────────────────────────────
310
+
311
+ function cmdWorkstreamSet(cwd, name, raw) {
312
+ if (!name || name === '--clear') {
313
+ if (name !== '--clear') {
314
+ error('Workstream name required. Usage: workstream set <name> (or workstream set --clear to unset)');
315
+ }
316
+ const previous = getActiveWorkstream(cwd);
317
+ setActiveWorkstream(cwd, null);
318
+ output({ active: null, cleared: true, previous: previous || null }, raw);
319
+ return;
320
+ }
321
+
322
+ if (!isValidActiveWorkstreamName(name)) {
323
+ output({ active: null, error: 'invalid_name', message: 'Workstream name must be alphanumeric, hyphens, underscores, or dots' }, raw);
324
+ return;
325
+ }
326
+
327
+ const wsDir = path.join(planningRoot(cwd), 'workstreams', name);
328
+ if (!fs.existsSync(wsDir)) {
329
+ output({ active: null, error: 'not_found', workstream: name }, raw);
330
+ return;
331
+ }
332
+
333
+ setActiveWorkstream(cwd, name);
334
+ output({ active: name, set: true }, raw, name);
335
+ }
336
+
337
+ function cmdWorkstreamGet(cwd, raw) {
338
+ const active = getActiveWorkstream(cwd);
339
+ const wsRoot = path.join(planningRoot(cwd), 'workstreams');
340
+ output({ active, mode: fs.existsSync(wsRoot) ? 'workstream' : 'flat' }, raw, active || 'none');
341
+ }
342
+
343
+ function cmdWorkstreamProgress(cwd, raw) {
344
+ const inventory = listWorkstreamInventories(cwd);
345
+ if (inventory.mode === 'flat') {
346
+ output({ mode: 'flat', workstreams: [], message: inventory.message }, raw);
347
+ return;
348
+ }
349
+
350
+ const workstreams = inventory.workstreams.map(ws => ({
351
+ name: ws.name,
352
+ active: ws.active,
353
+ status: ws.status,
354
+ current_phase: ws.current_phase,
355
+ phases: `${ws.completed_phases}/${ws.roadmap_phase_count}`,
356
+ plans: `${ws.completed_plans}/${ws.total_plans}`,
357
+ progress_percent: ws.progress_percent,
358
+ }));
359
+
360
+ output({ mode: 'workstream', active: inventory.active, workstreams, count: workstreams.length }, raw);
361
+ }
362
+
363
+ // ─── Collision Detection ────────────────────────────────────────────────────
364
+
365
+ /**
366
+ * Return other workstreams that are NOT complete.
367
+ * Used to detect whether the milestone has active parallel work
368
+ * when a workstream finishes its last phase.
369
+ */
370
+ function getOtherActiveWorkstreams(cwd, excludeWs) {
371
+ return getOtherActiveWorkstreamInventories(cwd, excludeWs).map(ws => ({
372
+ name: ws.name,
373
+ status: ws.status,
374
+ current_phase: ws.current_phase,
375
+ phases: `${ws.completed_phases}/${ws.phase_count}`,
376
+ }));
377
+ }
378
+
379
+ module.exports = {
380
+ migrateToWorkstreams,
381
+ cmdWorkstreamCreate,
382
+ cmdWorkstreamList,
383
+ cmdWorkstreamStatus,
384
+ cmdWorkstreamComplete,
385
+ cmdWorkstreamSet,
386
+ cmdWorkstreamGet,
387
+ cmdWorkstreamProgress,
388
+ getOtherActiveWorkstreams,
389
+ };