@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,755 @@
1
+ /**
2
+ * Open Artifact Audit — Cross-type unresolved state scanner
3
+ *
4
+ * Scans all .planning/ artifact categories for items with open/unresolved state.
5
+ * Returns structured JSON for workflow consumption.
6
+ * Called by: gsd-tools.cjs audit-open
7
+ * Used by: /gsd:complete-milestone pre-close gate
8
+ */
9
+
10
+ 'use strict';
11
+
12
+ const fs = require('fs');
13
+ const path = require('path');
14
+ const { toPosixPath } = require('./core.cjs');
15
+ const { platformReadSync } = require('./shell-command-projection.cjs');
16
+ const { planningDir } = require('./planning-workspace.cjs');
17
+ const { extractFrontmatter } = require('./frontmatter.cjs');
18
+ const { requireSafePath, sanitizeForDisplay } = require('./security.cjs');
19
+
20
+ /**
21
+ * Scan .planning/debug/ for open sessions.
22
+ * Open = status NOT in ['resolved', 'complete'].
23
+ * Ignores the resolved/ subdirectory.
24
+ */
25
+ function scanDebugSessions(planDir) {
26
+ const debugDir = path.join(planDir, 'debug');
27
+ if (!fs.existsSync(debugDir)) return [];
28
+
29
+ const results = [];
30
+ let files;
31
+ try {
32
+ files = fs.readdirSync(debugDir, { withFileTypes: true });
33
+ } catch {
34
+ return [{ scan_error: true }];
35
+ }
36
+
37
+ for (const entry of files) {
38
+ if (!entry.isFile()) continue;
39
+ if (!entry.name.endsWith('.md')) continue;
40
+
41
+ const filePath = path.join(debugDir, entry.name);
42
+
43
+ let safeFilePath;
44
+ try {
45
+ safeFilePath = requireSafePath(filePath, planDir, 'debug session file', { allowAbsolute: true });
46
+ } catch {
47
+ continue;
48
+ }
49
+
50
+ const content = platformReadSync(safeFilePath);
51
+ if (content === null) continue;
52
+
53
+ const fm = extractFrontmatter(content);
54
+ const status = (fm.status || 'unknown').toLowerCase();
55
+ if (status === 'resolved' || status === 'complete') continue;
56
+
57
+ // Extract hypothesis from "Current Focus" block if parseable
58
+ let hypothesis = '';
59
+ const focusMatch = content.match(/##\s*Current Focus[^\n]*\n([\s\S]*?)(?=\n##\s|$)/i);
60
+ if (focusMatch) {
61
+ const focusText = focusMatch[1].trim().split('\n')[0].trim();
62
+ hypothesis = sanitizeForDisplay(focusText.slice(0, 100));
63
+ }
64
+
65
+ const slug = path.basename(entry.name, '.md');
66
+ results.push({
67
+ slug: sanitizeForDisplay(slug),
68
+ status: sanitizeForDisplay(status),
69
+ updated: sanitizeForDisplay(String(fm.updated || fm.date || '')),
70
+ hypothesis,
71
+ });
72
+ }
73
+
74
+ return results;
75
+ }
76
+
77
+ /**
78
+ * Scan .planning/quick/ for incomplete tasks.
79
+ * Incomplete if SUMMARY.md missing or status !== 'complete'.
80
+ */
81
+ function scanQuickTasks(planDir) {
82
+ const quickDir = path.join(planDir, 'quick');
83
+ if (!fs.existsSync(quickDir)) return [];
84
+
85
+ let entries;
86
+ try {
87
+ entries = fs.readdirSync(quickDir, { withFileTypes: true });
88
+ } catch {
89
+ return [{ scan_error: true }];
90
+ }
91
+
92
+ const results = [];
93
+ for (const entry of entries) {
94
+ if (!entry.isDirectory()) continue;
95
+
96
+ const dirName = entry.name;
97
+ const taskDir = path.join(quickDir, dirName);
98
+
99
+ let safeTaskDir;
100
+ try {
101
+ safeTaskDir = requireSafePath(taskDir, planDir, 'quick task dir', { allowAbsolute: true });
102
+ } catch {
103
+ continue;
104
+ }
105
+
106
+ // workflows/quick.md mandates `${quick_id}-SUMMARY.md`; older flows used
107
+ // bare `SUMMARY.md`. Accept either to avoid false-positive "missing".
108
+ let summaryPath = null;
109
+ try {
110
+ const summaryFiles = fs.readdirSync(safeTaskDir, { withFileTypes: true })
111
+ .filter(e => e.isFile() && (e.name === 'SUMMARY.md' || e.name.endsWith('-SUMMARY.md')));
112
+ if (summaryFiles.length > 0) {
113
+ // Prefer the per-task `${quick_id}-SUMMARY.md` form when present.
114
+ const preferred = summaryFiles.find(e => e.name === `${dirName}-SUMMARY.md`)
115
+ || summaryFiles.find(e => e.name.endsWith('-SUMMARY.md'))
116
+ || summaryFiles[0];
117
+ summaryPath = path.join(safeTaskDir, preferred.name);
118
+ }
119
+ } catch {
120
+ // fall through with summaryPath = null → status: missing
121
+ }
122
+
123
+ let status = 'missing';
124
+ const description = '';
125
+
126
+ if (summaryPath && fs.existsSync(summaryPath)) {
127
+ let safeSum;
128
+ try {
129
+ safeSum = requireSafePath(summaryPath, planDir, 'quick task summary', { allowAbsolute: true });
130
+ } catch {
131
+ continue;
132
+ }
133
+ const content = platformReadSync(safeSum);
134
+ if (content === null) {
135
+ status = 'unreadable';
136
+ } else {
137
+ const fm = extractFrontmatter(content);
138
+ status = (fm.status || 'unknown').toLowerCase();
139
+ }
140
+ }
141
+
142
+ if (status === 'complete') continue;
143
+
144
+ // Parse date and slug from directory name: YYYYMMDD-slug or YYYY-MM-DD-slug
145
+ let date = '';
146
+ let slug = sanitizeForDisplay(dirName);
147
+ const dateMatch = dirName.match(/^(\d{4}-?\d{2}-?\d{2})-(.+)$/);
148
+ if (dateMatch) {
149
+ date = dateMatch[1];
150
+ slug = sanitizeForDisplay(dateMatch[2]);
151
+ }
152
+
153
+ results.push({
154
+ slug,
155
+ date,
156
+ status: sanitizeForDisplay(status),
157
+ description,
158
+ });
159
+ }
160
+
161
+ return results;
162
+ }
163
+
164
+ /**
165
+ * Scan .planning/threads/ for open threads.
166
+ * Open if status in ['open', 'in_progress', 'in progress'] (case-insensitive).
167
+ */
168
+ function scanThreads(planDir) {
169
+ const threadsDir = path.join(planDir, 'threads');
170
+ if (!fs.existsSync(threadsDir)) return [];
171
+
172
+ let files;
173
+ try {
174
+ files = fs.readdirSync(threadsDir, { withFileTypes: true });
175
+ } catch {
176
+ return [{ scan_error: true }];
177
+ }
178
+
179
+ const openStatuses = new Set(['open', 'in_progress', 'in progress']);
180
+ const results = [];
181
+
182
+ for (const entry of files) {
183
+ if (!entry.isFile()) continue;
184
+ if (!entry.name.endsWith('.md')) continue;
185
+
186
+ const filePath = path.join(threadsDir, entry.name);
187
+
188
+ let safeFilePath;
189
+ try {
190
+ safeFilePath = requireSafePath(filePath, planDir, 'thread file', { allowAbsolute: true });
191
+ } catch {
192
+ continue;
193
+ }
194
+
195
+ const content = platformReadSync(safeFilePath);
196
+ if (content === null) continue;
197
+
198
+ const fm = extractFrontmatter(content);
199
+ let status = (fm.status || '').toLowerCase().trim();
200
+
201
+ // Fall back to scanning body for ## Status: OPEN / IN PROGRESS
202
+ if (!status) {
203
+ const bodyStatusMatch = content.match(/##\s*Status:\s*(OPEN|IN PROGRESS|IN_PROGRESS)/i);
204
+ if (bodyStatusMatch) {
205
+ status = bodyStatusMatch[1].toLowerCase().replace(/ /g, '_');
206
+ }
207
+ }
208
+
209
+ if (!openStatuses.has(status)) continue;
210
+
211
+ // Extract title from # Thread: heading or frontmatter title
212
+ let title = sanitizeForDisplay(String(fm.title || ''));
213
+ if (!title) {
214
+ const headingMatch = content.match(/^#\s*Thread:\s*(.+)$/m);
215
+ if (headingMatch) {
216
+ title = sanitizeForDisplay(headingMatch[1].trim().slice(0, 100));
217
+ }
218
+ }
219
+
220
+ const slug = path.basename(entry.name, '.md');
221
+ results.push({
222
+ slug: sanitizeForDisplay(slug),
223
+ status: sanitizeForDisplay(status),
224
+ updated: sanitizeForDisplay(String(fm.updated || fm.date || '')),
225
+ title,
226
+ });
227
+ }
228
+
229
+ return results;
230
+ }
231
+
232
+ /**
233
+ * Scan .planning/todos/pending/ for pending todos.
234
+ * Returns array of { filename, priority, area, summary }.
235
+ * Display limited to first 5 + count of remainder.
236
+ */
237
+ function scanTodos(planDir) {
238
+ const pendingDir = path.join(planDir, 'todos', 'pending');
239
+ if (!fs.existsSync(pendingDir)) return [];
240
+
241
+ let files;
242
+ try {
243
+ files = fs.readdirSync(pendingDir, { withFileTypes: true });
244
+ } catch {
245
+ return [{ scan_error: true }];
246
+ }
247
+
248
+ const mdFiles = files.filter(e => e.isFile() && e.name.endsWith('.md'));
249
+ const results = [];
250
+
251
+ const displayFiles = mdFiles.slice(0, 5);
252
+ for (const entry of displayFiles) {
253
+ const filePath = path.join(pendingDir, entry.name);
254
+
255
+ let safeFilePath;
256
+ try {
257
+ safeFilePath = requireSafePath(filePath, planDir, 'todo file', { allowAbsolute: true });
258
+ } catch {
259
+ continue;
260
+ }
261
+
262
+ const content = platformReadSync(safeFilePath);
263
+ if (content === null) continue;
264
+
265
+ const fm = extractFrontmatter(content);
266
+
267
+ // Extract first line of body after frontmatter
268
+ const bodyMatch = content.replace(/^---[\s\S]*?---\n?/, '');
269
+ const firstLine = bodyMatch.trim().split('\n')[0] || '';
270
+ const summary = sanitizeForDisplay(firstLine.slice(0, 100));
271
+
272
+ results.push({
273
+ filename: sanitizeForDisplay(entry.name),
274
+ priority: sanitizeForDisplay(String(fm.priority || '')),
275
+ area: sanitizeForDisplay(String(fm.area || '')),
276
+ summary,
277
+ });
278
+ }
279
+
280
+ if (mdFiles.length > 5) {
281
+ results.push({ _remainder_count: mdFiles.length - 5 });
282
+ }
283
+
284
+ return results;
285
+ }
286
+
287
+ /**
288
+ * Scan .planning/seeds/SEED-*.md for unimplemented seeds.
289
+ * Unimplemented if status in ['dormant', 'active', 'triggered'].
290
+ */
291
+ function scanSeeds(planDir) {
292
+ const seedsDir = path.join(planDir, 'seeds');
293
+ if (!fs.existsSync(seedsDir)) return [];
294
+
295
+ let files;
296
+ try {
297
+ files = fs.readdirSync(seedsDir, { withFileTypes: true });
298
+ } catch {
299
+ return [{ scan_error: true }];
300
+ }
301
+
302
+ const unimplementedStatuses = new Set(['dormant', 'active', 'triggered']);
303
+ const results = [];
304
+
305
+ for (const entry of files) {
306
+ if (!entry.isFile()) continue;
307
+ if (!entry.name.startsWith('SEED-') || !entry.name.endsWith('.md')) continue;
308
+
309
+ const filePath = path.join(seedsDir, entry.name);
310
+
311
+ let safeFilePath;
312
+ try {
313
+ safeFilePath = requireSafePath(filePath, planDir, 'seed file', { allowAbsolute: true });
314
+ } catch {
315
+ continue;
316
+ }
317
+
318
+ const content = platformReadSync(safeFilePath);
319
+ if (content === null) continue;
320
+
321
+ const fm = extractFrontmatter(content);
322
+ const status = (fm.status || 'dormant').toLowerCase();
323
+
324
+ if (!unimplementedStatuses.has(status)) continue;
325
+
326
+ // Extract seed_id from filename or frontmatter
327
+ const seedIdMatch = entry.name.match(/^(SEED-[\w-]+)\.md$/);
328
+ const seed_id = seedIdMatch ? seedIdMatch[1] : path.basename(entry.name, '.md');
329
+ const slug = sanitizeForDisplay(seed_id.replace(/^SEED-/, ''));
330
+
331
+ let title = sanitizeForDisplay(String(fm.title || ''));
332
+ if (!title) {
333
+ const headingMatch = content.match(/^#\s*(.+)$/m);
334
+ if (headingMatch) title = sanitizeForDisplay(headingMatch[1].trim().slice(0, 100));
335
+ }
336
+
337
+ results.push({
338
+ seed_id: sanitizeForDisplay(seed_id),
339
+ slug,
340
+ status: sanitizeForDisplay(status),
341
+ title,
342
+ });
343
+ }
344
+
345
+ return results;
346
+ }
347
+
348
+ // Terminal UAT states: `complete` (legacy) and `resolved` (post-gap-closure
349
+ // per workflows/execute-phase.md). Hoisted outside scanUatGaps so the Set is
350
+ // not recreated on each loop iteration.
351
+ const TERMINAL_UAT_STATUSES = new Set(['complete', 'resolved']);
352
+
353
+ /**
354
+ * Scan .planning/phases for UAT gaps (UAT files with status != 'complete').
355
+ */
356
+ function scanUatGaps(planDir) {
357
+ const phasesDir = path.join(planDir, 'phases');
358
+ if (!fs.existsSync(phasesDir)) return [];
359
+
360
+ let dirs;
361
+ try {
362
+ dirs = fs.readdirSync(phasesDir, { withFileTypes: true })
363
+ .filter(e => e.isDirectory())
364
+ .map(e => e.name)
365
+ .sort();
366
+ } catch {
367
+ return [{ scan_error: true }];
368
+ }
369
+
370
+ const results = [];
371
+
372
+ for (const dir of dirs) {
373
+ const phaseDir = path.join(phasesDir, dir);
374
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
375
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
376
+
377
+ let files;
378
+ try {
379
+ files = fs.readdirSync(phaseDir);
380
+ } catch {
381
+ continue;
382
+ }
383
+
384
+ for (const file of files.filter(f => f.includes('-UAT') && f.endsWith('.md'))) {
385
+ const filePath = path.join(phaseDir, file);
386
+
387
+ let safeFilePath;
388
+ try {
389
+ safeFilePath = requireSafePath(filePath, planDir, 'UAT file', { allowAbsolute: true });
390
+ } catch {
391
+ continue;
392
+ }
393
+
394
+ const content = platformReadSync(safeFilePath);
395
+ if (content === null) continue;
396
+
397
+ const fm = extractFrontmatter(content);
398
+ const status = (fm.status || 'unknown').toLowerCase();
399
+ const result = (fm.result || '').toString().toLowerCase();
400
+
401
+ // Also accept `result: all_pass` as a fallback when status is absent
402
+ // — covers UATs that omit `status:`.
403
+ if (TERMINAL_UAT_STATUSES.has(status)) continue;
404
+ if (status === 'unknown' && result === 'all_pass') continue;
405
+
406
+ // Count open scenarios
407
+ const pendingMatches = (content.match(/result:\s*(?:pending|\[pending\])/gi) || []).length;
408
+
409
+ results.push({
410
+ phase: sanitizeForDisplay(phaseNum),
411
+ file: sanitizeForDisplay(file),
412
+ status: sanitizeForDisplay(status),
413
+ open_scenario_count: pendingMatches,
414
+ });
415
+ }
416
+ }
417
+
418
+ return results;
419
+ }
420
+
421
+ /**
422
+ * Scan .planning/phases for VERIFICATION gaps.
423
+ */
424
+ function scanVerificationGaps(planDir) {
425
+ const phasesDir = path.join(planDir, 'phases');
426
+ if (!fs.existsSync(phasesDir)) return [];
427
+
428
+ let dirs;
429
+ try {
430
+ dirs = fs.readdirSync(phasesDir, { withFileTypes: true })
431
+ .filter(e => e.isDirectory())
432
+ .map(e => e.name)
433
+ .sort();
434
+ } catch {
435
+ return [{ scan_error: true }];
436
+ }
437
+
438
+ const results = [];
439
+
440
+ for (const dir of dirs) {
441
+ const phaseDir = path.join(phasesDir, dir);
442
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
443
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
444
+
445
+ let files;
446
+ try {
447
+ files = fs.readdirSync(phaseDir);
448
+ } catch {
449
+ continue;
450
+ }
451
+
452
+ for (const file of files.filter(f => f.includes('-VERIFICATION') && f.endsWith('.md'))) {
453
+ const filePath = path.join(phaseDir, file);
454
+
455
+ let safeFilePath;
456
+ try {
457
+ safeFilePath = requireSafePath(filePath, planDir, 'VERIFICATION file', { allowAbsolute: true });
458
+ } catch {
459
+ continue;
460
+ }
461
+
462
+ const content = platformReadSync(safeFilePath);
463
+ if (content === null) continue;
464
+
465
+ const fm = extractFrontmatter(content);
466
+ const status = (fm.status || 'unknown').toLowerCase();
467
+
468
+ if (status !== 'gaps_found' && status !== 'human_needed') continue;
469
+
470
+ results.push({
471
+ phase: sanitizeForDisplay(phaseNum),
472
+ file: sanitizeForDisplay(file),
473
+ status: sanitizeForDisplay(status),
474
+ });
475
+ }
476
+ }
477
+
478
+ return results;
479
+ }
480
+
481
+ /**
482
+ * Scan .planning/phases for CONTEXT files with open_questions.
483
+ */
484
+ function scanContextQuestions(planDir) {
485
+ const phasesDir = path.join(planDir, 'phases');
486
+ if (!fs.existsSync(phasesDir)) return [];
487
+
488
+ let dirs;
489
+ try {
490
+ dirs = fs.readdirSync(phasesDir, { withFileTypes: true })
491
+ .filter(e => e.isDirectory())
492
+ .map(e => e.name)
493
+ .sort();
494
+ } catch {
495
+ return [{ scan_error: true }];
496
+ }
497
+
498
+ const results = [];
499
+
500
+ for (const dir of dirs) {
501
+ const phaseDir = path.join(phasesDir, dir);
502
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
503
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
504
+
505
+ let files;
506
+ try {
507
+ files = fs.readdirSync(phaseDir);
508
+ } catch {
509
+ continue;
510
+ }
511
+
512
+ for (const file of files.filter(f => f.includes('-CONTEXT') && f.endsWith('.md'))) {
513
+ const filePath = path.join(phaseDir, file);
514
+
515
+ let safeFilePath;
516
+ try {
517
+ safeFilePath = requireSafePath(filePath, planDir, 'CONTEXT file', { allowAbsolute: true });
518
+ } catch {
519
+ continue;
520
+ }
521
+
522
+ const content = platformReadSync(safeFilePath);
523
+ if (content === null) continue;
524
+
525
+ const fm = extractFrontmatter(content);
526
+
527
+ // Check frontmatter open_questions field
528
+ let questions = [];
529
+ if (fm.open_questions) {
530
+ if (Array.isArray(fm.open_questions) && fm.open_questions.length > 0) {
531
+ questions = fm.open_questions.map(q => sanitizeForDisplay(String(q).slice(0, 200)));
532
+ }
533
+ }
534
+
535
+ // Also check for ## Open Questions section in body
536
+ if (questions.length === 0) {
537
+ const oqMatch = content.match(/##\s*Open Questions[^\n]*\n([\s\S]*?)(?=\n##\s|$)/i);
538
+ if (oqMatch) {
539
+ const oqBody = oqMatch[1].trim();
540
+ if (oqBody && oqBody.length > 0 && !/^\s*none\s*$/i.test(oqBody)) {
541
+ const items = oqBody.split('\n')
542
+ .map(l => l.trim())
543
+ .filter(l => l && l !== '-' && l !== '*')
544
+ .filter(l => /^[-*\d]/.test(l) || l.includes('?'));
545
+ questions = items.slice(0, 3).map(q => sanitizeForDisplay(q.slice(0, 200)));
546
+ }
547
+ }
548
+ }
549
+
550
+ if (questions.length === 0) continue;
551
+
552
+ results.push({
553
+ phase: sanitizeForDisplay(phaseNum),
554
+ file: sanitizeForDisplay(file),
555
+ question_count: questions.length,
556
+ questions: questions.slice(0, 3),
557
+ });
558
+ }
559
+ }
560
+
561
+ return results;
562
+ }
563
+
564
+ /**
565
+ * Main audit function. Scans all .planning/ artifact categories.
566
+ *
567
+ * @param {string} cwd - Project root directory
568
+ * @returns {object} Structured audit result
569
+ */
570
+ function auditOpenArtifacts(cwd) {
571
+ const planDir = planningDir(cwd);
572
+
573
+ const debugSessions = (() => {
574
+ try { return scanDebugSessions(planDir); } catch { return [{ scan_error: true }]; }
575
+ })();
576
+
577
+ const quickTasks = (() => {
578
+ try { return scanQuickTasks(planDir); } catch { return [{ scan_error: true }]; }
579
+ })();
580
+
581
+ const threads = (() => {
582
+ try { return scanThreads(planDir); } catch { return [{ scan_error: true }]; }
583
+ })();
584
+
585
+ const todos = (() => {
586
+ try { return scanTodos(planDir); } catch { return [{ scan_error: true }]; }
587
+ })();
588
+
589
+ const seeds = (() => {
590
+ try { return scanSeeds(planDir); } catch { return [{ scan_error: true }]; }
591
+ })();
592
+
593
+ const uatGaps = (() => {
594
+ try { return scanUatGaps(planDir); } catch { return [{ scan_error: true }]; }
595
+ })();
596
+
597
+ const verificationGaps = (() => {
598
+ try { return scanVerificationGaps(planDir); } catch { return [{ scan_error: true }]; }
599
+ })();
600
+
601
+ const contextQuestions = (() => {
602
+ try { return scanContextQuestions(planDir); } catch { return [{ scan_error: true }]; }
603
+ })();
604
+
605
+ // Count real items (not scan_error sentinels)
606
+ const countReal = arr => arr.filter(i => !i.scan_error && !i._remainder_count).length;
607
+
608
+ const counts = {
609
+ debug_sessions: countReal(debugSessions),
610
+ quick_tasks: countReal(quickTasks),
611
+ threads: countReal(threads),
612
+ todos: countReal(todos),
613
+ seeds: countReal(seeds),
614
+ uat_gaps: countReal(uatGaps),
615
+ verification_gaps: countReal(verificationGaps),
616
+ context_questions: countReal(contextQuestions),
617
+ };
618
+ counts.total = Object.values(counts).reduce((s, n) => s + n, 0);
619
+
620
+ return {
621
+ scanned_at: new Date().toISOString(),
622
+ has_open_items: counts.total > 0,
623
+ counts,
624
+ items: {
625
+ debug_sessions: debugSessions,
626
+ quick_tasks: quickTasks,
627
+ threads,
628
+ todos,
629
+ seeds,
630
+ uat_gaps: uatGaps,
631
+ verification_gaps: verificationGaps,
632
+ context_questions: contextQuestions,
633
+ },
634
+ };
635
+ }
636
+
637
+ /**
638
+ * Format the audit result as a human-readable report.
639
+ *
640
+ * @param {object} auditResult - Result from auditOpenArtifacts()
641
+ * @returns {string} Formatted report
642
+ */
643
+ function formatAuditReport(auditResult) {
644
+ const { counts, items, has_open_items } = auditResult;
645
+ const lines = [];
646
+ const hr = '━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━';
647
+
648
+ lines.push(hr);
649
+ lines.push(' Milestone Close: Open Artifact Audit');
650
+ lines.push(hr);
651
+
652
+ if (!has_open_items) {
653
+ lines.push('');
654
+ lines.push(' All artifact types clear. Safe to proceed.');
655
+ lines.push('');
656
+ lines.push(hr);
657
+ return lines.join('\n');
658
+ }
659
+
660
+ // Debug sessions (blocking quality — red)
661
+ if (counts.debug_sessions > 0) {
662
+ lines.push('');
663
+ lines.push(`🔴 Debug Sessions (${counts.debug_sessions} open)`);
664
+ for (const item of items.debug_sessions.filter(i => !i.scan_error)) {
665
+ const hyp = item.hypothesis ? ` — ${item.hypothesis}` : '';
666
+ lines.push(` • ${item.slug} [${item.status}]${hyp}`);
667
+ }
668
+ }
669
+
670
+ // UAT gaps (blocking quality — red)
671
+ if (counts.uat_gaps > 0) {
672
+ lines.push('');
673
+ lines.push(`🔴 UAT Gaps (${counts.uat_gaps} phases with incomplete UAT)`);
674
+ for (const item of items.uat_gaps.filter(i => !i.scan_error)) {
675
+ lines.push(` • Phase ${item.phase}: ${item.file} [${item.status}] — ${item.open_scenario_count} pending scenarios`);
676
+ }
677
+ }
678
+
679
+ // Verification gaps (blocking quality — red)
680
+ if (counts.verification_gaps > 0) {
681
+ lines.push('');
682
+ lines.push(`🔴 Verification Gaps (${counts.verification_gaps} unresolved)`);
683
+ for (const item of items.verification_gaps.filter(i => !i.scan_error)) {
684
+ lines.push(` • Phase ${item.phase}: ${item.file} [${item.status}]`);
685
+ }
686
+ }
687
+
688
+ // Quick tasks (incomplete work — yellow)
689
+ if (counts.quick_tasks > 0) {
690
+ lines.push('');
691
+ lines.push(`🟡 Quick Tasks (${counts.quick_tasks} incomplete)`);
692
+ for (const item of items.quick_tasks.filter(i => !i.scan_error)) {
693
+ const d = item.date ? ` (${item.date})` : '';
694
+ lines.push(` • ${item.slug}${d} [${item.status}]`);
695
+ }
696
+ }
697
+
698
+ // Todos (incomplete work — yellow)
699
+ if (counts.todos > 0) {
700
+ const realTodos = items.todos.filter(i => !i.scan_error && !i._remainder_count);
701
+ const remainder = items.todos.find(i => i._remainder_count);
702
+ lines.push('');
703
+ lines.push(`🟡 Pending Todos (${counts.todos} pending)`);
704
+ for (const item of realTodos) {
705
+ const area = item.area ? ` [${item.area}]` : '';
706
+ const pri = item.priority ? ` (${item.priority})` : '';
707
+ lines.push(` • ${item.filename}${area}${pri}`);
708
+ if (item.summary) lines.push(` ${item.summary}`);
709
+ }
710
+ if (remainder) {
711
+ lines.push(` ... and ${remainder._remainder_count} more`);
712
+ }
713
+ }
714
+
715
+ // Threads (deferred decisions — blue)
716
+ if (counts.threads > 0) {
717
+ lines.push('');
718
+ lines.push(`🔵 Open Threads (${counts.threads} active)`);
719
+ for (const item of items.threads.filter(i => !i.scan_error)) {
720
+ const title = item.title ? ` — ${item.title}` : '';
721
+ lines.push(` • ${item.slug} [${item.status}]${title}`);
722
+ }
723
+ }
724
+
725
+ // Seeds (deferred decisions — blue)
726
+ if (counts.seeds > 0) {
727
+ lines.push('');
728
+ lines.push(`🔵 Unimplemented Seeds (${counts.seeds} pending)`);
729
+ for (const item of items.seeds.filter(i => !i.scan_error)) {
730
+ const title = item.title ? ` — ${item.title}` : '';
731
+ lines.push(` • ${item.seed_id} [${item.status}]${title}`);
732
+ }
733
+ }
734
+
735
+ // Context questions (deferred decisions — blue)
736
+ if (counts.context_questions > 0) {
737
+ lines.push('');
738
+ lines.push(`🔵 CONTEXT Open Questions (${counts.context_questions} phases with open questions)`);
739
+ for (const item of items.context_questions.filter(i => !i.scan_error)) {
740
+ lines.push(` • Phase ${item.phase}: ${item.file} (${item.question_count} question${item.question_count !== 1 ? 's' : ''})`);
741
+ for (const q of item.questions) {
742
+ lines.push(` - ${q}`);
743
+ }
744
+ }
745
+ }
746
+
747
+ lines.push('');
748
+ lines.push(hr);
749
+ lines.push(` ${counts.total} item${counts.total !== 1 ? 's' : ''} require decisions before close.`);
750
+ lines.push(hr);
751
+
752
+ return lines.join('\n');
753
+ }
754
+
755
+ module.exports = { auditOpenArtifacts, formatAuditReport };