@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,542 @@
1
+ 'use strict';
2
+
3
+ const { execFileSync } = require('node:child_process');
4
+ const { readdirSync, readFileSync, existsSync } = require('node:fs');
5
+ const path = require('node:path');
6
+
7
+ const { ExitError } = require('./lib/cli-exit.cjs');
8
+ const { suiteOf } = require('./run-tests.cjs');
9
+
10
+ const CRITICAL_PATHS = [
11
+ '.github/workflows/',
12
+ 'package.json',
13
+ 'package-lock.json',
14
+ 'scripts/run-tests.cjs',
15
+ 'scripts/affected-tests-lib.cjs',
16
+ 'scripts/run-affected-tests.cjs',
17
+ ];
18
+
19
+ // Suites that are push-only. PRs must never select or run these.
20
+ const PR_EXCLUDED_SUITES = new Set(['install', 'slow']);
21
+
22
+ // Suites run on every PR cell when the critical-path fallback fires.
23
+ const PR_FULL_SUITES = ['unit', 'integration', 'security'];
24
+
25
+ // Source trees to walk when building the forward graph (in addition to tests/).
26
+ // Relative to repoRoot. We walk these to discover SUT-internal requires so that
27
+ // a change to a deep helper propagates through re-export chains to tests.
28
+ const SOURCE_TREES = [
29
+ 'gsd-core/bin/lib',
30
+ 'bin/lib',
31
+ 'bin',
32
+ 'scripts',
33
+ 'commands',
34
+ 'hooks',
35
+ 'agents',
36
+ 'eslint-rules',
37
+ ];
38
+
39
+ function toPosixPath(input) {
40
+ return input.split(path.sep).join('/');
41
+ }
42
+
43
+ function parseRelativeSpecifiers(source) {
44
+ const specifiers = [];
45
+ const requireRe = /require\((['"])(.+?)\1\)/g;
46
+ const importFromRe = /from\s+(['"])(.+?)\1/g;
47
+ let match;
48
+
49
+ while ((match = requireRe.exec(source)) !== null) {
50
+ specifiers.push(match[2]);
51
+ }
52
+ while ((match = importFromRe.exec(source)) !== null) {
53
+ specifiers.push(match[2]);
54
+ }
55
+
56
+ return specifiers.filter(specifier => specifier.startsWith('.'));
57
+ }
58
+
59
+ // Extended candidate list now includes .ts/.cts/.mts/.json as well as the
60
+ // standard .js/.cjs/.mjs and index variants.
61
+ function resolveRelativeDependency(repoRoot, fromAbs, specifier) {
62
+ const base = path.resolve(path.dirname(fromAbs), specifier);
63
+ const candidates = [
64
+ base,
65
+ `${base}.js`,
66
+ `${base}.cjs`,
67
+ `${base}.mjs`,
68
+ `${base}.ts`,
69
+ `${base}.cts`,
70
+ `${base}.mts`,
71
+ `${base}.json`,
72
+ path.join(base, 'index.js'),
73
+ path.join(base, 'index.cjs'),
74
+ path.join(base, 'index.mjs'),
75
+ path.join(base, 'index.ts'),
76
+ ];
77
+
78
+ for (const candidate of candidates) {
79
+ if (existsSync(candidate)) {
80
+ return toPosixPath(path.relative(repoRoot, candidate));
81
+ }
82
+ }
83
+
84
+ return null;
85
+ }
86
+
87
+ // ---------------------------------------------------------------------------
88
+ // Source-file walker
89
+ // ---------------------------------------------------------------------------
90
+
91
+ /**
92
+ * Collect all .cjs / .mjs / .js / .ts / .cts / .mts / .json files under a
93
+ * directory tree, returned as repo-relative POSIX paths. Silently skips
94
+ * trees that don't exist.
95
+ */
96
+ function walkTree(repoRoot, relDir) {
97
+ const absDir = path.join(repoRoot, relDir);
98
+ if (!existsSync(absDir)) return [];
99
+
100
+ const results = [];
101
+ const queue = [absDir];
102
+
103
+ while (queue.length > 0) {
104
+ const cur = queue.shift();
105
+ let entries;
106
+ try {
107
+ entries = readdirSync(cur, { withFileTypes: true });
108
+ } catch {
109
+ continue;
110
+ }
111
+ for (const entry of entries) {
112
+ const abs = path.join(cur, entry.name);
113
+ if (entry.isDirectory()) {
114
+ // Skip node_modules
115
+ if (entry.name === 'node_modules') continue;
116
+ queue.push(abs);
117
+ } else if (entry.isFile()) {
118
+ const ext = path.extname(entry.name);
119
+ if (['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext)) {
120
+ results.push(toPosixPath(path.relative(repoRoot, abs)));
121
+ }
122
+ }
123
+ }
124
+ }
125
+
126
+ return results;
127
+ }
128
+
129
+ // ---------------------------------------------------------------------------
130
+ // Forward graph: Map<fileRel, Set<depRel>>
131
+ // ---------------------------------------------------------------------------
132
+
133
+ /**
134
+ * Build a forward dependency graph over test files PLUS source trees.
135
+ * For each file: read, parseRelativeSpecifiers, resolve each specifier.
136
+ * Returns Map<fileRel, Set<depRel>>.
137
+ */
138
+ function buildForwardGraph(repoRoot, testFiles) {
139
+ // Collect all files to index: test files + source files
140
+ const sourceFiles = [];
141
+ for (const tree of SOURCE_TREES) {
142
+ for (const f of walkTree(repoRoot, tree)) {
143
+ sourceFiles.push(f);
144
+ }
145
+ }
146
+
147
+ const allFiles = [...new Set([...testFiles, ...sourceFiles])];
148
+ const forward = new Map();
149
+
150
+ for (const fileRel of allFiles) {
151
+ const absFile = path.join(repoRoot, fileRel);
152
+ let source;
153
+ try {
154
+ source = readFileSync(absFile, 'utf8');
155
+ } catch {
156
+ continue;
157
+ }
158
+
159
+ const specs = parseRelativeSpecifiers(source);
160
+ const deps = new Set();
161
+
162
+ for (const specifier of specs) {
163
+ const dep = resolveRelativeDependency(repoRoot, absFile, specifier);
164
+ if (dep) deps.add(dep);
165
+ }
166
+
167
+ forward.set(fileRel, deps);
168
+ }
169
+
170
+ return forward;
171
+ }
172
+
173
+ // ---------------------------------------------------------------------------
174
+ // Reverse-transitive index: Map<depRel, Set<testRel>>
175
+ // ---------------------------------------------------------------------------
176
+
177
+ /**
178
+ * Build the TRANSITIVE reverse index: Map<depRel, Set<testRel>>.
179
+ *
180
+ * Algorithm:
181
+ * 1. Build forward graph over all test + source files.
182
+ * 2. Invert to direct reverse edges: Map<depRel, Set<dependentRel>>.
183
+ * 3. For each test file, BFS backwards through all direct reverse edges
184
+ * to find every ancestor. Map each ancestor → the test.
185
+ *
186
+ * Cycle safety: visited set per BFS — each node is enqueued at most once.
187
+ *
188
+ * @param {string} repoRoot
189
+ * @param {string[]} testFiles repo-relative posix paths (e.g. ['tests/foo.test.cjs'])
190
+ * @returns {Map<string, Set<string>>}
191
+ */
192
+ function buildTransitiveReverseIndex(repoRoot, testFiles) {
193
+ const forward = buildForwardGraph(repoRoot, testFiles);
194
+
195
+ // Build direct reverse edges: dep → Set of files that directly require dep
196
+ const directReverse = new Map();
197
+ for (const [fileRel, deps] of forward) {
198
+ for (const dep of deps) {
199
+ if (!directReverse.has(dep)) directReverse.set(dep, new Set());
200
+ directReverse.get(dep).add(fileRel);
201
+ }
202
+ }
203
+
204
+ // For each test file, BFS through direct reverse edges to collect all
205
+ // ancestors, then invert: ancestor → test.
206
+ // We do this test-file-first (not dep-first) so we know which test reached
207
+ // each ancestor.
208
+ const transitiveReverse = new Map();
209
+
210
+ for (const testFile of testFiles) {
211
+ // BFS from testFile following reverse edges (files that point TO testFile,
212
+ // then files that point to THOSE files, etc.).
213
+ // We want: "if X changed, would that eventually pull in testFile?"
214
+ // So we walk the FORWARD graph starting from testFile to find all deps,
215
+ // then any of those deps maps back to testFile.
216
+
217
+ // Actually simpler: for each test we do a forward BFS to find ALL files
218
+ // the test transitively depends on. Then we record testFile as a
219
+ // dependent of each of those files.
220
+ const visited = new Set();
221
+ visited.add(testFile);
222
+ const queue = [testFile];
223
+
224
+ while (queue.length > 0) {
225
+ const current = queue.shift();
226
+ const deps = forward.get(current);
227
+ if (!deps) continue;
228
+ for (const dep of deps) {
229
+ if (visited.has(dep)) continue;
230
+ visited.add(dep);
231
+ queue.push(dep);
232
+ }
233
+ }
234
+
235
+ // Every file in `visited` (except testFile itself) is a transitive dep.
236
+ // Record testFile as a dependent of each.
237
+ for (const dep of visited) {
238
+ if (dep === testFile) continue;
239
+ if (!transitiveReverse.has(dep)) transitiveReverse.set(dep, new Set());
240
+ transitiveReverse.get(dep).add(testFile);
241
+ }
242
+ }
243
+
244
+ return transitiveReverse;
245
+ }
246
+
247
+ // ---------------------------------------------------------------------------
248
+ // Legacy shim — kept so that runAffectedTests can call buildTransitiveReverseIndex
249
+ // and existing call sites that still call buildReverseIndex still work.
250
+ // ---------------------------------------------------------------------------
251
+ function buildReverseIndex(repoRoot, testFiles) {
252
+ return buildTransitiveReverseIndex(repoRoot, testFiles);
253
+ }
254
+
255
+ function shouldRunFullSuite(changedFiles) {
256
+ return changedFiles.some(file =>
257
+ CRITICAL_PATHS.some(critical => file === critical || file.startsWith(critical)),
258
+ );
259
+ }
260
+
261
+ function listTestFiles(repoRoot) {
262
+ return readdirSync(path.join(repoRoot, 'tests'))
263
+ .filter(file => file.endsWith('.test.cjs'))
264
+ .map(file => `tests/${file}`)
265
+ .sort();
266
+ }
267
+
268
+ /**
269
+ * Select the affected tests given a set of changed files and a reverse index.
270
+ *
271
+ * Options:
272
+ * detectWiden {boolean} — when true, attach `._widenRequired = true` to the
273
+ * returned array when a changed source file has zero transitive test
274
+ * dependents. The caller (runAffectedTests) uses this to widen to unit/all.
275
+ *
276
+ * The returned array is sorted and may have `._widenRequired` attached.
277
+ */
278
+ function pickAffectedTests(changedFiles, allTests, reverseIndex, options = {}) {
279
+ const { detectWiden = false } = options;
280
+ const selected = new Set();
281
+ let widenRequired = false;
282
+
283
+ // Build a fast lookup of currently-existing test files (from readdirSync — deleted files absent).
284
+ const allTestsSet = new Set(allTests);
285
+
286
+ // (a) directly-changed test files + (b) transitive test dependents
287
+ // Deleted test files are filtered out — they no longer exist and cannot be run.
288
+ // A deleted test file also must NOT trigger widen (the test is simply gone).
289
+ for (const file of changedFiles) {
290
+ if (file.startsWith('tests/') && file.endsWith('.test.cjs')) {
291
+ // Only select if the test file still exists (i.e. is present in allTests from readdirSync).
292
+ if (allTestsSet.has(file)) {
293
+ selected.add(file);
294
+ }
295
+ // Deleted test file — do not add to selected; do not look up reverse index.
296
+ } else {
297
+ const dependents = reverseIndex.get(file);
298
+ if (dependents) {
299
+ for (const testFile of dependents) selected.add(testFile);
300
+ }
301
+ }
302
+ }
303
+
304
+ // (c) stem heuristic — kept as secondary mechanism
305
+ for (const file of changedFiles) {
306
+ const stem = path.basename(file).replace(/\.[^.]+$/, '').toLowerCase();
307
+ if (!stem) continue;
308
+ for (const testFile of allTests) {
309
+ if (testFile.toLowerCase().includes(stem)) selected.add(testFile);
310
+ }
311
+ }
312
+
313
+ // Widen backstop: if a changed file is a non-test, non-CRITICAL_PATH source file
314
+ // (recognised extension) under a SOURCE_TREE, AND it is either deleted (no longer
315
+ // on disk — so never in the forward graph and has no static dependents) OR it
316
+ // exists with ZERO transitive test dependents — signal a widen.
317
+ // NOTE: we deliberately do NOT skip deleted files here; a deleted source file's
318
+ // absence from the forward graph means dependents===undefined, which is the same
319
+ // as zero static dependents, and is itself the widen trigger.
320
+ if (detectWiden) {
321
+ for (const file of changedFiles) {
322
+ // Only care about source files, not test files or docs
323
+ if (file.startsWith('tests/')) continue;
324
+ if (shouldRunFullSuite([file])) continue; // critical path already triggers full suite
325
+ // Check: is this a source file (has a recognised extension)?
326
+ const ext = path.extname(file);
327
+ const isSourceFile = ['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext);
328
+ if (!isSourceFile) continue;
329
+ // Check: is this file under a recognised source tree?
330
+ const isUnderSourceTree = SOURCE_TREES.some(
331
+ tree => file === tree || file.startsWith(tree + '/'),
332
+ );
333
+ if (!isUnderSourceTree) continue;
334
+ // Does it have any test dependents?
335
+ // A deleted file will have undefined here (not in the graph) — that is
336
+ // treated as zero static dependents and triggers widen conservatively.
337
+ const dependents = reverseIndex.get(file);
338
+ const hasStaticDependents = dependents && dependents.size > 0;
339
+ if (!hasStaticDependents) {
340
+ widenRequired = true;
341
+ break;
342
+ }
343
+ }
344
+ }
345
+
346
+ // Drop any file whose suite is push-only. This is the single chokepoint —
347
+ // it catches direct-change, reverse-index, AND stem-match selections.
348
+ for (const file of selected) {
349
+ const suite = suiteOf(path.basename(file));
350
+ if (PR_EXCLUDED_SUITES.has(suite)) selected.delete(file);
351
+ }
352
+
353
+ // When nothing maps, return an empty array. The caller decides the fallback.
354
+ const result = [...selected].sort();
355
+ if (widenRequired) result._widenRequired = true;
356
+ return result;
357
+ }
358
+
359
+ function changedFilesSinceBase(repoRoot, baseRef) {
360
+ const out = execFileSync(
361
+ 'git',
362
+ ['diff', '--name-only', '--no-renames', '--diff-filter=ACMRD', `${baseRef}...HEAD`],
363
+ { cwd: repoRoot, encoding: 'utf8' },
364
+ ).trim();
365
+ if (!out) return [];
366
+ return out.split('\n').map(line => line.trim()).filter(Boolean);
367
+ }
368
+
369
+ function runNodeTestFiles(repoRoot, files) {
370
+ const defaultConcurrency = process.platform === 'win32' ? 2 : 4;
371
+ const concurrency = process.env.TEST_CONCURRENCY
372
+ ? `--test-concurrency=${process.env.TEST_CONCURRENCY}`
373
+ : `--test-concurrency=${defaultConcurrency}`;
374
+ const absoluteFiles = files.map(file => path.join(repoRoot, file));
375
+
376
+ // Keep chunks bounded for Windows CreateProcess command-length limits.
377
+ const maxChars = process.env.RUN_TESTS_MAX_CMDLINE_CHARS
378
+ ? Number(process.env.RUN_TESTS_MAX_CMDLINE_CHARS)
379
+ : 28000;
380
+ const fixed = process.execPath.length + '--test'.length + concurrency.length + 8;
381
+ const chunks = [];
382
+ let current = [];
383
+ let currentLen = fixed;
384
+
385
+ for (const file of absoluteFiles) {
386
+ const add = file.length + 1;
387
+ if (current.length > 0 && currentLen + add > maxChars) {
388
+ chunks.push(current);
389
+ current = [];
390
+ currentLen = fixed;
391
+ }
392
+ current.push(file);
393
+ currentLen += add;
394
+ }
395
+ if (current.length > 0) chunks.push(current);
396
+
397
+ let firstFailure = 0;
398
+ for (let i = 0; i < chunks.length; i++) {
399
+ if (chunks.length > 1) {
400
+ console.error(`affected-tests: chunk ${i + 1}/${chunks.length} (${chunks[i].length} files)`);
401
+ }
402
+ try {
403
+ execFileSync(process.execPath, ['--test', concurrency, ...chunks[i]], {
404
+ cwd: repoRoot,
405
+ stdio: 'inherit',
406
+ env: { ...process.env },
407
+ });
408
+ } catch (error) {
409
+ const code = error.status || 1;
410
+ if (firstFailure === 0) firstFailure = code;
411
+ }
412
+ }
413
+ if (firstFailure !== 0) throw new ExitError(firstFailure);
414
+ }
415
+
416
+ function runSuite(repoRoot, suite) {
417
+ execFileSync(process.execPath, ['scripts/run-tests.cjs', '--suite', suite], {
418
+ cwd: repoRoot,
419
+ stdio: 'inherit',
420
+ env: { ...process.env },
421
+ });
422
+ }
423
+
424
+ function resolveBaseRef() {
425
+ if (process.env.GSD_AFFECTED_BASE) return process.env.GSD_AFFECTED_BASE;
426
+ if (process.env.GITHUB_BASE_REF) return `origin/${process.env.GITHUB_BASE_REF}`;
427
+ return 'origin/main';
428
+ }
429
+
430
+ /**
431
+ * Pure function: given the outputs of the selection phase, return a run plan
432
+ * describing what should be executed. No I/O is performed here.
433
+ *
434
+ * Return shapes:
435
+ * { mode: 'suite', suite: 'unit' } — no changed files
436
+ * { mode: 'suites', suites: PR_FULL_SUITES } — critical path triggered
437
+ * { mode: 'suites', suites: PR_FULL_SUITES } — widen required (orphan src file)
438
+ * { mode: 'suite', suite: 'unit' } — selection empty after widen=false
439
+ * { mode: 'files', files: string[] } — concrete selection, no widen
440
+ *
441
+ * Invariant: when widenRequired is true the executed set is ALWAYS ⊇ selected,
442
+ * because PR_FULL_SUITES covers every PR-eligible suite (unit + integration +
443
+ * security), so every concrete match that pickAffectedTests put into `selected`
444
+ * belongs to one of those suites and will be exercised by running all three.
445
+ */
446
+ function resolveRunPlan({ changedFiles: _changedFiles, selected, widenRequired, criticalPath, noChanges }) {
447
+ if (noChanges) {
448
+ return { mode: 'suite', suite: 'unit' };
449
+ }
450
+ if (criticalPath) {
451
+ return { mode: 'suites', suites: PR_FULL_SUITES };
452
+ }
453
+ if (widenRequired) {
454
+ return { mode: 'suites', suites: PR_FULL_SUITES };
455
+ }
456
+ if (selected.length === 0) {
457
+ return { mode: 'suite', suite: 'unit' };
458
+ }
459
+ return { mode: 'files', files: selected };
460
+ }
461
+
462
+ function runAffectedTests(options = {}) {
463
+ const repoRoot = options.repoRoot || path.resolve(__dirname, '..');
464
+ const baseRef = options.baseRef || resolveBaseRef();
465
+ const changed = changedFilesSinceBase(repoRoot, baseRef);
466
+
467
+ if (changed.length === 0) {
468
+ console.error(`affected-tests: no changed files against ${baseRef}; running unit suite`);
469
+ runSuite(repoRoot, 'unit');
470
+ return;
471
+ }
472
+
473
+ if (shouldRunFullSuite(changed)) {
474
+ console.error('affected-tests: critical CI/runtime files changed; running PR suites (unit, integration, security)');
475
+ for (const suite of PR_FULL_SUITES) {
476
+ runSuite(repoRoot, suite);
477
+ }
478
+ return;
479
+ }
480
+
481
+ const allTests = listTestFiles(repoRoot);
482
+ const reverseIndex = buildTransitiveReverseIndex(repoRoot, allTests);
483
+ const selected = pickAffectedTests(changed, allTests, reverseIndex, { detectWiden: true });
484
+
485
+ console.error(`affected-tests: base=${baseRef} changed=${changed.length} selected=${selected.length}`);
486
+ console.error(`affected-tests: ${selected.join(' ')}`);
487
+
488
+ const plan = resolveRunPlan({
489
+ changedFiles: changed,
490
+ selected,
491
+ widenRequired: selected._widenRequired === true,
492
+ criticalPath: false,
493
+ noChanges: false,
494
+ });
495
+
496
+ if (plan.mode === 'suites') {
497
+ // Widen backstop: a source file changed that has no static test dependents.
498
+ // Run all PR suites (unit + integration + security) — a strict superset of
499
+ // the concretely-selected tests — so no integration/security match is lost.
500
+ for (const file of changed) {
501
+ const ext = path.extname(file);
502
+ const isSourceFile = ['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext);
503
+ if (!isSourceFile || file.startsWith('tests/') || shouldRunFullSuite([file])) continue;
504
+ const dependents = reverseIndex.get(file);
505
+ if (!dependents || dependents.size === 0) {
506
+ console.error(
507
+ `affected-tests: ${file} has no static test dependents; widening to PR suites (unit+integration+security)`,
508
+ );
509
+ }
510
+ }
511
+ for (const suite of plan.suites) {
512
+ runSuite(repoRoot, suite);
513
+ }
514
+ return;
515
+ }
516
+
517
+ if (plan.mode === 'suite') {
518
+ console.error('affected-tests: no affected tests found; running unit suite as smoke');
519
+ runSuite(repoRoot, plan.suite);
520
+ return;
521
+ }
522
+
523
+ // plan.mode === 'files'
524
+ runNodeTestFiles(repoRoot, plan.files);
525
+ }
526
+
527
+ module.exports = {
528
+ CRITICAL_PATHS,
529
+ PR_EXCLUDED_SUITES,
530
+ PR_FULL_SUITES,
531
+ buildForwardGraph,
532
+ buildReverseIndex,
533
+ buildTransitiveReverseIndex,
534
+ parseRelativeSpecifiers,
535
+ pickAffectedTests,
536
+ resolveBaseRef,
537
+ resolveRelativeDependency,
538
+ resolveRunPlan,
539
+ shouldRunFullSuite,
540
+ toPosixPath,
541
+ runAffectedTests,
542
+ };
@@ -0,0 +1,73 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Post-install path audit for workflow-invoked scripts (#2995).
5
+ *
6
+ * Walks workflowsDir, extracts every `${GSD_HOME[...]}/<path>.<cjs|js|sh>`
7
+ * token, and asserts:
8
+ * 1. the file exists in the repo at that <path> (catches typos)
9
+ * 2. <path>'s first segment is in installedPrefixes (catches the
10
+ * #2994 class: source-vs-deployed-path mismatches)
11
+ *
12
+ * Pure function over (workflowsDir, repoRoot, installedPrefixes); no
13
+ * filesystem mutation. Tests assert on the typed AUDIT_FINDING enum.
14
+ */
15
+
16
+ const fs = require('node:fs');
17
+ const path = require('node:path');
18
+
19
+ const AUDIT_FINDING = Object.freeze({
20
+ MISSING_FROM_REPO: 'missing_from_repo',
21
+ NOT_INSTALLED: 'not_installed',
22
+ });
23
+
24
+ // Match `${GSD_HOME}` or `${GSD_HOME:-...}` followed by a /-rooted path
25
+ // ending in .cjs/.js/.sh. The path is captured verbatim (relative to
26
+ // the install root).
27
+ const REF_RE = /\$\{GSD_HOME(?::-[^}]*)?\}\/([A-Za-z0-9_./-]+\.(?:cjs|js|sh))/g;
28
+
29
+ function listWorkflowFiles(dir) {
30
+ if (!fs.existsSync(dir)) return [];
31
+ return fs
32
+ .readdirSync(dir, { withFileTypes: true })
33
+ .filter((e) => e.isFile() && e.name.endsWith('.md'))
34
+ .map((e) => path.join(dir, e.name));
35
+ }
36
+
37
+ function extractReferences(content) {
38
+ const out = [];
39
+ let m;
40
+ // RegExp objects with /g state must be reset per call.
41
+ const re = new RegExp(REF_RE.source, 'g');
42
+ while ((m = re.exec(content)) !== null) {
43
+ out.push(m[1]);
44
+ }
45
+ return out;
46
+ }
47
+
48
+ function auditWorkflowScriptPaths({ workflowsDir, repoRoot, installedPrefixes }) {
49
+ const findings = [];
50
+ const installedSet = new Set(installedPrefixes);
51
+ for (const file of listWorkflowFiles(workflowsDir)) {
52
+ const content = fs.readFileSync(file, 'utf8');
53
+ const workflow = path.basename(file);
54
+ for (const ref of extractReferences(content)) {
55
+ const firstSegment = ref.split('/')[0];
56
+ // #2996 CR: emit BOTH findings simultaneously when a reference is
57
+ // both outside an installed prefix AND missing from the repo. The
58
+ // earlier `continue` short-circuited MISSING_FROM_REPO, so a
59
+ // developer who moved a missing reference to an installed prefix
60
+ // would only discover the second issue on a subsequent CI run.
61
+ if (!installedSet.has(firstSegment)) {
62
+ findings.push({ workflow, path: ref, kind: AUDIT_FINDING.NOT_INSTALLED });
63
+ }
64
+ const sourceFile = path.join(repoRoot, ref);
65
+ if (!fs.existsSync(sourceFile)) {
66
+ findings.push({ workflow, path: ref, kind: AUDIT_FINDING.MISSING_FROM_REPO });
67
+ }
68
+ }
69
+ }
70
+ return { ok: findings.length === 0, findings };
71
+ }
72
+
73
+ module.exports = { auditWorkflowScriptPaths, AUDIT_FINDING, extractReferences };