@therocketcode/gsd-core 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (568) hide show
  1. package/.claude-plugin/plugin.json +23 -0
  2. package/GEMINI.md +53 -0
  3. package/LICENSE +21 -0
  4. package/README.ja-JP.md +125 -0
  5. package/README.ko-KR.md +125 -0
  6. package/README.md +144 -0
  7. package/README.pt-BR.md +125 -0
  8. package/README.zh-CN.md +125 -0
  9. package/agents/gsd-advisor-researcher.md +108 -0
  10. package/agents/gsd-ai-researcher.md +114 -0
  11. package/agents/gsd-assumptions-analyzer.md +105 -0
  12. package/agents/gsd-code-fixer.md +668 -0
  13. package/agents/gsd-code-reviewer.md +387 -0
  14. package/agents/gsd-codebase-mapper.md +853 -0
  15. package/agents/gsd-debug-session-manager.md +314 -0
  16. package/agents/gsd-debugger.md +1452 -0
  17. package/agents/gsd-doc-classifier.md +168 -0
  18. package/agents/gsd-doc-synthesizer.md +204 -0
  19. package/agents/gsd-doc-verifier.md +217 -0
  20. package/agents/gsd-doc-writer.md +616 -0
  21. package/agents/gsd-domain-researcher.md +147 -0
  22. package/agents/gsd-eval-auditor.md +191 -0
  23. package/agents/gsd-eval-planner.md +154 -0
  24. package/agents/gsd-executor.md +785 -0
  25. package/agents/gsd-framework-selector.md +160 -0
  26. package/agents/gsd-integration-checker.md +470 -0
  27. package/agents/gsd-intel-updater.md +342 -0
  28. package/agents/gsd-nyquist-auditor.md +203 -0
  29. package/agents/gsd-pattern-mapper.md +335 -0
  30. package/agents/gsd-phase-researcher.md +867 -0
  31. package/agents/gsd-plan-checker.md +978 -0
  32. package/agents/gsd-planner.md +1204 -0
  33. package/agents/gsd-project-researcher.md +611 -0
  34. package/agents/gsd-research-synthesizer.md +259 -0
  35. package/agents/gsd-roadmapper.md +688 -0
  36. package/agents/gsd-security-auditor.md +155 -0
  37. package/agents/gsd-ui-auditor.md +495 -0
  38. package/agents/gsd-ui-checker.md +309 -0
  39. package/agents/gsd-ui-researcher.md +374 -0
  40. package/agents/gsd-user-profiler.md +171 -0
  41. package/agents/gsd-verifier.md +923 -0
  42. package/assets/gsd-logo-2000-transparent.png +0 -0
  43. package/assets/gsd-logo-2000-transparent.svg +17 -0
  44. package/assets/gsd-logo-2000.png +0 -0
  45. package/assets/gsd-logo-2000.svg +21 -0
  46. package/assets/terminal.svg +68 -0
  47. package/bin/install.js +12726 -0
  48. package/bin/lib/ui-safety-gate.cjs +107 -0
  49. package/commands/gsd/add-tests.md +42 -0
  50. package/commands/gsd/ai-integration-phase.md +37 -0
  51. package/commands/gsd/audit-fix.md +34 -0
  52. package/commands/gsd/audit-milestone.md +37 -0
  53. package/commands/gsd/audit-uat.md +24 -0
  54. package/commands/gsd/autonomous.md +48 -0
  55. package/commands/gsd/capture.md +62 -0
  56. package/commands/gsd/cleanup.md +24 -0
  57. package/commands/gsd/code-review.md +59 -0
  58. package/commands/gsd/complete-milestone.md +143 -0
  59. package/commands/gsd/config.md +56 -0
  60. package/commands/gsd/debug.md +52 -0
  61. package/commands/gsd/discover-product.md +65 -0
  62. package/commands/gsd/discuss-phase.md +77 -0
  63. package/commands/gsd/docs-update.md +49 -0
  64. package/commands/gsd/eval-review.md +33 -0
  65. package/commands/gsd/execute-phase.md +66 -0
  66. package/commands/gsd/explore.md +27 -0
  67. package/commands/gsd/extract-learnings.md +23 -0
  68. package/commands/gsd/fast.md +31 -0
  69. package/commands/gsd/forensics.md +57 -0
  70. package/commands/gsd/graphify.md +204 -0
  71. package/commands/gsd/health.md +31 -0
  72. package/commands/gsd/help.md +28 -0
  73. package/commands/gsd/import.md +45 -0
  74. package/commands/gsd/inbox.md +39 -0
  75. package/commands/gsd/ingest-docs.md +42 -0
  76. package/commands/gsd/manager.md +45 -0
  77. package/commands/gsd/map-codebase.md +83 -0
  78. package/commands/gsd/milestone-summary.md +51 -0
  79. package/commands/gsd/model-domain.md +65 -0
  80. package/commands/gsd/mvp-phase.md +45 -0
  81. package/commands/gsd/new-milestone.md +45 -0
  82. package/commands/gsd/new-project.md +47 -0
  83. package/commands/gsd/ns-context.md +23 -0
  84. package/commands/gsd/ns-ideate.md +24 -0
  85. package/commands/gsd/ns-manage.md +29 -0
  86. package/commands/gsd/ns-project.md +22 -0
  87. package/commands/gsd/ns-review.md +26 -0
  88. package/commands/gsd/ns-workflow.md +28 -0
  89. package/commands/gsd/pause-work.md +43 -0
  90. package/commands/gsd/phase.md +56 -0
  91. package/commands/gsd/plan-phase.md +64 -0
  92. package/commands/gsd/plan-review-convergence.md +59 -0
  93. package/commands/gsd/pr-branch.md +26 -0
  94. package/commands/gsd/profile-user.md +46 -0
  95. package/commands/gsd/progress.md +48 -0
  96. package/commands/gsd/quick.md +174 -0
  97. package/commands/gsd/recommend-architecture.md +64 -0
  98. package/commands/gsd/resume-work.md +30 -0
  99. package/commands/gsd/review-backlog.md +63 -0
  100. package/commands/gsd/review.md +42 -0
  101. package/commands/gsd/secure-phase.md +36 -0
  102. package/commands/gsd/settings.md +29 -0
  103. package/commands/gsd/ship.md +24 -0
  104. package/commands/gsd/sketch.md +60 -0
  105. package/commands/gsd/spec-phase.md +63 -0
  106. package/commands/gsd/spike.md +57 -0
  107. package/commands/gsd/stats.md +20 -0
  108. package/commands/gsd/surface.md +155 -0
  109. package/commands/gsd/testing-strategy.md +65 -0
  110. package/commands/gsd/thread.md +24 -0
  111. package/commands/gsd/ui-phase.md +35 -0
  112. package/commands/gsd/ui-review.md +33 -0
  113. package/commands/gsd/ultraplan-phase.md +34 -0
  114. package/commands/gsd/undo.md +35 -0
  115. package/commands/gsd/update.md +49 -0
  116. package/commands/gsd/validate-phase.md +36 -0
  117. package/commands/gsd/verify-work.md +39 -0
  118. package/commands/gsd/workspace.md +52 -0
  119. package/commands/gsd/workstreams.md +70 -0
  120. package/gemini-extension.json +6 -0
  121. package/gsd-core/bin/check-latest-version.cjs +161 -0
  122. package/gsd-core/bin/gsd-tools.cjs +1928 -0
  123. package/gsd-core/bin/lib/active-workstream-store.cjs +291 -0
  124. package/gsd-core/bin/lib/adr-parser.cjs +399 -0
  125. package/gsd-core/bin/lib/agent-command-router.cjs +68 -0
  126. package/gsd-core/bin/lib/artifacts.cjs +51 -0
  127. package/gsd-core/bin/lib/audit.cjs +743 -0
  128. package/gsd-core/bin/lib/check-command-router.cjs +343 -0
  129. package/gsd-core/bin/lib/cjs-command-router-adapter.cjs +81 -0
  130. package/gsd-core/bin/lib/cli-exit.cjs +42 -0
  131. package/gsd-core/bin/lib/clock.cjs +95 -0
  132. package/gsd-core/bin/lib/clusters.cjs +132 -0
  133. package/gsd-core/bin/lib/code-review-flags.cjs +59 -0
  134. package/gsd-core/bin/lib/command-aliases.cjs +809 -0
  135. package/gsd-core/bin/lib/command-arg-projection.cjs +55 -0
  136. package/gsd-core/bin/lib/command-routing-hub.cjs +300 -0
  137. package/gsd-core/bin/lib/commands.cjs +1203 -0
  138. package/gsd-core/bin/lib/config-schema.cjs +29 -0
  139. package/gsd-core/bin/lib/config-types.cjs +19 -0
  140. package/gsd-core/bin/lib/config.cjs +738 -0
  141. package/gsd-core/bin/lib/configuration.cjs +239 -0
  142. package/gsd-core/bin/lib/context-utilization.cjs +48 -0
  143. package/gsd-core/bin/lib/core.cjs +2051 -0
  144. package/gsd-core/bin/lib/decisions.cjs +118 -0
  145. package/gsd-core/bin/lib/docs.cjs +252 -0
  146. package/gsd-core/bin/lib/drift.cjs +364 -0
  147. package/gsd-core/bin/lib/fallow-runner.cjs +115 -0
  148. package/gsd-core/bin/lib/frontmatter.cjs +442 -0
  149. package/gsd-core/bin/lib/gap-checker.cjs +257 -0
  150. package/gsd-core/bin/lib/graphify.cjs +496 -0
  151. package/gsd-core/bin/lib/gsd2-import.cjs +456 -0
  152. package/gsd-core/bin/lib/init-command-router.cjs +62 -0
  153. package/gsd-core/bin/lib/init.cjs +1815 -0
  154. package/gsd-core/bin/lib/install-profiles.cjs +584 -0
  155. package/gsd-core/bin/lib/installer-migration-authoring.cjs +122 -0
  156. package/gsd-core/bin/lib/installer-migration-report.cjs +350 -0
  157. package/gsd-core/bin/lib/installer-migrations/000-first-time-baseline.cjs +218 -0
  158. package/gsd-core/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +48 -0
  159. package/gsd-core/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +94 -0
  160. package/gsd-core/bin/lib/installer-migrations/003-rename-get-shit-done-to-gsd-core.cjs +108 -0
  161. package/gsd-core/bin/lib/installer-migrations.cjs +823 -0
  162. package/gsd-core/bin/lib/intel.cjs +590 -0
  163. package/gsd-core/bin/lib/learnings.cjs +270 -0
  164. package/gsd-core/bin/lib/legacy-cleanup.cjs +253 -0
  165. package/gsd-core/bin/lib/milestone.cjs +373 -0
  166. package/gsd-core/bin/lib/model-catalog.cjs +154 -0
  167. package/gsd-core/bin/lib/model-profiles.cjs +24 -0
  168. package/gsd-core/bin/lib/observability/event.cjs +51 -0
  169. package/gsd-core/bin/lib/observability/logger.cjs +146 -0
  170. package/gsd-core/bin/lib/observability/redaction.cjs +48 -0
  171. package/gsd-core/bin/lib/package-identity.cjs +35 -0
  172. package/gsd-core/bin/lib/package-legitimacy.cjs +368 -0
  173. package/gsd-core/bin/lib/phase-command-router.cjs +189 -0
  174. package/gsd-core/bin/lib/phase-lifecycle.cjs +74 -0
  175. package/gsd-core/bin/lib/phase.cjs +1307 -0
  176. package/gsd-core/bin/lib/phases-command-router.cjs +43 -0
  177. package/gsd-core/bin/lib/plan-scan.cjs +91 -0
  178. package/gsd-core/bin/lib/planning-workspace.cjs +245 -0
  179. package/gsd-core/bin/lib/profile-output.cjs +1120 -0
  180. package/gsd-core/bin/lib/profile-pipeline.cjs +517 -0
  181. package/gsd-core/bin/lib/project-root.cjs +119 -0
  182. package/gsd-core/bin/lib/prompt-budget.cjs +305 -0
  183. package/gsd-core/bin/lib/research-provider.cjs +137 -0
  184. package/gsd-core/bin/lib/research-store.cjs +167 -0
  185. package/gsd-core/bin/lib/review-reviewer-selection.cjs +121 -0
  186. package/gsd-core/bin/lib/roadmap-command-router.cjs +166 -0
  187. package/gsd-core/bin/lib/roadmap-upgrade.cjs +476 -0
  188. package/gsd-core/bin/lib/roadmap.cjs +600 -0
  189. package/gsd-core/bin/lib/runtime-artifact-layout.cjs +312 -0
  190. package/gsd-core/bin/lib/runtime-config-adapter-registry.cjs +56 -0
  191. package/gsd-core/bin/lib/runtime-homes.cjs +190 -0
  192. package/gsd-core/bin/lib/runtime-name-policy.cjs +96 -0
  193. package/gsd-core/bin/lib/runtime-slash.cjs +119 -0
  194. package/gsd-core/bin/lib/schema-detect.cjs +159 -0
  195. package/gsd-core/bin/lib/secrets.cjs +34 -0
  196. package/gsd-core/bin/lib/security.cjs +480 -0
  197. package/gsd-core/bin/lib/semver-compare.cjs +42 -0
  198. package/gsd-core/bin/lib/shell-command-projection.cjs +533 -0
  199. package/gsd-core/bin/lib/state-command-router.cjs +160 -0
  200. package/gsd-core/bin/lib/state-document.cjs +259 -0
  201. package/gsd-core/bin/lib/state.cjs +2010 -0
  202. package/gsd-core/bin/lib/surface.cjs +449 -0
  203. package/gsd-core/bin/lib/task-command-router.cjs +85 -0
  204. package/gsd-core/bin/lib/template.cjs +237 -0
  205. package/gsd-core/bin/lib/uat.cjs +297 -0
  206. package/gsd-core/bin/lib/ui-safety-gate.cjs +98 -0
  207. package/gsd-core/bin/lib/update-context.cjs +218 -0
  208. package/gsd-core/bin/lib/validate-command-router.cjs +91 -0
  209. package/gsd-core/bin/lib/validate.cjs +112 -0
  210. package/gsd-core/bin/lib/verification-command-router.cjs +31 -0
  211. package/gsd-core/bin/lib/verification.cjs +193 -0
  212. package/gsd-core/bin/lib/verify-command-router.cjs +44 -0
  213. package/gsd-core/bin/lib/verify.cjs +1451 -0
  214. package/gsd-core/bin/lib/workstream-inventory-builder.cjs +81 -0
  215. package/gsd-core/bin/lib/workstream-inventory.cjs +147 -0
  216. package/gsd-core/bin/lib/workstream-name-policy.cjs +91 -0
  217. package/gsd-core/bin/lib/workstream.cjs +380 -0
  218. package/gsd-core/bin/lib/worktree-base-ref.cjs +325 -0
  219. package/gsd-core/bin/lib/worktree-safety.cjs +943 -0
  220. package/gsd-core/bin/shared/config-defaults.manifest.json +98 -0
  221. package/gsd-core/bin/shared/config-schema.manifest.json +192 -0
  222. package/gsd-core/bin/shared/model-catalog.json +149 -0
  223. package/gsd-core/bin/shared/runtime-aliases.manifest.json +75 -0
  224. package/gsd-core/bin/verify-reapply-patches.cjs +349 -0
  225. package/gsd-core/contexts/dev.md +21 -0
  226. package/gsd-core/contexts/research.md +22 -0
  227. package/gsd-core/contexts/review.md +23 -0
  228. package/gsd-core/references/agent-contracts.md +79 -0
  229. package/gsd-core/references/ai-evals.md +156 -0
  230. package/gsd-core/references/ai-frameworks.md +186 -0
  231. package/gsd-core/references/architecture-decision.md +74 -0
  232. package/gsd-core/references/artifact-types.md +131 -0
  233. package/gsd-core/references/auth-in-tests.md +91 -0
  234. package/gsd-core/references/autonomous-smart-discuss.md +277 -0
  235. package/gsd-core/references/checkpoints.md +814 -0
  236. package/gsd-core/references/common-bug-patterns.md +114 -0
  237. package/gsd-core/references/context-budget.md +85 -0
  238. package/gsd-core/references/continuation-format.md +253 -0
  239. package/gsd-core/references/db-test-isolation.md +54 -0
  240. package/gsd-core/references/debugger-philosophy.md +76 -0
  241. package/gsd-core/references/decimal-phase-calculation.md +64 -0
  242. package/gsd-core/references/doc-conflict-engine.md +91 -0
  243. package/gsd-core/references/domain-modeling.md +80 -0
  244. package/gsd-core/references/domain-probes.md +125 -0
  245. package/gsd-core/references/e2e-tiering.md +35 -0
  246. package/gsd-core/references/execute-mvp-tdd.md +81 -0
  247. package/gsd-core/references/executor-examples.md +110 -0
  248. package/gsd-core/references/few-shot-examples/plan-checker.md +73 -0
  249. package/gsd-core/references/few-shot-examples/verifier.md +109 -0
  250. package/gsd-core/references/flaky-test-checklist.md +22 -0
  251. package/gsd-core/references/gate-prompts.md +100 -0
  252. package/gsd-core/references/gates.md +70 -0
  253. package/gsd-core/references/git-integration.md +298 -0
  254. package/gsd-core/references/git-planning-commit.md +40 -0
  255. package/gsd-core/references/ios-scaffold.md +123 -0
  256. package/gsd-core/references/mandatory-initial-read.md +2 -0
  257. package/gsd-core/references/model-profile-resolution.md +38 -0
  258. package/gsd-core/references/model-profiles.md +245 -0
  259. package/gsd-core/references/mvp-concepts.md +49 -0
  260. package/gsd-core/references/phase-argument-parsing.md +61 -0
  261. package/gsd-core/references/planner-antipatterns.md +89 -0
  262. package/gsd-core/references/planner-chunked.md +49 -0
  263. package/gsd-core/references/planner-gap-closure.md +62 -0
  264. package/gsd-core/references/planner-graphify-auto-update.md +67 -0
  265. package/gsd-core/references/planner-human-verify-mode.md +57 -0
  266. package/gsd-core/references/planner-interface-context.md +62 -0
  267. package/gsd-core/references/planner-load-graph-context.md +36 -0
  268. package/gsd-core/references/planner-mvp-mode.md +53 -0
  269. package/gsd-core/references/planner-reviews.md +39 -0
  270. package/gsd-core/references/planner-revision.md +87 -0
  271. package/gsd-core/references/planner-source-audit.md +73 -0
  272. package/gsd-core/references/planning-config.md +473 -0
  273. package/gsd-core/references/product-discovery.md +49 -0
  274. package/gsd-core/references/project-skills-discovery.md +19 -0
  275. package/gsd-core/references/questioning.md +162 -0
  276. package/gsd-core/references/realistic-test-data.md +44 -0
  277. package/gsd-core/references/research-documentation-lookup.md +29 -0
  278. package/gsd-core/references/research-philosophy.md +29 -0
  279. package/gsd-core/references/research-verification-protocol.md +27 -0
  280. package/gsd-core/references/revision-loop.md +97 -0
  281. package/gsd-core/references/scout-codebase.md +51 -0
  282. package/gsd-core/references/skeleton-template.md +48 -0
  283. package/gsd-core/references/sketch-interactivity.md +41 -0
  284. package/gsd-core/references/sketch-theme-system.md +94 -0
  285. package/gsd-core/references/sketch-tooling.md +45 -0
  286. package/gsd-core/references/sketch-variant-patterns.md +81 -0
  287. package/gsd-core/references/spidr-splitting.md +69 -0
  288. package/gsd-core/references/tdd.md +330 -0
  289. package/gsd-core/references/test-containers.md +55 -0
  290. package/gsd-core/references/test-strategy.md +75 -0
  291. package/gsd-core/references/thinking-models-debug.md +44 -0
  292. package/gsd-core/references/thinking-models-execution.md +50 -0
  293. package/gsd-core/references/thinking-models-planning.md +62 -0
  294. package/gsd-core/references/thinking-models-research.md +50 -0
  295. package/gsd-core/references/thinking-models-verification.md +55 -0
  296. package/gsd-core/references/thinking-partner.md +96 -0
  297. package/gsd-core/references/ui-brand.md +162 -0
  298. package/gsd-core/references/universal-anti-patterns.md +63 -0
  299. package/gsd-core/references/user-profiling.md +681 -0
  300. package/gsd-core/references/user-story-template.md +58 -0
  301. package/gsd-core/references/verification-overrides.md +227 -0
  302. package/gsd-core/references/verification-patterns.md +612 -0
  303. package/gsd-core/references/verify-mvp-mode.md +85 -0
  304. package/gsd-core/references/workstream-flag.md +111 -0
  305. package/gsd-core/references/worktree-branch-check.md +38 -0
  306. package/gsd-core/references/worktree-path-safety.md +67 -0
  307. package/gsd-core/templates/AI-SPEC.md +246 -0
  308. package/gsd-core/templates/DEBUG.md +169 -0
  309. package/gsd-core/templates/README.md +77 -0
  310. package/gsd-core/templates/SECURITY.md +61 -0
  311. package/gsd-core/templates/UAT.md +265 -0
  312. package/gsd-core/templates/UI-SPEC.md +100 -0
  313. package/gsd-core/templates/VALIDATION.md +76 -0
  314. package/gsd-core/templates/adr.md +58 -0
  315. package/gsd-core/templates/claude-md.md +145 -0
  316. package/gsd-core/templates/codebase/architecture.md +255 -0
  317. package/gsd-core/templates/codebase/concerns.md +310 -0
  318. package/gsd-core/templates/codebase/conventions.md +307 -0
  319. package/gsd-core/templates/codebase/integrations.md +280 -0
  320. package/gsd-core/templates/codebase/stack.md +186 -0
  321. package/gsd-core/templates/codebase/structure.md +285 -0
  322. package/gsd-core/templates/codebase/testing.md +480 -0
  323. package/gsd-core/templates/config.json +62 -0
  324. package/gsd-core/templates/context.md +352 -0
  325. package/gsd-core/templates/continue-here.md +78 -0
  326. package/gsd-core/templates/copilot-instructions.md +7 -0
  327. package/gsd-core/templates/debug-subagent-prompt.md +91 -0
  328. package/gsd-core/templates/dev-preferences.md +21 -0
  329. package/gsd-core/templates/discovery.md +146 -0
  330. package/gsd-core/templates/discussion-log.md +63 -0
  331. package/gsd-core/templates/domain-model.md +54 -0
  332. package/gsd-core/templates/milestone-archive.md +123 -0
  333. package/gsd-core/templates/milestone.md +115 -0
  334. package/gsd-core/templates/phase-prompt.md +610 -0
  335. package/gsd-core/templates/planner-subagent-prompt.md +117 -0
  336. package/gsd-core/templates/product-brief.md +55 -0
  337. package/gsd-core/templates/project.md +186 -0
  338. package/gsd-core/templates/requirements.md +231 -0
  339. package/gsd-core/templates/research-project/ARCHITECTURE.md +204 -0
  340. package/gsd-core/templates/research-project/FEATURES.md +147 -0
  341. package/gsd-core/templates/research-project/PITFALLS.md +200 -0
  342. package/gsd-core/templates/research-project/STACK.md +120 -0
  343. package/gsd-core/templates/research-project/SUMMARY.md +170 -0
  344. package/gsd-core/templates/research.md +592 -0
  345. package/gsd-core/templates/retrospective.md +54 -0
  346. package/gsd-core/templates/roadmap.md +202 -0
  347. package/gsd-core/templates/spec.md +307 -0
  348. package/gsd-core/templates/state.md +195 -0
  349. package/gsd-core/templates/summary-complex.md +59 -0
  350. package/gsd-core/templates/summary-minimal.md +41 -0
  351. package/gsd-core/templates/summary-standard.md +48 -0
  352. package/gsd-core/templates/summary.md +248 -0
  353. package/gsd-core/templates/test-strategy.md +50 -0
  354. package/gsd-core/templates/user-profile.md +146 -0
  355. package/gsd-core/templates/user-setup.md +311 -0
  356. package/gsd-core/templates/verification-report.md +322 -0
  357. package/gsd-core/workflows/_runtime-launcher.snippet.sh +1 -0
  358. package/gsd-core/workflows/add-backlog.md +91 -0
  359. package/gsd-core/workflows/add-phase.md +113 -0
  360. package/gsd-core/workflows/add-tests.md +355 -0
  361. package/gsd-core/workflows/add-todo.md +161 -0
  362. package/gsd-core/workflows/ai-integration-phase.md +295 -0
  363. package/gsd-core/workflows/analyze-dependencies.md +96 -0
  364. package/gsd-core/workflows/audit-fix.md +178 -0
  365. package/gsd-core/workflows/audit-milestone.md +360 -0
  366. package/gsd-core/workflows/audit-uat.md +110 -0
  367. package/gsd-core/workflows/autonomous.md +797 -0
  368. package/gsd-core/workflows/check-todos.md +180 -0
  369. package/gsd-core/workflows/cleanup.md +195 -0
  370. package/gsd-core/workflows/code-review-fix.md +502 -0
  371. package/gsd-core/workflows/code-review.md +658 -0
  372. package/gsd-core/workflows/complete-milestone.md +855 -0
  373. package/gsd-core/workflows/debug.md +237 -0
  374. package/gsd-core/workflows/diagnose-issues.md +245 -0
  375. package/gsd-core/workflows/discover-product.md +112 -0
  376. package/gsd-core/workflows/discovery-phase.md +291 -0
  377. package/gsd-core/workflows/discuss-phase/modes/advisor.md +176 -0
  378. package/gsd-core/workflows/discuss-phase/modes/all.md +28 -0
  379. package/gsd-core/workflows/discuss-phase/modes/analyze.md +44 -0
  380. package/gsd-core/workflows/discuss-phase/modes/auto.md +57 -0
  381. package/gsd-core/workflows/discuss-phase/modes/batch.md +52 -0
  382. package/gsd-core/workflows/discuss-phase/modes/chain.md +98 -0
  383. package/gsd-core/workflows/discuss-phase/modes/default.md +141 -0
  384. package/gsd-core/workflows/discuss-phase/modes/power.md +44 -0
  385. package/gsd-core/workflows/discuss-phase/modes/text.md +55 -0
  386. package/gsd-core/workflows/discuss-phase/templates/checkpoint.json +18 -0
  387. package/gsd-core/workflows/discuss-phase/templates/context.md +136 -0
  388. package/gsd-core/workflows/discuss-phase/templates/discussion-log.md +50 -0
  389. package/gsd-core/workflows/discuss-phase-assumptions.md +675 -0
  390. package/gsd-core/workflows/discuss-phase-power.md +291 -0
  391. package/gsd-core/workflows/discuss-phase.md +499 -0
  392. package/gsd-core/workflows/do.md +111 -0
  393. package/gsd-core/workflows/docs-update.md +1176 -0
  394. package/gsd-core/workflows/edit-phase.md +295 -0
  395. package/gsd-core/workflows/eval-review.md +156 -0
  396. package/gsd-core/workflows/execute-phase/steps/codebase-drift-gate.md +95 -0
  397. package/gsd-core/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  398. package/gsd-core/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  399. package/gsd-core/workflows/execute-phase.md +1752 -0
  400. package/gsd-core/workflows/execute-plan.md +526 -0
  401. package/gsd-core/workflows/explore.md +146 -0
  402. package/gsd-core/workflows/extract-learnings.md +243 -0
  403. package/gsd-core/workflows/fast.md +124 -0
  404. package/gsd-core/workflows/forensics.md +279 -0
  405. package/gsd-core/workflows/graduation.md +196 -0
  406. package/gsd-core/workflows/health.md +224 -0
  407. package/gsd-core/workflows/help/modes/brief.md +22 -0
  408. package/gsd-core/workflows/help/modes/default.md +50 -0
  409. package/gsd-core/workflows/help/modes/full.md +789 -0
  410. package/gsd-core/workflows/help/modes/topic.md +74 -0
  411. package/gsd-core/workflows/help.md +24 -0
  412. package/gsd-core/workflows/import.md +256 -0
  413. package/gsd-core/workflows/inbox.md +387 -0
  414. package/gsd-core/workflows/ingest-docs.md +340 -0
  415. package/gsd-core/workflows/insert-phase.md +152 -0
  416. package/gsd-core/workflows/list-phase-assumptions.md +178 -0
  417. package/gsd-core/workflows/list-workspaces.md +57 -0
  418. package/gsd-core/workflows/manager.md +393 -0
  419. package/gsd-core/workflows/map-codebase.md +446 -0
  420. package/gsd-core/workflows/milestone-summary.md +224 -0
  421. package/gsd-core/workflows/model-domain.md +162 -0
  422. package/gsd-core/workflows/mvp-phase.md +222 -0
  423. package/gsd-core/workflows/new-milestone.md +635 -0
  424. package/gsd-core/workflows/new-project.md +1555 -0
  425. package/gsd-core/workflows/new-workspace.md +240 -0
  426. package/gsd-core/workflows/next.md +299 -0
  427. package/gsd-core/workflows/node-repair.md +92 -0
  428. package/gsd-core/workflows/note.md +158 -0
  429. package/gsd-core/workflows/pause-work.md +244 -0
  430. package/gsd-core/workflows/plan-milestone-gaps.md +281 -0
  431. package/gsd-core/workflows/plan-phase.md +1814 -0
  432. package/gsd-core/workflows/plan-review-convergence.md +346 -0
  433. package/gsd-core/workflows/plant-seed.md +230 -0
  434. package/gsd-core/workflows/pr-branch.md +157 -0
  435. package/gsd-core/workflows/profile-user.md +453 -0
  436. package/gsd-core/workflows/progress.md +699 -0
  437. package/gsd-core/workflows/quick.md +1017 -0
  438. package/gsd-core/workflows/reapply-patches.md +426 -0
  439. package/gsd-core/workflows/recommend-architecture.md +135 -0
  440. package/gsd-core/workflows/remove-phase.md +156 -0
  441. package/gsd-core/workflows/remove-workspace.md +108 -0
  442. package/gsd-core/workflows/resume-project.md +332 -0
  443. package/gsd-core/workflows/review.md +748 -0
  444. package/gsd-core/workflows/scan.md +107 -0
  445. package/gsd-core/workflows/secure-phase.md +182 -0
  446. package/gsd-core/workflows/session-report.md +146 -0
  447. package/gsd-core/workflows/settings-advanced.md +810 -0
  448. package/gsd-core/workflows/settings-integrations.md +312 -0
  449. package/gsd-core/workflows/settings.md +566 -0
  450. package/gsd-core/workflows/ship.md +405 -0
  451. package/gsd-core/workflows/sketch-wrap-up.md +286 -0
  452. package/gsd-core/workflows/sketch.md +361 -0
  453. package/gsd-core/workflows/spec-phase.md +263 -0
  454. package/gsd-core/workflows/spike-wrap-up.md +307 -0
  455. package/gsd-core/workflows/spike.md +453 -0
  456. package/gsd-core/workflows/stats.md +80 -0
  457. package/gsd-core/workflows/sync-skills.md +182 -0
  458. package/gsd-core/workflows/testing-strategy.md +122 -0
  459. package/gsd-core/workflows/thread.md +222 -0
  460. package/gsd-core/workflows/transition.md +694 -0
  461. package/gsd-core/workflows/ui-phase.md +328 -0
  462. package/gsd-core/workflows/ui-review.md +193 -0
  463. package/gsd-core/workflows/ultraplan-phase.md +199 -0
  464. package/gsd-core/workflows/undo.md +314 -0
  465. package/gsd-core/workflows/update.md +496 -0
  466. package/gsd-core/workflows/validate-phase.md +181 -0
  467. package/gsd-core/workflows/verify-phase.md +544 -0
  468. package/gsd-core/workflows/verify-work.md +781 -0
  469. package/hooks/dist/gsd-check-update-worker.js +108 -0
  470. package/hooks/dist/gsd-check-update.js +66 -0
  471. package/hooks/dist/gsd-config-reload.js +133 -0
  472. package/hooks/dist/gsd-context-monitor.js +195 -0
  473. package/hooks/dist/gsd-cursor-post-tool.js +75 -0
  474. package/hooks/dist/gsd-cursor-session-start.js +52 -0
  475. package/hooks/dist/gsd-graphify-update.sh +158 -0
  476. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  477. package/hooks/dist/gsd-prompt-guard.js +97 -0
  478. package/hooks/dist/gsd-read-guard.js +101 -0
  479. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  480. package/hooks/dist/gsd-session-state.sh +59 -0
  481. package/hooks/dist/gsd-statusline.js +566 -0
  482. package/hooks/dist/gsd-update-banner.js +138 -0
  483. package/hooks/dist/gsd-validate-commit.sh +57 -0
  484. package/hooks/dist/gsd-workflow-guard.js +167 -0
  485. package/hooks/dist/gsd-worktree-path-guard.js +169 -0
  486. package/hooks/dist/lib/git-cmd.js +150 -0
  487. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  488. package/hooks/dist/managed-hooks-registry.cjs +38 -0
  489. package/hooks/gsd-check-update-worker.js +108 -0
  490. package/hooks/gsd-check-update.js +66 -0
  491. package/hooks/gsd-config-reload.js +133 -0
  492. package/hooks/gsd-context-monitor.js +195 -0
  493. package/hooks/gsd-cursor-post-tool.js +75 -0
  494. package/hooks/gsd-cursor-session-start.js +52 -0
  495. package/hooks/gsd-graphify-update.sh +158 -0
  496. package/hooks/gsd-phase-boundary.sh +47 -0
  497. package/hooks/gsd-prompt-guard.js +97 -0
  498. package/hooks/gsd-read-guard.js +101 -0
  499. package/hooks/gsd-read-injection-scanner.js +203 -0
  500. package/hooks/gsd-session-state.sh +59 -0
  501. package/hooks/gsd-statusline.js +566 -0
  502. package/hooks/gsd-update-banner.js +138 -0
  503. package/hooks/gsd-validate-commit.sh +57 -0
  504. package/hooks/gsd-workflow-guard.js +167 -0
  505. package/hooks/gsd-worktree-path-guard.js +169 -0
  506. package/hooks/hooks.json +69 -0
  507. package/hooks/lib/git-cmd.js +150 -0
  508. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  509. package/hooks/managed-hooks-registry.cjs +38 -0
  510. package/package.json +115 -0
  511. package/scripts/affected-tests-lib.cjs +542 -0
  512. package/scripts/audit-workflow-script-paths.cjs +73 -0
  513. package/scripts/base64-scan.sh +351 -0
  514. package/scripts/build-hooks.js +247 -0
  515. package/scripts/changeset/README.md +129 -0
  516. package/scripts/changeset/cli.cjs +590 -0
  517. package/scripts/changeset/github-release-notes.cjs +199 -0
  518. package/scripts/changeset/lint.cjs +111 -0
  519. package/scripts/changeset/new.cjs +137 -0
  520. package/scripts/changeset/parse.cjs +114 -0
  521. package/scripts/changeset/render.cjs +34 -0
  522. package/scripts/changeset/serialize.cjs +130 -0
  523. package/scripts/check-alias-drift.cjs +114 -0
  524. package/scripts/check-env.cjs +312 -0
  525. package/scripts/check-npm-integrity.cjs +215 -0
  526. package/scripts/ci-guard-runner.cjs +22 -0
  527. package/scripts/ci-prepare-test-scope.cjs +51 -0
  528. package/scripts/ci-rebase-check.cjs +86 -0
  529. package/scripts/ci-test-scope.cjs +431 -0
  530. package/scripts/command-contract-helpers.cjs +64 -0
  531. package/scripts/diff-touches-shipped-paths.cjs +155 -0
  532. package/scripts/fix-slash-commands.cjs +147 -0
  533. package/scripts/gen-inventory-manifest.cjs +115 -0
  534. package/scripts/gen-research-agents.cjs +276 -0
  535. package/scripts/generate-package-identity.cjs +125 -0
  536. package/scripts/issue-dedupe.cjs +278 -0
  537. package/scripts/lib/allowlist-ratchet.cjs +136 -0
  538. package/scripts/lib/cli-exit.cjs +56 -0
  539. package/scripts/lint-command-contract.cjs +114 -0
  540. package/scripts/lint-descriptions.cjs +87 -0
  541. package/scripts/lint-docs-required.cjs +222 -0
  542. package/scripts/lint-legacy-dir-name.cjs +160 -0
  543. package/scripts/lint-package-identity-drift.cjs +141 -0
  544. package/scripts/lint-pr-check-project-dir.cjs +99 -0
  545. package/scripts/lint-shell-command-projection-drift.cjs +62 -0
  546. package/scripts/lint-skill-deps.cjs +185 -0
  547. package/scripts/lint-test-file-count.allowlist.json +135 -0
  548. package/scripts/lint-test-file-count.cjs +246 -0
  549. package/scripts/mutation-matrix.cjs +222 -0
  550. package/scripts/pr-template-policy.cjs +268 -0
  551. package/scripts/prompt-injection-scan.sh +207 -0
  552. package/scripts/release-notes/discord-release-summary.cjs +373 -0
  553. package/scripts/release-notes/format-github-release-notes.cjs +261 -0
  554. package/scripts/release-tarball-smoke.cjs +629 -0
  555. package/scripts/research-profiles.cjs +149 -0
  556. package/scripts/run-affected-tests.cjs +7 -0
  557. package/scripts/run-cross-platform-tests.cjs +67 -0
  558. package/scripts/run-tests.cjs +315 -0
  559. package/scripts/secret-scan-lint.sh +231 -0
  560. package/scripts/secret-scan.sh +358 -0
  561. package/scripts/setup-branch-protection.sh +236 -0
  562. package/scripts/strip-prose-atrefs.cjs +106 -0
  563. package/scripts/sync-manifest-versions.cjs +119 -0
  564. package/scripts/sync-rulesets.sh +34 -0
  565. package/scripts/sync-runtime-launcher.cjs +399 -0
  566. package/scripts/test-failure-reasons.cjs +34 -0
  567. package/scripts/verify-npm-publish.cjs +240 -0
  568. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,743 @@
1
+ "use strict";
2
+ /**
3
+ * Open Artifact Audit — Cross-type unresolved state scanner
4
+ *
5
+ * Scans all .planning/ artifact categories for items with open/unresolved state.
6
+ * Returns structured JSON for workflow consumption.
7
+ * Called by: gsd-tools.cjs audit-open
8
+ * Used by: /gsd:complete-milestone pre-close gate
9
+ *
10
+ * ADR-457 build-at-publish: the hand-written bin/lib/audit.cjs collapsed
11
+ * to a TypeScript source of truth. Behaviour is preserved byte-for-behaviour
12
+ * from the prior hand-written .cjs; only strict types are added.
13
+ */
14
+ var __importDefault = (this && this.__importDefault) || function (mod) {
15
+ return (mod && mod.__esModule) ? mod : { "default": mod };
16
+ };
17
+ const node_fs_1 = __importDefault(require("node:fs"));
18
+ const node_path_1 = __importDefault(require("node:path"));
19
+ const shell_command_projection_cjs_1 = require("./shell-command-projection.cjs");
20
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
21
+ const planningWorkspace = require("./planning-workspace.cjs");
22
+ const { planningDir } = planningWorkspace;
23
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
24
+ const frontmatter = require("./frontmatter.cjs");
25
+ const { extractFrontmatter } = frontmatter;
26
+ const security_cjs_1 = require("./security.cjs");
27
+ // Terminal UAT states: `complete` (legacy) and `resolved` (post-gap-closure
28
+ // per workflows/execute-phase.md). Hoisted outside scanUatGaps so the Set is
29
+ // not recreated on each loop iteration.
30
+ const TERMINAL_UAT_STATUSES = new Set(['complete', 'resolved']);
31
+ // ─── scanDebugSessions ────────────────────────────────────────────────────────
32
+ /**
33
+ * Scan .planning/debug/ for open sessions.
34
+ * Open = status NOT in ['resolved', 'complete'].
35
+ * Ignores the resolved/ subdirectory.
36
+ */
37
+ function scanDebugSessions(planDir) {
38
+ const debugDir = node_path_1.default.join(planDir, 'debug');
39
+ if (!node_fs_1.default.existsSync(debugDir))
40
+ return [];
41
+ const results = [];
42
+ let files;
43
+ try {
44
+ files = node_fs_1.default.readdirSync(debugDir, { withFileTypes: true });
45
+ }
46
+ catch {
47
+ return [{ scan_error: true, slug: '', status: '', updated: '', hypothesis: '' }];
48
+ }
49
+ for (const entry of files) {
50
+ if (!entry.isFile())
51
+ continue;
52
+ if (!entry.name.endsWith('.md'))
53
+ continue;
54
+ const filePath = node_path_1.default.join(debugDir, entry.name);
55
+ let safeFilePath;
56
+ try {
57
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'debug session file', { allowAbsolute: true });
58
+ }
59
+ catch {
60
+ continue;
61
+ }
62
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
63
+ if (content === null)
64
+ continue;
65
+ const fm = extractFrontmatter(content);
66
+ const status = (fm.status || 'unknown').toLowerCase();
67
+ if (status === 'resolved' || status === 'complete')
68
+ continue;
69
+ // Extract hypothesis from "Current Focus" block if parseable
70
+ let hypothesis = '';
71
+ const focusMatch = content.match(/##\s*Current Focus[^\n]*\n([\s\S]*?)(?=\n##\s|$)/i);
72
+ if (focusMatch) {
73
+ const focusText = focusMatch[1].trim().split('\n')[0].trim();
74
+ hypothesis = (0, security_cjs_1.sanitizeForDisplay)(focusText.slice(0, 100));
75
+ }
76
+ const slug = node_path_1.default.basename(entry.name, '.md');
77
+ results.push({
78
+ slug: (0, security_cjs_1.sanitizeForDisplay)(slug),
79
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
80
+ updated: (0, security_cjs_1.sanitizeForDisplay)(fm.updated || fm.date || ''),
81
+ hypothesis,
82
+ });
83
+ }
84
+ return results;
85
+ }
86
+ // ─── scanQuickTasks ───────────────────────────────────────────────────────────
87
+ /**
88
+ * Scan .planning/quick/ for incomplete tasks.
89
+ * Incomplete if SUMMARY.md missing or status !== 'complete'.
90
+ */
91
+ function scanQuickTasks(planDir) {
92
+ const quickDir = node_path_1.default.join(planDir, 'quick');
93
+ if (!node_fs_1.default.existsSync(quickDir))
94
+ return [];
95
+ let entries;
96
+ try {
97
+ entries = node_fs_1.default.readdirSync(quickDir, { withFileTypes: true });
98
+ }
99
+ catch {
100
+ return [{ scan_error: true, slug: '', date: '', status: '', description: '' }];
101
+ }
102
+ const results = [];
103
+ for (const entry of entries) {
104
+ if (!entry.isDirectory())
105
+ continue;
106
+ const dirName = entry.name;
107
+ const taskDir = node_path_1.default.join(quickDir, dirName);
108
+ let safeTaskDir;
109
+ try {
110
+ safeTaskDir = (0, security_cjs_1.requireSafePath)(taskDir, planDir, 'quick task dir', { allowAbsolute: true });
111
+ }
112
+ catch {
113
+ continue;
114
+ }
115
+ // workflows/quick.md mandates `${quick_id}-SUMMARY.md`; older flows used
116
+ // bare `SUMMARY.md`. Accept either to avoid false-positive "missing".
117
+ let summaryPath = null;
118
+ try {
119
+ const summaryFiles = node_fs_1.default.readdirSync(safeTaskDir, { withFileTypes: true })
120
+ .filter(e => e.isFile() && (e.name === 'SUMMARY.md' || e.name.endsWith('-SUMMARY.md')));
121
+ if (summaryFiles.length > 0) {
122
+ // Prefer the per-task `${quick_id}-SUMMARY.md` form when present.
123
+ const preferred = summaryFiles.find(e => e.name === `${dirName}-SUMMARY.md`)
124
+ || summaryFiles.find(e => e.name.endsWith('-SUMMARY.md'))
125
+ || summaryFiles[0];
126
+ summaryPath = node_path_1.default.join(safeTaskDir, preferred.name);
127
+ }
128
+ }
129
+ catch {
130
+ // fall through with summaryPath = null → status: missing
131
+ }
132
+ let status = 'missing';
133
+ const description = '';
134
+ if (summaryPath && node_fs_1.default.existsSync(summaryPath)) {
135
+ let safeSum;
136
+ try {
137
+ safeSum = (0, security_cjs_1.requireSafePath)(summaryPath, planDir, 'quick task summary', { allowAbsolute: true });
138
+ }
139
+ catch {
140
+ continue;
141
+ }
142
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeSum);
143
+ if (content === null) {
144
+ status = 'unreadable';
145
+ }
146
+ else {
147
+ const fm = extractFrontmatter(content);
148
+ status = (fm.status || 'unknown').toLowerCase();
149
+ }
150
+ }
151
+ if (status === 'complete')
152
+ continue;
153
+ // Parse date and slug from directory name: YYYYMMDD-slug or YYYY-MM-DD-slug
154
+ let date = '';
155
+ let slug = (0, security_cjs_1.sanitizeForDisplay)(dirName);
156
+ const dateMatch = dirName.match(/^(\d{4}-?\d{2}-?\d{2})-(.+)$/);
157
+ if (dateMatch) {
158
+ date = dateMatch[1];
159
+ slug = (0, security_cjs_1.sanitizeForDisplay)(dateMatch[2]);
160
+ }
161
+ results.push({
162
+ slug,
163
+ date,
164
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
165
+ description,
166
+ });
167
+ }
168
+ return results;
169
+ }
170
+ // ─── scanThreads ──────────────────────────────────────────────────────────────
171
+ /**
172
+ * Scan .planning/threads/ for open threads.
173
+ * Open if status in ['open', 'in_progress', 'in progress'] (case-insensitive).
174
+ */
175
+ function scanThreads(planDir) {
176
+ const threadsDir = node_path_1.default.join(planDir, 'threads');
177
+ if (!node_fs_1.default.existsSync(threadsDir))
178
+ return [];
179
+ let files;
180
+ try {
181
+ files = node_fs_1.default.readdirSync(threadsDir, { withFileTypes: true });
182
+ }
183
+ catch {
184
+ return [{ scan_error: true, slug: '', status: '', updated: '', title: '' }];
185
+ }
186
+ const openStatuses = new Set(['open', 'in_progress', 'in progress']);
187
+ const results = [];
188
+ for (const entry of files) {
189
+ if (!entry.isFile())
190
+ continue;
191
+ if (!entry.name.endsWith('.md'))
192
+ continue;
193
+ const filePath = node_path_1.default.join(threadsDir, entry.name);
194
+ let safeFilePath;
195
+ try {
196
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'thread file', { allowAbsolute: true });
197
+ }
198
+ catch {
199
+ continue;
200
+ }
201
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
202
+ if (content === null)
203
+ continue;
204
+ const fm = extractFrontmatter(content);
205
+ let status = (fm.status || '').toLowerCase().trim();
206
+ // Fall back to scanning body for ## Status: OPEN / IN PROGRESS
207
+ if (!status) {
208
+ const bodyStatusMatch = content.match(/##\s*Status:\s*(OPEN|IN PROGRESS|IN_PROGRESS)/i);
209
+ if (bodyStatusMatch) {
210
+ status = bodyStatusMatch[1].toLowerCase().replace(/ /g, '_');
211
+ }
212
+ }
213
+ if (!openStatuses.has(status))
214
+ continue;
215
+ // Extract title from # Thread: heading or frontmatter title
216
+ let title = (0, security_cjs_1.sanitizeForDisplay)(fm.title || '');
217
+ if (!title) {
218
+ const headingMatch = content.match(/^#\s*Thread:\s*(.+)$/m);
219
+ if (headingMatch) {
220
+ title = (0, security_cjs_1.sanitizeForDisplay)(headingMatch[1].trim().slice(0, 100));
221
+ }
222
+ }
223
+ const slug = node_path_1.default.basename(entry.name, '.md');
224
+ results.push({
225
+ slug: (0, security_cjs_1.sanitizeForDisplay)(slug),
226
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
227
+ updated: (0, security_cjs_1.sanitizeForDisplay)(fm.updated || fm.date || ''),
228
+ title,
229
+ });
230
+ }
231
+ return results;
232
+ }
233
+ // ─── scanTodos ────────────────────────────────────────────────────────────────
234
+ /**
235
+ * Scan .planning/todos/pending/ for pending todos.
236
+ * Returns array of { filename, priority, area, summary }.
237
+ * Display limited to first 5 + count of remainder.
238
+ */
239
+ function scanTodos(planDir) {
240
+ const pendingDir = node_path_1.default.join(planDir, 'todos', 'pending');
241
+ if (!node_fs_1.default.existsSync(pendingDir))
242
+ return [];
243
+ let files;
244
+ try {
245
+ files = node_fs_1.default.readdirSync(pendingDir, { withFileTypes: true });
246
+ }
247
+ catch {
248
+ return [{ scan_error: true, filename: '', priority: '', area: '', summary: '' }];
249
+ }
250
+ const mdFiles = files.filter(e => e.isFile() && e.name.endsWith('.md'));
251
+ const results = [];
252
+ const displayFiles = mdFiles.slice(0, 5);
253
+ for (const entry of displayFiles) {
254
+ const filePath = node_path_1.default.join(pendingDir, entry.name);
255
+ let safeFilePath;
256
+ try {
257
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'todo file', { allowAbsolute: true });
258
+ }
259
+ catch {
260
+ continue;
261
+ }
262
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
263
+ if (content === null)
264
+ continue;
265
+ const fm = extractFrontmatter(content);
266
+ // Extract first line of body after frontmatter
267
+ const bodyMatch = content.replace(/^---[\s\S]*?---\n?/, '');
268
+ const firstLine = bodyMatch.trim().split('\n')[0] || '';
269
+ const summary = (0, security_cjs_1.sanitizeForDisplay)(firstLine.slice(0, 100));
270
+ results.push({
271
+ filename: (0, security_cjs_1.sanitizeForDisplay)(entry.name),
272
+ priority: (0, security_cjs_1.sanitizeForDisplay)(fm.priority || ''),
273
+ area: (0, security_cjs_1.sanitizeForDisplay)(fm.area || ''),
274
+ summary,
275
+ });
276
+ }
277
+ if (mdFiles.length > 5) {
278
+ results.push({ _remainder_count: mdFiles.length - 5, filename: '', priority: '', area: '', summary: '' });
279
+ }
280
+ return results;
281
+ }
282
+ // ─── scanSeeds ────────────────────────────────────────────────────────────────
283
+ /**
284
+ * Scan .planning/seeds/SEED-*.md for unimplemented seeds.
285
+ * Unimplemented if status in ['dormant', 'active', 'triggered'].
286
+ */
287
+ function scanSeeds(planDir) {
288
+ const seedsDir = node_path_1.default.join(planDir, 'seeds');
289
+ if (!node_fs_1.default.existsSync(seedsDir))
290
+ return [];
291
+ let files;
292
+ try {
293
+ files = node_fs_1.default.readdirSync(seedsDir, { withFileTypes: true });
294
+ }
295
+ catch {
296
+ return [{ scan_error: true, seed_id: '', slug: '', status: '', title: '' }];
297
+ }
298
+ const unimplementedStatuses = new Set(['dormant', 'active', 'triggered']);
299
+ const results = [];
300
+ for (const entry of files) {
301
+ if (!entry.isFile())
302
+ continue;
303
+ if (!entry.name.startsWith('SEED-') || !entry.name.endsWith('.md'))
304
+ continue;
305
+ const filePath = node_path_1.default.join(seedsDir, entry.name);
306
+ let safeFilePath;
307
+ try {
308
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'seed file', { allowAbsolute: true });
309
+ }
310
+ catch {
311
+ continue;
312
+ }
313
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
314
+ if (content === null)
315
+ continue;
316
+ const fm = extractFrontmatter(content);
317
+ const status = (fm.status || 'dormant').toLowerCase();
318
+ if (!unimplementedStatuses.has(status))
319
+ continue;
320
+ // Extract seed_id from filename or frontmatter
321
+ const seedIdMatch = entry.name.match(/^(SEED-[\w-]+)\.md$/);
322
+ const seed_id = seedIdMatch ? seedIdMatch[1] : node_path_1.default.basename(entry.name, '.md');
323
+ const slug = (0, security_cjs_1.sanitizeForDisplay)(seed_id.replace(/^SEED-/, ''));
324
+ let title = (0, security_cjs_1.sanitizeForDisplay)(fm.title || '');
325
+ if (!title) {
326
+ const headingMatch = content.match(/^#\s*(.+)$/m);
327
+ if (headingMatch)
328
+ title = (0, security_cjs_1.sanitizeForDisplay)(headingMatch[1].trim().slice(0, 100));
329
+ }
330
+ results.push({
331
+ seed_id: (0, security_cjs_1.sanitizeForDisplay)(seed_id),
332
+ slug,
333
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
334
+ title,
335
+ });
336
+ }
337
+ return results;
338
+ }
339
+ // ─── scanUatGaps ──────────────────────────────────────────────────────────────
340
+ /**
341
+ * Scan .planning/phases for UAT gaps (UAT files with status != 'complete').
342
+ */
343
+ function scanUatGaps(planDir) {
344
+ const phasesDir = node_path_1.default.join(planDir, 'phases');
345
+ if (!node_fs_1.default.existsSync(phasesDir))
346
+ return [];
347
+ let dirs;
348
+ try {
349
+ dirs = node_fs_1.default.readdirSync(phasesDir, { withFileTypes: true })
350
+ .filter(e => e.isDirectory())
351
+ .map(e => e.name)
352
+ .sort();
353
+ }
354
+ catch {
355
+ return [{ scan_error: true, phase: '', file: '', status: '', open_scenario_count: 0 }];
356
+ }
357
+ const results = [];
358
+ for (const dir of dirs) {
359
+ const phaseDir = node_path_1.default.join(phasesDir, dir);
360
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
361
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
362
+ let files;
363
+ try {
364
+ files = node_fs_1.default.readdirSync(phaseDir);
365
+ }
366
+ catch {
367
+ continue;
368
+ }
369
+ for (const file of files.filter(f => f.includes('-UAT') && f.endsWith('.md'))) {
370
+ const filePath = node_path_1.default.join(phaseDir, file);
371
+ let safeFilePath;
372
+ try {
373
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'UAT file', { allowAbsolute: true });
374
+ }
375
+ catch {
376
+ continue;
377
+ }
378
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
379
+ if (content === null)
380
+ continue;
381
+ const fm = extractFrontmatter(content);
382
+ const status = (fm.status || 'unknown').toLowerCase();
383
+ const result = (fm.result || '').toLowerCase();
384
+ // Also accept `result: all_pass` as a fallback when status is absent
385
+ // — covers UATs that omit `status:`.
386
+ if (TERMINAL_UAT_STATUSES.has(status))
387
+ continue;
388
+ if (status === 'unknown' && result === 'all_pass')
389
+ continue;
390
+ // Count open scenarios
391
+ const pendingMatches = (content.match(/result:\s*(?:pending|\[pending\])/gi) || []).length;
392
+ results.push({
393
+ phase: (0, security_cjs_1.sanitizeForDisplay)(phaseNum),
394
+ file: (0, security_cjs_1.sanitizeForDisplay)(file),
395
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
396
+ open_scenario_count: pendingMatches,
397
+ });
398
+ }
399
+ }
400
+ return results;
401
+ }
402
+ // ─── scanVerificationGaps ─────────────────────────────────────────────────────
403
+ /**
404
+ * Scan .planning/phases for VERIFICATION gaps.
405
+ */
406
+ function scanVerificationGaps(planDir) {
407
+ const phasesDir = node_path_1.default.join(planDir, 'phases');
408
+ if (!node_fs_1.default.existsSync(phasesDir))
409
+ return [];
410
+ let dirs;
411
+ try {
412
+ dirs = node_fs_1.default.readdirSync(phasesDir, { withFileTypes: true })
413
+ .filter(e => e.isDirectory())
414
+ .map(e => e.name)
415
+ .sort();
416
+ }
417
+ catch {
418
+ return [{ scan_error: true, phase: '', file: '', status: '' }];
419
+ }
420
+ const results = [];
421
+ for (const dir of dirs) {
422
+ const phaseDir = node_path_1.default.join(phasesDir, dir);
423
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
424
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
425
+ let files;
426
+ try {
427
+ files = node_fs_1.default.readdirSync(phaseDir);
428
+ }
429
+ catch {
430
+ continue;
431
+ }
432
+ for (const file of files.filter(f => f.includes('-VERIFICATION') && f.endsWith('.md'))) {
433
+ const filePath = node_path_1.default.join(phaseDir, file);
434
+ let safeFilePath;
435
+ try {
436
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'VERIFICATION file', { allowAbsolute: true });
437
+ }
438
+ catch {
439
+ continue;
440
+ }
441
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
442
+ if (content === null)
443
+ continue;
444
+ const fm = extractFrontmatter(content);
445
+ const status = (fm.status || 'unknown').toLowerCase();
446
+ if (status !== 'gaps_found' && status !== 'human_needed')
447
+ continue;
448
+ results.push({
449
+ phase: (0, security_cjs_1.sanitizeForDisplay)(phaseNum),
450
+ file: (0, security_cjs_1.sanitizeForDisplay)(file),
451
+ status: (0, security_cjs_1.sanitizeForDisplay)(status),
452
+ });
453
+ }
454
+ }
455
+ return results;
456
+ }
457
+ // ─── scanContextQuestions ─────────────────────────────────────────────────────
458
+ /**
459
+ * Scan .planning/phases for CONTEXT files with open_questions.
460
+ */
461
+ function scanContextQuestions(planDir) {
462
+ const phasesDir = node_path_1.default.join(planDir, 'phases');
463
+ if (!node_fs_1.default.existsSync(phasesDir))
464
+ return [];
465
+ let dirs;
466
+ try {
467
+ dirs = node_fs_1.default.readdirSync(phasesDir, { withFileTypes: true })
468
+ .filter(e => e.isDirectory())
469
+ .map(e => e.name)
470
+ .sort();
471
+ }
472
+ catch {
473
+ return [{ scan_error: true, phase: '', file: '', question_count: 0, questions: [] }];
474
+ }
475
+ const results = [];
476
+ for (const dir of dirs) {
477
+ const phaseDir = node_path_1.default.join(phasesDir, dir);
478
+ const phaseMatch = dir.match(/^(\d+[A-Z]?(?:\.\d+)*)/i);
479
+ const phaseNum = phaseMatch ? phaseMatch[1] : dir;
480
+ let files;
481
+ try {
482
+ files = node_fs_1.default.readdirSync(phaseDir);
483
+ }
484
+ catch {
485
+ continue;
486
+ }
487
+ for (const file of files.filter(f => f.includes('-CONTEXT') && f.endsWith('.md'))) {
488
+ const filePath = node_path_1.default.join(phaseDir, file);
489
+ let safeFilePath;
490
+ try {
491
+ safeFilePath = (0, security_cjs_1.requireSafePath)(filePath, planDir, 'CONTEXT file', { allowAbsolute: true });
492
+ }
493
+ catch {
494
+ continue;
495
+ }
496
+ const content = (0, shell_command_projection_cjs_1.platformReadSync)(safeFilePath);
497
+ if (content === null)
498
+ continue;
499
+ const fm = extractFrontmatter(content);
500
+ // Check frontmatter open_questions field
501
+ let questions = [];
502
+ if (fm.open_questions) {
503
+ if (Array.isArray(fm.open_questions) && fm.open_questions.length > 0) {
504
+ questions = fm.open_questions.map(q => (0, security_cjs_1.sanitizeForDisplay)(String(q).slice(0, 200)));
505
+ }
506
+ }
507
+ // Also check for ## Open Questions section in body
508
+ if (questions.length === 0) {
509
+ const oqMatch = content.match(/##\s*Open Questions[^\n]*\n([\s\S]*?)(?=\n##\s|$)/i);
510
+ if (oqMatch) {
511
+ const oqBody = oqMatch[1].trim();
512
+ if (oqBody && oqBody.length > 0 && !/^\s*none\s*$/i.test(oqBody)) {
513
+ const items = oqBody.split('\n')
514
+ .map((l) => l.trim())
515
+ .filter((l) => l && l !== '-' && l !== '*')
516
+ .filter((l) => /^[-*\d]/.test(l) || l.includes('?'));
517
+ questions = items.slice(0, 3).map((q) => (0, security_cjs_1.sanitizeForDisplay)(q.slice(0, 200)));
518
+ }
519
+ }
520
+ }
521
+ if (questions.length === 0)
522
+ continue;
523
+ results.push({
524
+ phase: (0, security_cjs_1.sanitizeForDisplay)(phaseNum),
525
+ file: (0, security_cjs_1.sanitizeForDisplay)(file),
526
+ question_count: questions.length,
527
+ questions: questions.slice(0, 3),
528
+ });
529
+ }
530
+ }
531
+ return results;
532
+ }
533
+ // ─── auditOpenArtifacts ───────────────────────────────────────────────────────
534
+ /**
535
+ * Main audit function. Scans all .planning/ artifact categories.
536
+ *
537
+ * @param cwd - Project root directory
538
+ * @returns Structured audit result
539
+ */
540
+ function auditOpenArtifacts(cwd) {
541
+ const planDir = planningDir(cwd);
542
+ const debugSessions = (() => {
543
+ try {
544
+ return scanDebugSessions(planDir);
545
+ }
546
+ catch {
547
+ return [{ scan_error: true, slug: '', status: '', updated: '', hypothesis: '' }];
548
+ }
549
+ })();
550
+ const quickTasks = (() => {
551
+ try {
552
+ return scanQuickTasks(planDir);
553
+ }
554
+ catch {
555
+ return [{ scan_error: true, slug: '', date: '', status: '', description: '' }];
556
+ }
557
+ })();
558
+ const threads = (() => {
559
+ try {
560
+ return scanThreads(planDir);
561
+ }
562
+ catch {
563
+ return [{ scan_error: true, slug: '', status: '', updated: '', title: '' }];
564
+ }
565
+ })();
566
+ const todos = (() => {
567
+ try {
568
+ return scanTodos(planDir);
569
+ }
570
+ catch {
571
+ return [{ scan_error: true, filename: '', priority: '', area: '', summary: '' }];
572
+ }
573
+ })();
574
+ const seeds = (() => {
575
+ try {
576
+ return scanSeeds(planDir);
577
+ }
578
+ catch {
579
+ return [{ scan_error: true, seed_id: '', slug: '', status: '', title: '' }];
580
+ }
581
+ })();
582
+ const uatGaps = (() => {
583
+ try {
584
+ return scanUatGaps(planDir);
585
+ }
586
+ catch {
587
+ return [{ scan_error: true, phase: '', file: '', status: '', open_scenario_count: 0 }];
588
+ }
589
+ })();
590
+ const verificationGaps = (() => {
591
+ try {
592
+ return scanVerificationGaps(planDir);
593
+ }
594
+ catch {
595
+ return [{ scan_error: true, phase: '', file: '', status: '' }];
596
+ }
597
+ })();
598
+ const contextQuestions = (() => {
599
+ try {
600
+ return scanContextQuestions(planDir);
601
+ }
602
+ catch {
603
+ return [{ scan_error: true, phase: '', file: '', question_count: 0, questions: [] }];
604
+ }
605
+ })();
606
+ // Count real items (not scan_error sentinels)
607
+ const countReal = (arr) => arr.filter(i => !i.scan_error && !i._remainder_count).length;
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
+ total: 0,
618
+ };
619
+ counts.total = counts.debug_sessions + counts.quick_tasks + counts.threads + counts.todos + counts.seeds + counts.uat_gaps + counts.verification_gaps + counts.context_questions;
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
+ // ─── formatAuditReport ────────────────────────────────────────────────────────
637
+ /**
638
+ * Format the audit result as a human-readable report.
639
+ *
640
+ * @param auditResult - Result from auditOpenArtifacts()
641
+ * @returns Formatted report
642
+ */
643
+ function formatAuditReport(auditResult) {
644
+ const { counts, items, has_open_items } = auditResult;
645
+ const lines = [];
646
+ const hr = '━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━';
647
+ lines.push(hr);
648
+ lines.push(' Milestone Close: Open Artifact Audit');
649
+ lines.push(hr);
650
+ if (!has_open_items) {
651
+ lines.push('');
652
+ lines.push(' All artifact types clear. Safe to proceed.');
653
+ lines.push('');
654
+ lines.push(hr);
655
+ return lines.join('\n');
656
+ }
657
+ // Debug sessions (blocking quality — red)
658
+ if (counts.debug_sessions > 0) {
659
+ lines.push('');
660
+ lines.push(`🔴 Debug Sessions (${counts.debug_sessions} open)`);
661
+ for (const item of items.debug_sessions.filter(i => !i.scan_error)) {
662
+ const hyp = item.hypothesis ? ` — ${item.hypothesis}` : '';
663
+ lines.push(` • ${item.slug} [${item.status}]${hyp}`);
664
+ }
665
+ }
666
+ // UAT gaps (blocking quality — red)
667
+ if (counts.uat_gaps > 0) {
668
+ lines.push('');
669
+ lines.push(`🔴 UAT Gaps (${counts.uat_gaps} phases with incomplete UAT)`);
670
+ for (const item of items.uat_gaps.filter(i => !i.scan_error)) {
671
+ lines.push(` • Phase ${item.phase}: ${item.file} [${item.status}] — ${item.open_scenario_count} pending scenarios`);
672
+ }
673
+ }
674
+ // Verification gaps (blocking quality — red)
675
+ if (counts.verification_gaps > 0) {
676
+ lines.push('');
677
+ lines.push(`🔴 Verification Gaps (${counts.verification_gaps} unresolved)`);
678
+ for (const item of items.verification_gaps.filter(i => !i.scan_error)) {
679
+ lines.push(` • Phase ${item.phase}: ${item.file} [${item.status}]`);
680
+ }
681
+ }
682
+ // Quick tasks (incomplete work — yellow)
683
+ if (counts.quick_tasks > 0) {
684
+ lines.push('');
685
+ lines.push(`🟡 Quick Tasks (${counts.quick_tasks} incomplete)`);
686
+ for (const item of items.quick_tasks.filter(i => !i.scan_error)) {
687
+ const d = item.date ? ` (${item.date})` : '';
688
+ lines.push(` • ${item.slug}${d} [${item.status}]`);
689
+ }
690
+ }
691
+ // Todos (incomplete work — yellow)
692
+ if (counts.todos > 0) {
693
+ const realTodos = items.todos.filter(i => !i.scan_error && !i._remainder_count);
694
+ const remainder = items.todos.find(i => i._remainder_count);
695
+ lines.push('');
696
+ lines.push(`🟡 Pending Todos (${counts.todos} pending)`);
697
+ for (const item of realTodos) {
698
+ const area = item.area ? ` [${item.area}]` : '';
699
+ const pri = item.priority ? ` (${item.priority})` : '';
700
+ lines.push(` • ${item.filename}${area}${pri}`);
701
+ if (item.summary)
702
+ lines.push(` ${item.summary}`);
703
+ }
704
+ if (remainder) {
705
+ lines.push(` ... and ${remainder._remainder_count} more`);
706
+ }
707
+ }
708
+ // Threads (deferred decisions — blue)
709
+ if (counts.threads > 0) {
710
+ lines.push('');
711
+ lines.push(`🔵 Open Threads (${counts.threads} active)`);
712
+ for (const item of items.threads.filter(i => !i.scan_error)) {
713
+ const title = item.title ? ` — ${item.title}` : '';
714
+ lines.push(` • ${item.slug} [${item.status}]${title}`);
715
+ }
716
+ }
717
+ // Seeds (deferred decisions — blue)
718
+ if (counts.seeds > 0) {
719
+ lines.push('');
720
+ lines.push(`🔵 Unimplemented Seeds (${counts.seeds} pending)`);
721
+ for (const item of items.seeds.filter(i => !i.scan_error)) {
722
+ const title = item.title ? ` — ${item.title}` : '';
723
+ lines.push(` • ${item.seed_id} [${item.status}]${title}`);
724
+ }
725
+ }
726
+ // Context questions (deferred decisions — blue)
727
+ if (counts.context_questions > 0) {
728
+ lines.push('');
729
+ lines.push(`🔵 CONTEXT Open Questions (${counts.context_questions} phases with open questions)`);
730
+ for (const item of items.context_questions.filter(i => !i.scan_error)) {
731
+ lines.push(` • Phase ${item.phase}: ${item.file} (${item.question_count} question${item.question_count !== 1 ? 's' : ''})`);
732
+ for (const q of item.questions) {
733
+ lines.push(` - ${q}`);
734
+ }
735
+ }
736
+ }
737
+ lines.push('');
738
+ lines.push(hr);
739
+ lines.push(` ${counts.total} item${counts.total !== 1 ? 's' : ''} require decisions before close.`);
740
+ lines.push(hr);
741
+ return lines.join('\n');
742
+ }
743
+ module.exports = { auditOpenArtifacts, formatAuditReport };