@opengsd/gsd-core 1.2.0-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (503) hide show
  1. package/LICENSE +21 -0
  2. package/README.ja-JP.md +870 -0
  3. package/README.ko-KR.md +861 -0
  4. package/README.md +301 -0
  5. package/README.pt-BR.md +492 -0
  6. package/README.zh-CN.md +842 -0
  7. package/agents/gsd-advisor-researcher.md +127 -0
  8. package/agents/gsd-ai-researcher.md +133 -0
  9. package/agents/gsd-assumptions-analyzer.md +105 -0
  10. package/agents/gsd-code-fixer.md +668 -0
  11. package/agents/gsd-code-reviewer.md +387 -0
  12. package/agents/gsd-codebase-mapper.md +853 -0
  13. package/agents/gsd-debug-session-manager.md +314 -0
  14. package/agents/gsd-debugger.md +1452 -0
  15. package/agents/gsd-doc-classifier.md +168 -0
  16. package/agents/gsd-doc-synthesizer.md +204 -0
  17. package/agents/gsd-doc-verifier.md +217 -0
  18. package/agents/gsd-doc-writer.md +615 -0
  19. package/agents/gsd-domain-researcher.md +153 -0
  20. package/agents/gsd-eval-auditor.md +191 -0
  21. package/agents/gsd-eval-planner.md +154 -0
  22. package/agents/gsd-executor.md +772 -0
  23. package/agents/gsd-framework-selector.md +160 -0
  24. package/agents/gsd-integration-checker.md +470 -0
  25. package/agents/gsd-intel-updater.md +342 -0
  26. package/agents/gsd-nyquist-auditor.md +203 -0
  27. package/agents/gsd-pattern-mapper.md +335 -0
  28. package/agents/gsd-phase-researcher.md +928 -0
  29. package/agents/gsd-plan-checker.md +978 -0
  30. package/agents/gsd-planner.md +1218 -0
  31. package/agents/gsd-project-researcher.md +677 -0
  32. package/agents/gsd-research-synthesizer.md +255 -0
  33. package/agents/gsd-roadmapper.md +688 -0
  34. package/agents/gsd-security-auditor.md +155 -0
  35. package/agents/gsd-ui-auditor.md +495 -0
  36. package/agents/gsd-ui-checker.md +309 -0
  37. package/agents/gsd-ui-researcher.md +380 -0
  38. package/agents/gsd-user-profiler.md +171 -0
  39. package/agents/gsd-verifier.md +917 -0
  40. package/bin/install.js +10936 -0
  41. package/bin/lib/ui-safety-gate.cjs +107 -0
  42. package/commands/gsd/add-tests.md +42 -0
  43. package/commands/gsd/ai-integration-phase.md +37 -0
  44. package/commands/gsd/audit-fix.md +34 -0
  45. package/commands/gsd/audit-milestone.md +37 -0
  46. package/commands/gsd/audit-uat.md +24 -0
  47. package/commands/gsd/autonomous.md +46 -0
  48. package/commands/gsd/capture.md +62 -0
  49. package/commands/gsd/cleanup.md +24 -0
  50. package/commands/gsd/code-review.md +59 -0
  51. package/commands/gsd/complete-milestone.md +143 -0
  52. package/commands/gsd/config.md +56 -0
  53. package/commands/gsd/debug.md +52 -0
  54. package/commands/gsd/discuss-phase.md +76 -0
  55. package/commands/gsd/docs-update.md +49 -0
  56. package/commands/gsd/eval-review.md +33 -0
  57. package/commands/gsd/execute-phase.md +64 -0
  58. package/commands/gsd/explore.md +27 -0
  59. package/commands/gsd/extract-learnings.md +23 -0
  60. package/commands/gsd/fast.md +31 -0
  61. package/commands/gsd/forensics.md +57 -0
  62. package/commands/gsd/graphify.md +199 -0
  63. package/commands/gsd/health.md +31 -0
  64. package/commands/gsd/help.md +28 -0
  65. package/commands/gsd/import.md +41 -0
  66. package/commands/gsd/inbox.md +39 -0
  67. package/commands/gsd/ingest-docs.md +42 -0
  68. package/commands/gsd/manager.md +45 -0
  69. package/commands/gsd/map-codebase.md +83 -0
  70. package/commands/gsd/milestone-summary.md +51 -0
  71. package/commands/gsd/mvp-phase.md +45 -0
  72. package/commands/gsd/new-milestone.md +45 -0
  73. package/commands/gsd/new-project.md +47 -0
  74. package/commands/gsd/ns-context.md +23 -0
  75. package/commands/gsd/ns-ideate.md +24 -0
  76. package/commands/gsd/ns-manage.md +29 -0
  77. package/commands/gsd/ns-project.md +22 -0
  78. package/commands/gsd/ns-review.md +26 -0
  79. package/commands/gsd/ns-workflow.md +28 -0
  80. package/commands/gsd/pause-work.md +43 -0
  81. package/commands/gsd/phase.md +56 -0
  82. package/commands/gsd/plan-phase.md +62 -0
  83. package/commands/gsd/plan-review-convergence.md +59 -0
  84. package/commands/gsd/pr-branch.md +26 -0
  85. package/commands/gsd/profile-user.md +46 -0
  86. package/commands/gsd/progress.md +47 -0
  87. package/commands/gsd/quick.md +174 -0
  88. package/commands/gsd/resume-work.md +30 -0
  89. package/commands/gsd/review-backlog.md +63 -0
  90. package/commands/gsd/review.md +41 -0
  91. package/commands/gsd/secure-phase.md +36 -0
  92. package/commands/gsd/settings.md +29 -0
  93. package/commands/gsd/ship.md +24 -0
  94. package/commands/gsd/sketch.md +60 -0
  95. package/commands/gsd/spec-phase.md +63 -0
  96. package/commands/gsd/spike.md +57 -0
  97. package/commands/gsd/stats.md +19 -0
  98. package/commands/gsd/surface.md +155 -0
  99. package/commands/gsd/thread.md +24 -0
  100. package/commands/gsd/ui-phase.md +35 -0
  101. package/commands/gsd/ui-review.md +33 -0
  102. package/commands/gsd/ultraplan-phase.md +34 -0
  103. package/commands/gsd/undo.md +35 -0
  104. package/commands/gsd/update.md +48 -0
  105. package/commands/gsd/validate-phase.md +36 -0
  106. package/commands/gsd/verify-work.md +39 -0
  107. package/commands/gsd/workspace.md +52 -0
  108. package/commands/gsd/workstreams.md +70 -0
  109. package/get-shit-done/bin/check-latest-version.cjs +106 -0
  110. package/get-shit-done/bin/gsd-tools.cjs +1676 -0
  111. package/get-shit-done/bin/lib/active-workstream-store.cjs +302 -0
  112. package/get-shit-done/bin/lib/adr-parser.cjs +394 -0
  113. package/get-shit-done/bin/lib/agent-command-router.cjs +65 -0
  114. package/get-shit-done/bin/lib/artifacts.cjs +53 -0
  115. package/get-shit-done/bin/lib/audit.cjs +755 -0
  116. package/get-shit-done/bin/lib/check-command-router.cjs +333 -0
  117. package/get-shit-done/bin/lib/cjs-command-router-adapter.cjs +118 -0
  118. package/get-shit-done/bin/lib/clock.cjs +96 -0
  119. package/get-shit-done/bin/lib/clusters.cjs +135 -0
  120. package/get-shit-done/bin/lib/code-review-flags.cjs +74 -0
  121. package/get-shit-done/bin/lib/command-aliases.cjs +815 -0
  122. package/get-shit-done/bin/lib/command-arg-projection.cjs +62 -0
  123. package/get-shit-done/bin/lib/command-routing-hub.cjs +388 -0
  124. package/get-shit-done/bin/lib/commands.cjs +1188 -0
  125. package/get-shit-done/bin/lib/config-schema.cjs +31 -0
  126. package/get-shit-done/bin/lib/config.cjs +728 -0
  127. package/get-shit-done/bin/lib/configuration.cjs +248 -0
  128. package/get-shit-done/bin/lib/context-utilization.cjs +47 -0
  129. package/get-shit-done/bin/lib/core.cjs +2121 -0
  130. package/get-shit-done/bin/lib/decisions.cjs +116 -0
  131. package/get-shit-done/bin/lib/docs.cjs +270 -0
  132. package/get-shit-done/bin/lib/drift.cjs +388 -0
  133. package/get-shit-done/bin/lib/fallow-runner.cjs +109 -0
  134. package/get-shit-done/bin/lib/frontmatter.cjs +389 -0
  135. package/get-shit-done/bin/lib/gap-checker.cjs +205 -0
  136. package/get-shit-done/bin/lib/graphify.cjs +592 -0
  137. package/get-shit-done/bin/lib/gsd2-import.cjs +514 -0
  138. package/get-shit-done/bin/lib/init-command-router.cjs +58 -0
  139. package/get-shit-done/bin/lib/init.cjs +2112 -0
  140. package/get-shit-done/bin/lib/install-profiles.cjs +603 -0
  141. package/get-shit-done/bin/lib/installer-migration-authoring.cjs +117 -0
  142. package/get-shit-done/bin/lib/installer-migration-report.cjs +354 -0
  143. package/get-shit-done/bin/lib/installer-migrations/000-first-time-baseline.cjs +220 -0
  144. package/get-shit-done/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +41 -0
  145. package/get-shit-done/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +80 -0
  146. package/get-shit-done/bin/lib/installer-migrations.cjs +778 -0
  147. package/get-shit-done/bin/lib/intel.cjs +708 -0
  148. package/get-shit-done/bin/lib/learnings.cjs +421 -0
  149. package/get-shit-done/bin/lib/milestone.cjs +314 -0
  150. package/get-shit-done/bin/lib/model-catalog.cjs +212 -0
  151. package/get-shit-done/bin/lib/model-profiles.cjs +31 -0
  152. package/get-shit-done/bin/lib/observability/event.cjs +82 -0
  153. package/get-shit-done/bin/lib/observability/logger.cjs +174 -0
  154. package/get-shit-done/bin/lib/observability/redaction.cjs +50 -0
  155. package/get-shit-done/bin/lib/package-identity.cjs +31 -0
  156. package/get-shit-done/bin/lib/phase-command-router.cjs +191 -0
  157. package/get-shit-done/bin/lib/phase-lifecycle.cjs +80 -0
  158. package/get-shit-done/bin/lib/phase.cjs +1607 -0
  159. package/get-shit-done/bin/lib/phases-command-router.cjs +39 -0
  160. package/get-shit-done/bin/lib/plan-scan.cjs +97 -0
  161. package/get-shit-done/bin/lib/planning-workspace.cjs +238 -0
  162. package/get-shit-done/bin/lib/profile-output.cjs +1141 -0
  163. package/get-shit-done/bin/lib/profile-pipeline.cjs +539 -0
  164. package/get-shit-done/bin/lib/project-root.cjs +112 -0
  165. package/get-shit-done/bin/lib/prompt-budget.cjs +399 -0
  166. package/get-shit-done/bin/lib/review-reviewer-selection.cjs +125 -0
  167. package/get-shit-done/bin/lib/roadmap-command-router.cjs +28 -0
  168. package/get-shit-done/bin/lib/roadmap.cjs +650 -0
  169. package/get-shit-done/bin/lib/runtime-artifact-layout.cjs +301 -0
  170. package/get-shit-done/bin/lib/runtime-homes.cjs +222 -0
  171. package/get-shit-done/bin/lib/runtime-name-policy.cjs +83 -0
  172. package/get-shit-done/bin/lib/runtime-slash.cjs +112 -0
  173. package/get-shit-done/bin/lib/schema-detect.cjs +165 -0
  174. package/get-shit-done/bin/lib/secrets.cjs +32 -0
  175. package/get-shit-done/bin/lib/security.cjs +600 -0
  176. package/get-shit-done/bin/lib/semver-compare.cjs +35 -0
  177. package/get-shit-done/bin/lib/shell-command-projection.cjs +500 -0
  178. package/get-shit-done/bin/lib/state-command-router.cjs +252 -0
  179. package/get-shit-done/bin/lib/state-document.cjs +263 -0
  180. package/get-shit-done/bin/lib/state.cjs +2038 -0
  181. package/get-shit-done/bin/lib/surface.cjs +470 -0
  182. package/get-shit-done/bin/lib/task-command-router.cjs +81 -0
  183. package/get-shit-done/bin/lib/template.cjs +228 -0
  184. package/get-shit-done/bin/lib/uat.cjs +289 -0
  185. package/get-shit-done/bin/lib/update-context.cjs +209 -0
  186. package/get-shit-done/bin/lib/validate-command-router.cjs +83 -0
  187. package/get-shit-done/bin/lib/validate.cjs +92 -0
  188. package/get-shit-done/bin/lib/verify-command-router.cjs +40 -0
  189. package/get-shit-done/bin/lib/verify.cjs +1511 -0
  190. package/get-shit-done/bin/lib/workstream-inventory-builder.cjs +74 -0
  191. package/get-shit-done/bin/lib/workstream-inventory.cjs +146 -0
  192. package/get-shit-done/bin/lib/workstream-name-policy.cjs +94 -0
  193. package/get-shit-done/bin/lib/workstream.cjs +389 -0
  194. package/get-shit-done/bin/lib/worktree-safety.cjs +985 -0
  195. package/get-shit-done/bin/shared/config-defaults.manifest.json +97 -0
  196. package/get-shit-done/bin/shared/config-schema.manifest.json +175 -0
  197. package/get-shit-done/bin/shared/model-catalog.json +122 -0
  198. package/get-shit-done/bin/shared/runtime-aliases.manifest.json +75 -0
  199. package/get-shit-done/bin/verify-reapply-patches.cjs +352 -0
  200. package/get-shit-done/contexts/dev.md +21 -0
  201. package/get-shit-done/contexts/research.md +22 -0
  202. package/get-shit-done/contexts/review.md +23 -0
  203. package/get-shit-done/references/agent-contracts.md +79 -0
  204. package/get-shit-done/references/ai-evals.md +156 -0
  205. package/get-shit-done/references/ai-frameworks.md +186 -0
  206. package/get-shit-done/references/artifact-types.md +131 -0
  207. package/get-shit-done/references/autonomous-smart-discuss.md +277 -0
  208. package/get-shit-done/references/checkpoints.md +814 -0
  209. package/get-shit-done/references/common-bug-patterns.md +114 -0
  210. package/get-shit-done/references/context-budget.md +85 -0
  211. package/get-shit-done/references/continuation-format.md +253 -0
  212. package/get-shit-done/references/debugger-philosophy.md +76 -0
  213. package/get-shit-done/references/decimal-phase-calculation.md +64 -0
  214. package/get-shit-done/references/doc-conflict-engine.md +91 -0
  215. package/get-shit-done/references/domain-probes.md +125 -0
  216. package/get-shit-done/references/execute-mvp-tdd.md +81 -0
  217. package/get-shit-done/references/executor-examples.md +110 -0
  218. package/get-shit-done/references/few-shot-examples/plan-checker.md +73 -0
  219. package/get-shit-done/references/few-shot-examples/verifier.md +109 -0
  220. package/get-shit-done/references/gate-prompts.md +100 -0
  221. package/get-shit-done/references/gates.md +70 -0
  222. package/get-shit-done/references/git-integration.md +298 -0
  223. package/get-shit-done/references/git-planning-commit.md +40 -0
  224. package/get-shit-done/references/ios-scaffold.md +123 -0
  225. package/get-shit-done/references/mandatory-initial-read.md +2 -0
  226. package/get-shit-done/references/model-profile-resolution.md +38 -0
  227. package/get-shit-done/references/model-profiles.md +245 -0
  228. package/get-shit-done/references/mvp-concepts.md +49 -0
  229. package/get-shit-done/references/phase-argument-parsing.md +61 -0
  230. package/get-shit-done/references/planner-antipatterns.md +89 -0
  231. package/get-shit-done/references/planner-chunked.md +49 -0
  232. package/get-shit-done/references/planner-gap-closure.md +62 -0
  233. package/get-shit-done/references/planner-graphify-auto-update.md +67 -0
  234. package/get-shit-done/references/planner-human-verify-mode.md +57 -0
  235. package/get-shit-done/references/planner-interface-context.md +62 -0
  236. package/get-shit-done/references/planner-mvp-mode.md +53 -0
  237. package/get-shit-done/references/planner-reviews.md +39 -0
  238. package/get-shit-done/references/planner-revision.md +87 -0
  239. package/get-shit-done/references/planner-source-audit.md +73 -0
  240. package/get-shit-done/references/planning-config.md +471 -0
  241. package/get-shit-done/references/project-skills-discovery.md +19 -0
  242. package/get-shit-done/references/questioning.md +162 -0
  243. package/get-shit-done/references/revision-loop.md +97 -0
  244. package/get-shit-done/references/scout-codebase.md +51 -0
  245. package/get-shit-done/references/skeleton-template.md +48 -0
  246. package/get-shit-done/references/sketch-interactivity.md +41 -0
  247. package/get-shit-done/references/sketch-theme-system.md +94 -0
  248. package/get-shit-done/references/sketch-tooling.md +45 -0
  249. package/get-shit-done/references/sketch-variant-patterns.md +81 -0
  250. package/get-shit-done/references/spidr-splitting.md +69 -0
  251. package/get-shit-done/references/tdd.md +330 -0
  252. package/get-shit-done/references/thinking-models-debug.md +44 -0
  253. package/get-shit-done/references/thinking-models-execution.md +50 -0
  254. package/get-shit-done/references/thinking-models-planning.md +62 -0
  255. package/get-shit-done/references/thinking-models-research.md +50 -0
  256. package/get-shit-done/references/thinking-models-verification.md +55 -0
  257. package/get-shit-done/references/thinking-partner.md +96 -0
  258. package/get-shit-done/references/ui-brand.md +160 -0
  259. package/get-shit-done/references/universal-anti-patterns.md +63 -0
  260. package/get-shit-done/references/user-profiling.md +681 -0
  261. package/get-shit-done/references/user-story-template.md +58 -0
  262. package/get-shit-done/references/verification-overrides.md +227 -0
  263. package/get-shit-done/references/verification-patterns.md +612 -0
  264. package/get-shit-done/references/verify-mvp-mode.md +85 -0
  265. package/get-shit-done/references/workstream-flag.md +111 -0
  266. package/get-shit-done/references/worktree-path-safety.md +89 -0
  267. package/get-shit-done/templates/AI-SPEC.md +246 -0
  268. package/get-shit-done/templates/DEBUG.md +169 -0
  269. package/get-shit-done/templates/README.md +77 -0
  270. package/get-shit-done/templates/SECURITY.md +61 -0
  271. package/get-shit-done/templates/UAT.md +265 -0
  272. package/get-shit-done/templates/UI-SPEC.md +100 -0
  273. package/get-shit-done/templates/VALIDATION.md +76 -0
  274. package/get-shit-done/templates/claude-md.md +145 -0
  275. package/get-shit-done/templates/codebase/architecture.md +255 -0
  276. package/get-shit-done/templates/codebase/concerns.md +310 -0
  277. package/get-shit-done/templates/codebase/conventions.md +307 -0
  278. package/get-shit-done/templates/codebase/integrations.md +280 -0
  279. package/get-shit-done/templates/codebase/stack.md +186 -0
  280. package/get-shit-done/templates/codebase/structure.md +285 -0
  281. package/get-shit-done/templates/codebase/testing.md +480 -0
  282. package/get-shit-done/templates/config.json +62 -0
  283. package/get-shit-done/templates/context.md +352 -0
  284. package/get-shit-done/templates/continue-here.md +78 -0
  285. package/get-shit-done/templates/copilot-instructions.md +7 -0
  286. package/get-shit-done/templates/debug-subagent-prompt.md +91 -0
  287. package/get-shit-done/templates/dev-preferences.md +21 -0
  288. package/get-shit-done/templates/discovery.md +146 -0
  289. package/get-shit-done/templates/discussion-log.md +63 -0
  290. package/get-shit-done/templates/milestone-archive.md +123 -0
  291. package/get-shit-done/templates/milestone.md +115 -0
  292. package/get-shit-done/templates/phase-prompt.md +610 -0
  293. package/get-shit-done/templates/planner-subagent-prompt.md +117 -0
  294. package/get-shit-done/templates/project.md +186 -0
  295. package/get-shit-done/templates/requirements.md +231 -0
  296. package/get-shit-done/templates/research-project/ARCHITECTURE.md +204 -0
  297. package/get-shit-done/templates/research-project/FEATURES.md +147 -0
  298. package/get-shit-done/templates/research-project/PITFALLS.md +200 -0
  299. package/get-shit-done/templates/research-project/STACK.md +120 -0
  300. package/get-shit-done/templates/research-project/SUMMARY.md +170 -0
  301. package/get-shit-done/templates/research.md +592 -0
  302. package/get-shit-done/templates/retrospective.md +54 -0
  303. package/get-shit-done/templates/roadmap.md +202 -0
  304. package/get-shit-done/templates/spec.md +307 -0
  305. package/get-shit-done/templates/state.md +195 -0
  306. package/get-shit-done/templates/summary-complex.md +59 -0
  307. package/get-shit-done/templates/summary-minimal.md +41 -0
  308. package/get-shit-done/templates/summary-standard.md +48 -0
  309. package/get-shit-done/templates/summary.md +248 -0
  310. package/get-shit-done/templates/user-profile.md +146 -0
  311. package/get-shit-done/templates/user-setup.md +311 -0
  312. package/get-shit-done/templates/verification-report.md +322 -0
  313. package/get-shit-done/workflows/_runtime-launcher.snippet.sh +1 -0
  314. package/get-shit-done/workflows/add-backlog.md +91 -0
  315. package/get-shit-done/workflows/add-phase.md +113 -0
  316. package/get-shit-done/workflows/add-tests.md +355 -0
  317. package/get-shit-done/workflows/add-todo.md +161 -0
  318. package/get-shit-done/workflows/ai-integration-phase.md +295 -0
  319. package/get-shit-done/workflows/analyze-dependencies.md +96 -0
  320. package/get-shit-done/workflows/audit-fix.md +178 -0
  321. package/get-shit-done/workflows/audit-milestone.md +358 -0
  322. package/get-shit-done/workflows/audit-uat.md +110 -0
  323. package/get-shit-done/workflows/autonomous.md +795 -0
  324. package/get-shit-done/workflows/check-todos.md +180 -0
  325. package/get-shit-done/workflows/cleanup.md +155 -0
  326. package/get-shit-done/workflows/code-review-fix.md +502 -0
  327. package/get-shit-done/workflows/code-review.md +656 -0
  328. package/get-shit-done/workflows/complete-milestone.md +855 -0
  329. package/get-shit-done/workflows/debug.md +232 -0
  330. package/get-shit-done/workflows/diagnose-issues.md +241 -0
  331. package/get-shit-done/workflows/discovery-phase.md +291 -0
  332. package/get-shit-done/workflows/discuss-phase/modes/advisor.md +176 -0
  333. package/get-shit-done/workflows/discuss-phase/modes/all.md +28 -0
  334. package/get-shit-done/workflows/discuss-phase/modes/analyze.md +44 -0
  335. package/get-shit-done/workflows/discuss-phase/modes/auto.md +57 -0
  336. package/get-shit-done/workflows/discuss-phase/modes/batch.md +52 -0
  337. package/get-shit-done/workflows/discuss-phase/modes/chain.md +98 -0
  338. package/get-shit-done/workflows/discuss-phase/modes/default.md +141 -0
  339. package/get-shit-done/workflows/discuss-phase/modes/power.md +44 -0
  340. package/get-shit-done/workflows/discuss-phase/modes/text.md +55 -0
  341. package/get-shit-done/workflows/discuss-phase/templates/checkpoint.json +18 -0
  342. package/get-shit-done/workflows/discuss-phase/templates/context.md +136 -0
  343. package/get-shit-done/workflows/discuss-phase/templates/discussion-log.md +50 -0
  344. package/get-shit-done/workflows/discuss-phase-assumptions.md +675 -0
  345. package/get-shit-done/workflows/discuss-phase-power.md +291 -0
  346. package/get-shit-done/workflows/discuss-phase.md +499 -0
  347. package/get-shit-done/workflows/do.md +111 -0
  348. package/get-shit-done/workflows/docs-update.md +1162 -0
  349. package/get-shit-done/workflows/edit-phase.md +295 -0
  350. package/get-shit-done/workflows/eval-review.md +156 -0
  351. package/get-shit-done/workflows/execute-phase/steps/codebase-drift-gate.md +82 -0
  352. package/get-shit-done/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  353. package/get-shit-done/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  354. package/get-shit-done/workflows/execute-phase.md +1709 -0
  355. package/get-shit-done/workflows/execute-plan.md +526 -0
  356. package/get-shit-done/workflows/explore.md +144 -0
  357. package/get-shit-done/workflows/extract-learnings.md +243 -0
  358. package/get-shit-done/workflows/fast.md +124 -0
  359. package/get-shit-done/workflows/forensics.md +279 -0
  360. package/get-shit-done/workflows/graduation.md +196 -0
  361. package/get-shit-done/workflows/health.md +224 -0
  362. package/get-shit-done/workflows/help/modes/brief.md +22 -0
  363. package/get-shit-done/workflows/help/modes/default.md +50 -0
  364. package/get-shit-done/workflows/help/modes/full.md +784 -0
  365. package/get-shit-done/workflows/help/modes/topic.md +74 -0
  366. package/get-shit-done/workflows/help.md +24 -0
  367. package/get-shit-done/workflows/import.md +254 -0
  368. package/get-shit-done/workflows/inbox.md +387 -0
  369. package/get-shit-done/workflows/ingest-docs.md +339 -0
  370. package/get-shit-done/workflows/insert-phase.md +152 -0
  371. package/get-shit-done/workflows/list-phase-assumptions.md +178 -0
  372. package/get-shit-done/workflows/list-workspaces.md +57 -0
  373. package/get-shit-done/workflows/manager.md +393 -0
  374. package/get-shit-done/workflows/map-codebase.md +444 -0
  375. package/get-shit-done/workflows/milestone-summary.md +224 -0
  376. package/get-shit-done/workflows/mvp-phase.md +222 -0
  377. package/get-shit-done/workflows/new-milestone.md +635 -0
  378. package/get-shit-done/workflows/new-project.md +1555 -0
  379. package/get-shit-done/workflows/new-workspace.md +240 -0
  380. package/get-shit-done/workflows/next.md +299 -0
  381. package/get-shit-done/workflows/node-repair.md +92 -0
  382. package/get-shit-done/workflows/note.md +158 -0
  383. package/get-shit-done/workflows/pause-work.md +244 -0
  384. package/get-shit-done/workflows/plan-milestone-gaps.md +281 -0
  385. package/get-shit-done/workflows/plan-phase.md +1809 -0
  386. package/get-shit-done/workflows/plan-review-convergence.md +346 -0
  387. package/get-shit-done/workflows/plant-seed.md +230 -0
  388. package/get-shit-done/workflows/pr-branch.md +157 -0
  389. package/get-shit-done/workflows/profile-user.md +453 -0
  390. package/get-shit-done/workflows/progress.md +699 -0
  391. package/get-shit-done/workflows/quick.md +1039 -0
  392. package/get-shit-done/workflows/reapply-patches.md +426 -0
  393. package/get-shit-done/workflows/remove-phase.md +156 -0
  394. package/get-shit-done/workflows/remove-workspace.md +108 -0
  395. package/get-shit-done/workflows/resume-project.md +332 -0
  396. package/get-shit-done/workflows/review.md +623 -0
  397. package/get-shit-done/workflows/scan.md +105 -0
  398. package/get-shit-done/workflows/secure-phase.md +180 -0
  399. package/get-shit-done/workflows/session-report.md +146 -0
  400. package/get-shit-done/workflows/settings-advanced.md +620 -0
  401. package/get-shit-done/workflows/settings-integrations.md +312 -0
  402. package/get-shit-done/workflows/settings.md +552 -0
  403. package/get-shit-done/workflows/ship.md +356 -0
  404. package/get-shit-done/workflows/sketch-wrap-up.md +286 -0
  405. package/get-shit-done/workflows/sketch.md +361 -0
  406. package/get-shit-done/workflows/spec-phase.md +262 -0
  407. package/get-shit-done/workflows/spike-wrap-up.md +307 -0
  408. package/get-shit-done/workflows/spike.md +453 -0
  409. package/get-shit-done/workflows/stats.md +80 -0
  410. package/get-shit-done/workflows/sync-skills.md +182 -0
  411. package/get-shit-done/workflows/thread.md +222 -0
  412. package/get-shit-done/workflows/transition.md +694 -0
  413. package/get-shit-done/workflows/ui-phase.md +328 -0
  414. package/get-shit-done/workflows/ui-review.md +193 -0
  415. package/get-shit-done/workflows/ultraplan-phase.md +199 -0
  416. package/get-shit-done/workflows/undo.md +314 -0
  417. package/get-shit-done/workflows/update.md +443 -0
  418. package/get-shit-done/workflows/validate-phase.md +179 -0
  419. package/get-shit-done/workflows/verify-phase.md +544 -0
  420. package/get-shit-done/workflows/verify-work.md +781 -0
  421. package/hooks/dist/gsd-check-update-worker.js +95 -0
  422. package/hooks/dist/gsd-check-update.js +64 -0
  423. package/hooks/dist/gsd-context-monitor.js +195 -0
  424. package/hooks/dist/gsd-graphify-update.sh +158 -0
  425. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  426. package/hooks/dist/gsd-prompt-guard.js +97 -0
  427. package/hooks/dist/gsd-read-guard.js +101 -0
  428. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  429. package/hooks/dist/gsd-session-state.sh +59 -0
  430. package/hooks/dist/gsd-statusline.js +548 -0
  431. package/hooks/dist/gsd-update-banner.js +134 -0
  432. package/hooks/dist/gsd-validate-commit.sh +57 -0
  433. package/hooks/dist/gsd-workflow-guard.js +166 -0
  434. package/hooks/dist/lib/git-cmd.js +150 -0
  435. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  436. package/hooks/gsd-check-update-worker.js +95 -0
  437. package/hooks/gsd-check-update.js +64 -0
  438. package/hooks/gsd-context-monitor.js +195 -0
  439. package/hooks/gsd-graphify-update.sh +158 -0
  440. package/hooks/gsd-phase-boundary.sh +47 -0
  441. package/hooks/gsd-prompt-guard.js +97 -0
  442. package/hooks/gsd-read-guard.js +101 -0
  443. package/hooks/gsd-read-injection-scanner.js +203 -0
  444. package/hooks/gsd-session-state.sh +59 -0
  445. package/hooks/gsd-statusline.js +548 -0
  446. package/hooks/gsd-update-banner.js +134 -0
  447. package/hooks/gsd-validate-commit.sh +57 -0
  448. package/hooks/gsd-workflow-guard.js +166 -0
  449. package/hooks/lib/git-cmd.js +150 -0
  450. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  451. package/hooks/managed-hooks-registry.cjs +34 -0
  452. package/package.json +102 -0
  453. package/scripts/affected-tests-lib.cjs +541 -0
  454. package/scripts/audit-workflow-script-paths.cjs +73 -0
  455. package/scripts/base64-scan.sh +339 -0
  456. package/scripts/build-hooks.js +236 -0
  457. package/scripts/changeset/README.md +129 -0
  458. package/scripts/changeset/cli.cjs +392 -0
  459. package/scripts/changeset/github-release-notes.cjs +199 -0
  460. package/scripts/changeset/lint.cjs +110 -0
  461. package/scripts/changeset/new.cjs +137 -0
  462. package/scripts/changeset/parse.cjs +114 -0
  463. package/scripts/changeset/render.cjs +34 -0
  464. package/scripts/changeset/serialize.cjs +130 -0
  465. package/scripts/check-alias-drift.cjs +108 -0
  466. package/scripts/check-env.cjs +302 -0
  467. package/scripts/check-npm-integrity.cjs +209 -0
  468. package/scripts/ci-guard-runner.cjs +16 -0
  469. package/scripts/ci-prepare-test-scope.cjs +46 -0
  470. package/scripts/ci-rebase-check.cjs +85 -0
  471. package/scripts/ci-test-scope.cjs +302 -0
  472. package/scripts/command-contract-helpers.cjs +64 -0
  473. package/scripts/diff-touches-shipped-paths.cjs +147 -0
  474. package/scripts/fix-slash-commands.cjs +147 -0
  475. package/scripts/gen-inventory-manifest.cjs +109 -0
  476. package/scripts/generate-package-identity.cjs +104 -0
  477. package/scripts/lint-command-contract.cjs +108 -0
  478. package/scripts/lint-descriptions.cjs +83 -0
  479. package/scripts/lint-docs-required.cjs +222 -0
  480. package/scripts/lint-no-source-grep-extras.cjs +81 -0
  481. package/scripts/lint-no-source-grep.cjs +174 -0
  482. package/scripts/lint-package-identity-drift.cjs +141 -0
  483. package/scripts/lint-pr-check-project-dir.cjs +98 -0
  484. package/scripts/lint-shared-module-handsync.cjs +388 -0
  485. package/scripts/lint-shell-command-projection-drift.cjs +57 -0
  486. package/scripts/lint-skill-deps.cjs +180 -0
  487. package/scripts/lint-test-file-count.allowlist.json +36 -0
  488. package/scripts/lint-test-file-count.cjs +190 -0
  489. package/scripts/pr-template-policy.cjs +268 -0
  490. package/scripts/prompt-injection-scan.sh +203 -0
  491. package/scripts/release-tarball-smoke.cjs +627 -0
  492. package/scripts/run-affected-tests.cjs +6 -0
  493. package/scripts/run-cross-platform-tests.cjs +63 -0
  494. package/scripts/run-tests.cjs +282 -0
  495. package/scripts/secret-scan-lint.sh +231 -0
  496. package/scripts/secret-scan.sh +358 -0
  497. package/scripts/setup-branch-protection.sh +236 -0
  498. package/scripts/shared-module-handsync-allowlist.json +183 -0
  499. package/scripts/strip-prose-atrefs.cjs +106 -0
  500. package/scripts/sync-rulesets.sh +34 -0
  501. package/scripts/sync-runtime-launcher.cjs +402 -0
  502. package/scripts/test-failure-reasons.cjs +34 -0
  503. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,541 @@
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 { suiteOf } = require('./run-tests.cjs');
8
+
9
+ const CRITICAL_PATHS = [
10
+ '.github/workflows/',
11
+ 'package.json',
12
+ 'package-lock.json',
13
+ 'scripts/run-tests.cjs',
14
+ 'scripts/affected-tests-lib.cjs',
15
+ 'scripts/run-affected-tests.cjs',
16
+ ];
17
+
18
+ // Suites that are push-only. PRs must never select or run these.
19
+ const PR_EXCLUDED_SUITES = new Set(['install', 'slow']);
20
+
21
+ // Suites run on every PR cell when the critical-path fallback fires.
22
+ const PR_FULL_SUITES = ['unit', 'integration', 'security'];
23
+
24
+ // Source trees to walk when building the forward graph (in addition to tests/).
25
+ // Relative to repoRoot. We walk these to discover SUT-internal requires so that
26
+ // a change to a deep helper propagates through re-export chains to tests.
27
+ const SOURCE_TREES = [
28
+ 'get-shit-done/bin/lib',
29
+ 'bin/lib',
30
+ 'bin',
31
+ 'scripts',
32
+ 'commands',
33
+ 'hooks',
34
+ 'agents',
35
+ 'eslint-rules',
36
+ ];
37
+
38
+ function toPosixPath(input) {
39
+ return input.split(path.sep).join('/');
40
+ }
41
+
42
+ function parseRelativeSpecifiers(source) {
43
+ const specifiers = [];
44
+ const requireRe = /require\((['"])(.+?)\1\)/g;
45
+ const importFromRe = /from\s+(['"])(.+?)\1/g;
46
+ let match;
47
+
48
+ while ((match = requireRe.exec(source)) !== null) {
49
+ specifiers.push(match[2]);
50
+ }
51
+ while ((match = importFromRe.exec(source)) !== null) {
52
+ specifiers.push(match[2]);
53
+ }
54
+
55
+ return specifiers.filter(specifier => specifier.startsWith('.'));
56
+ }
57
+
58
+ // Extended candidate list now includes .ts/.cts/.mts/.json as well as the
59
+ // standard .js/.cjs/.mjs and index variants.
60
+ function resolveRelativeDependency(repoRoot, fromAbs, specifier) {
61
+ const base = path.resolve(path.dirname(fromAbs), specifier);
62
+ const candidates = [
63
+ base,
64
+ `${base}.js`,
65
+ `${base}.cjs`,
66
+ `${base}.mjs`,
67
+ `${base}.ts`,
68
+ `${base}.cts`,
69
+ `${base}.mts`,
70
+ `${base}.json`,
71
+ path.join(base, 'index.js'),
72
+ path.join(base, 'index.cjs'),
73
+ path.join(base, 'index.mjs'),
74
+ path.join(base, 'index.ts'),
75
+ ];
76
+
77
+ for (const candidate of candidates) {
78
+ if (existsSync(candidate)) {
79
+ return toPosixPath(path.relative(repoRoot, candidate));
80
+ }
81
+ }
82
+
83
+ return null;
84
+ }
85
+
86
+ // ---------------------------------------------------------------------------
87
+ // Source-file walker
88
+ // ---------------------------------------------------------------------------
89
+
90
+ /**
91
+ * Collect all .cjs / .mjs / .js / .ts / .cts / .mts / .json files under a
92
+ * directory tree, returned as repo-relative POSIX paths. Silently skips
93
+ * trees that don't exist.
94
+ */
95
+ function walkTree(repoRoot, relDir) {
96
+ const absDir = path.join(repoRoot, relDir);
97
+ if (!existsSync(absDir)) return [];
98
+
99
+ const results = [];
100
+ const queue = [absDir];
101
+
102
+ while (queue.length > 0) {
103
+ const cur = queue.shift();
104
+ let entries;
105
+ try {
106
+ entries = readdirSync(cur, { withFileTypes: true });
107
+ } catch {
108
+ continue;
109
+ }
110
+ for (const entry of entries) {
111
+ const abs = path.join(cur, entry.name);
112
+ if (entry.isDirectory()) {
113
+ // Skip node_modules
114
+ if (entry.name === 'node_modules') continue;
115
+ queue.push(abs);
116
+ } else if (entry.isFile()) {
117
+ const ext = path.extname(entry.name);
118
+ if (['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext)) {
119
+ results.push(toPosixPath(path.relative(repoRoot, abs)));
120
+ }
121
+ }
122
+ }
123
+ }
124
+
125
+ return results;
126
+ }
127
+
128
+ // ---------------------------------------------------------------------------
129
+ // Forward graph: Map<fileRel, Set<depRel>>
130
+ // ---------------------------------------------------------------------------
131
+
132
+ /**
133
+ * Build a forward dependency graph over test files PLUS source trees.
134
+ * For each file: read, parseRelativeSpecifiers, resolve each specifier.
135
+ * Returns Map<fileRel, Set<depRel>>.
136
+ */
137
+ function buildForwardGraph(repoRoot, testFiles) {
138
+ // Collect all files to index: test files + source files
139
+ const sourceFiles = [];
140
+ for (const tree of SOURCE_TREES) {
141
+ for (const f of walkTree(repoRoot, tree)) {
142
+ sourceFiles.push(f);
143
+ }
144
+ }
145
+
146
+ const allFiles = [...new Set([...testFiles, ...sourceFiles])];
147
+ const forward = new Map();
148
+
149
+ for (const fileRel of allFiles) {
150
+ const absFile = path.join(repoRoot, fileRel);
151
+ let source;
152
+ try {
153
+ source = readFileSync(absFile, 'utf8');
154
+ } catch {
155
+ continue;
156
+ }
157
+
158
+ const specs = parseRelativeSpecifiers(source);
159
+ const deps = new Set();
160
+
161
+ for (const specifier of specs) {
162
+ const dep = resolveRelativeDependency(repoRoot, absFile, specifier);
163
+ if (dep) deps.add(dep);
164
+ }
165
+
166
+ forward.set(fileRel, deps);
167
+ }
168
+
169
+ return forward;
170
+ }
171
+
172
+ // ---------------------------------------------------------------------------
173
+ // Reverse-transitive index: Map<depRel, Set<testRel>>
174
+ // ---------------------------------------------------------------------------
175
+
176
+ /**
177
+ * Build the TRANSITIVE reverse index: Map<depRel, Set<testRel>>.
178
+ *
179
+ * Algorithm:
180
+ * 1. Build forward graph over all test + source files.
181
+ * 2. Invert to direct reverse edges: Map<depRel, Set<dependentRel>>.
182
+ * 3. For each test file, BFS backwards through all direct reverse edges
183
+ * to find every ancestor. Map each ancestor → the test.
184
+ *
185
+ * Cycle safety: visited set per BFS — each node is enqueued at most once.
186
+ *
187
+ * @param {string} repoRoot
188
+ * @param {string[]} testFiles repo-relative posix paths (e.g. ['tests/foo.test.cjs'])
189
+ * @returns {Map<string, Set<string>>}
190
+ */
191
+ function buildTransitiveReverseIndex(repoRoot, testFiles) {
192
+ const forward = buildForwardGraph(repoRoot, testFiles);
193
+
194
+ // Build direct reverse edges: dep → Set of files that directly require dep
195
+ const directReverse = new Map();
196
+ for (const [fileRel, deps] of forward) {
197
+ for (const dep of deps) {
198
+ if (!directReverse.has(dep)) directReverse.set(dep, new Set());
199
+ directReverse.get(dep).add(fileRel);
200
+ }
201
+ }
202
+
203
+ // For each test file, BFS through direct reverse edges to collect all
204
+ // ancestors, then invert: ancestor → test.
205
+ // We do this test-file-first (not dep-first) so we know which test reached
206
+ // each ancestor.
207
+ const transitiveReverse = new Map();
208
+
209
+ for (const testFile of testFiles) {
210
+ // BFS from testFile following reverse edges (files that point TO testFile,
211
+ // then files that point to THOSE files, etc.).
212
+ // We want: "if X changed, would that eventually pull in testFile?"
213
+ // So we walk the FORWARD graph starting from testFile to find all deps,
214
+ // then any of those deps maps back to testFile.
215
+
216
+ // Actually simpler: for each test we do a forward BFS to find ALL files
217
+ // the test transitively depends on. Then we record testFile as a
218
+ // dependent of each of those files.
219
+ const visited = new Set();
220
+ visited.add(testFile);
221
+ const queue = [testFile];
222
+
223
+ while (queue.length > 0) {
224
+ const current = queue.shift();
225
+ const deps = forward.get(current);
226
+ if (!deps) continue;
227
+ for (const dep of deps) {
228
+ if (visited.has(dep)) continue;
229
+ visited.add(dep);
230
+ queue.push(dep);
231
+ }
232
+ }
233
+
234
+ // Every file in `visited` (except testFile itself) is a transitive dep.
235
+ // Record testFile as a dependent of each.
236
+ for (const dep of visited) {
237
+ if (dep === testFile) continue;
238
+ if (!transitiveReverse.has(dep)) transitiveReverse.set(dep, new Set());
239
+ transitiveReverse.get(dep).add(testFile);
240
+ }
241
+ }
242
+
243
+ return transitiveReverse;
244
+ }
245
+
246
+ // ---------------------------------------------------------------------------
247
+ // Legacy shim — kept so that runAffectedTests can call buildTransitiveReverseIndex
248
+ // and existing call sites that still call buildReverseIndex still work.
249
+ // ---------------------------------------------------------------------------
250
+ function buildReverseIndex(repoRoot, testFiles) {
251
+ return buildTransitiveReverseIndex(repoRoot, testFiles);
252
+ }
253
+
254
+ function shouldRunFullSuite(changedFiles) {
255
+ return changedFiles.some(file =>
256
+ CRITICAL_PATHS.some(critical => file === critical || file.startsWith(critical)),
257
+ );
258
+ }
259
+
260
+ function listTestFiles(repoRoot) {
261
+ return readdirSync(path.join(repoRoot, 'tests'))
262
+ .filter(file => file.endsWith('.test.cjs'))
263
+ .map(file => `tests/${file}`)
264
+ .sort();
265
+ }
266
+
267
+ /**
268
+ * Select the affected tests given a set of changed files and a reverse index.
269
+ *
270
+ * Options:
271
+ * detectWiden {boolean} — when true, attach `._widenRequired = true` to the
272
+ * returned array when a changed source file has zero transitive test
273
+ * dependents. The caller (runAffectedTests) uses this to widen to unit/all.
274
+ *
275
+ * The returned array is sorted and may have `._widenRequired` attached.
276
+ */
277
+ function pickAffectedTests(changedFiles, allTests, reverseIndex, options = {}) {
278
+ const { detectWiden = false } = options;
279
+ const selected = new Set();
280
+ let widenRequired = false;
281
+
282
+ // Build a fast lookup of currently-existing test files (from readdirSync — deleted files absent).
283
+ const allTestsSet = new Set(allTests);
284
+
285
+ // (a) directly-changed test files + (b) transitive test dependents
286
+ // Deleted test files are filtered out — they no longer exist and cannot be run.
287
+ // A deleted test file also must NOT trigger widen (the test is simply gone).
288
+ for (const file of changedFiles) {
289
+ if (file.startsWith('tests/') && file.endsWith('.test.cjs')) {
290
+ // Only select if the test file still exists (i.e. is present in allTests from readdirSync).
291
+ if (allTestsSet.has(file)) {
292
+ selected.add(file);
293
+ }
294
+ // Deleted test file — do not add to selected; do not look up reverse index.
295
+ } else {
296
+ const dependents = reverseIndex.get(file);
297
+ if (dependents) {
298
+ for (const testFile of dependents) selected.add(testFile);
299
+ }
300
+ }
301
+ }
302
+
303
+ // (c) stem heuristic — kept as secondary mechanism
304
+ for (const file of changedFiles) {
305
+ const stem = path.basename(file).replace(/\.[^.]+$/, '').toLowerCase();
306
+ if (!stem) continue;
307
+ for (const testFile of allTests) {
308
+ if (testFile.toLowerCase().includes(stem)) selected.add(testFile);
309
+ }
310
+ }
311
+
312
+ // Widen backstop: if a changed file is a non-test, non-CRITICAL_PATH source file
313
+ // (recognised extension) under a SOURCE_TREE, AND it is either deleted (no longer
314
+ // on disk — so never in the forward graph and has no static dependents) OR it
315
+ // exists with ZERO transitive test dependents — signal a widen.
316
+ // NOTE: we deliberately do NOT skip deleted files here; a deleted source file's
317
+ // absence from the forward graph means dependents===undefined, which is the same
318
+ // as zero static dependents, and is itself the widen trigger.
319
+ if (detectWiden) {
320
+ for (const file of changedFiles) {
321
+ // Only care about source files, not test files or docs
322
+ if (file.startsWith('tests/')) continue;
323
+ if (shouldRunFullSuite([file])) continue; // critical path already triggers full suite
324
+ // Check: is this a source file (has a recognised extension)?
325
+ const ext = path.extname(file);
326
+ const isSourceFile = ['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext);
327
+ if (!isSourceFile) continue;
328
+ // Check: is this file under a recognised source tree?
329
+ const isUnderSourceTree = SOURCE_TREES.some(
330
+ tree => file === tree || file.startsWith(tree + '/'),
331
+ );
332
+ if (!isUnderSourceTree) continue;
333
+ // Does it have any test dependents?
334
+ // A deleted file will have undefined here (not in the graph) — that is
335
+ // treated as zero static dependents and triggers widen conservatively.
336
+ const dependents = reverseIndex.get(file);
337
+ const hasStaticDependents = dependents && dependents.size > 0;
338
+ if (!hasStaticDependents) {
339
+ widenRequired = true;
340
+ break;
341
+ }
342
+ }
343
+ }
344
+
345
+ // Drop any file whose suite is push-only. This is the single chokepoint —
346
+ // it catches direct-change, reverse-index, AND stem-match selections.
347
+ for (const file of selected) {
348
+ const suite = suiteOf(path.basename(file));
349
+ if (PR_EXCLUDED_SUITES.has(suite)) selected.delete(file);
350
+ }
351
+
352
+ // When nothing maps, return an empty array. The caller decides the fallback.
353
+ const result = [...selected].sort();
354
+ if (widenRequired) result._widenRequired = true;
355
+ return result;
356
+ }
357
+
358
+ function changedFilesSinceBase(repoRoot, baseRef) {
359
+ const out = execFileSync(
360
+ 'git',
361
+ ['diff', '--name-only', '--no-renames', '--diff-filter=ACMRD', `${baseRef}...HEAD`],
362
+ { cwd: repoRoot, encoding: 'utf8' },
363
+ ).trim();
364
+ if (!out) return [];
365
+ return out.split('\n').map(line => line.trim()).filter(Boolean);
366
+ }
367
+
368
+ function runNodeTestFiles(repoRoot, files) {
369
+ const defaultConcurrency = process.platform === 'win32' ? 2 : 4;
370
+ const concurrency = process.env.TEST_CONCURRENCY
371
+ ? `--test-concurrency=${process.env.TEST_CONCURRENCY}`
372
+ : `--test-concurrency=${defaultConcurrency}`;
373
+ const absoluteFiles = files.map(file => path.join(repoRoot, file));
374
+
375
+ // Keep chunks bounded for Windows CreateProcess command-length limits.
376
+ const maxChars = process.env.RUN_TESTS_MAX_CMDLINE_CHARS
377
+ ? Number(process.env.RUN_TESTS_MAX_CMDLINE_CHARS)
378
+ : 28000;
379
+ const fixed = process.execPath.length + '--test'.length + concurrency.length + 8;
380
+ const chunks = [];
381
+ let current = [];
382
+ let currentLen = fixed;
383
+
384
+ for (const file of absoluteFiles) {
385
+ const add = file.length + 1;
386
+ if (current.length > 0 && currentLen + add > maxChars) {
387
+ chunks.push(current);
388
+ current = [];
389
+ currentLen = fixed;
390
+ }
391
+ current.push(file);
392
+ currentLen += add;
393
+ }
394
+ if (current.length > 0) chunks.push(current);
395
+
396
+ let firstFailure = 0;
397
+ for (let i = 0; i < chunks.length; i++) {
398
+ if (chunks.length > 1) {
399
+ console.error(`affected-tests: chunk ${i + 1}/${chunks.length} (${chunks[i].length} files)`);
400
+ }
401
+ try {
402
+ execFileSync(process.execPath, ['--test', concurrency, ...chunks[i]], {
403
+ cwd: repoRoot,
404
+ stdio: 'inherit',
405
+ env: { ...process.env },
406
+ });
407
+ } catch (error) {
408
+ const code = error.status || 1;
409
+ if (firstFailure === 0) firstFailure = code;
410
+ }
411
+ }
412
+ if (firstFailure !== 0) process.exit(firstFailure);
413
+ }
414
+
415
+ function runSuite(repoRoot, suite) {
416
+ execFileSync(process.execPath, ['scripts/run-tests.cjs', '--suite', suite], {
417
+ cwd: repoRoot,
418
+ stdio: 'inherit',
419
+ env: { ...process.env },
420
+ });
421
+ }
422
+
423
+ function resolveBaseRef() {
424
+ if (process.env.GSD_AFFECTED_BASE) return process.env.GSD_AFFECTED_BASE;
425
+ if (process.env.GITHUB_BASE_REF) return `origin/${process.env.GITHUB_BASE_REF}`;
426
+ return 'origin/main';
427
+ }
428
+
429
+ /**
430
+ * Pure function: given the outputs of the selection phase, return a run plan
431
+ * describing what should be executed. No I/O is performed here.
432
+ *
433
+ * Return shapes:
434
+ * { mode: 'suite', suite: 'unit' } — no changed files
435
+ * { mode: 'suites', suites: PR_FULL_SUITES } — critical path triggered
436
+ * { mode: 'suites', suites: PR_FULL_SUITES } — widen required (orphan src file)
437
+ * { mode: 'suite', suite: 'unit' } — selection empty after widen=false
438
+ * { mode: 'files', files: string[] } — concrete selection, no widen
439
+ *
440
+ * Invariant: when widenRequired is true the executed set is ALWAYS ⊇ selected,
441
+ * because PR_FULL_SUITES covers every PR-eligible suite (unit + integration +
442
+ * security), so every concrete match that pickAffectedTests put into `selected`
443
+ * belongs to one of those suites and will be exercised by running all three.
444
+ */
445
+ function resolveRunPlan({ changedFiles, selected, widenRequired, criticalPath, noChanges }) {
446
+ if (noChanges) {
447
+ return { mode: 'suite', suite: 'unit' };
448
+ }
449
+ if (criticalPath) {
450
+ return { mode: 'suites', suites: PR_FULL_SUITES };
451
+ }
452
+ if (widenRequired) {
453
+ return { mode: 'suites', suites: PR_FULL_SUITES };
454
+ }
455
+ if (selected.length === 0) {
456
+ return { mode: 'suite', suite: 'unit' };
457
+ }
458
+ return { mode: 'files', files: selected };
459
+ }
460
+
461
+ function runAffectedTests(options = {}) {
462
+ const repoRoot = options.repoRoot || path.resolve(__dirname, '..');
463
+ const baseRef = options.baseRef || resolveBaseRef();
464
+ const changed = changedFilesSinceBase(repoRoot, baseRef);
465
+
466
+ if (changed.length === 0) {
467
+ console.error(`affected-tests: no changed files against ${baseRef}; running unit suite`);
468
+ runSuite(repoRoot, 'unit');
469
+ return;
470
+ }
471
+
472
+ if (shouldRunFullSuite(changed)) {
473
+ console.error('affected-tests: critical CI/runtime files changed; running PR suites (unit, integration, security)');
474
+ for (const suite of PR_FULL_SUITES) {
475
+ runSuite(repoRoot, suite);
476
+ }
477
+ return;
478
+ }
479
+
480
+ const allTests = listTestFiles(repoRoot);
481
+ const reverseIndex = buildTransitiveReverseIndex(repoRoot, allTests);
482
+ const selected = pickAffectedTests(changed, allTests, reverseIndex, { detectWiden: true });
483
+
484
+ console.error(`affected-tests: base=${baseRef} changed=${changed.length} selected=${selected.length}`);
485
+ console.error(`affected-tests: ${selected.join(' ')}`);
486
+
487
+ const plan = resolveRunPlan({
488
+ changedFiles: changed,
489
+ selected,
490
+ widenRequired: selected._widenRequired === true,
491
+ criticalPath: false,
492
+ noChanges: false,
493
+ });
494
+
495
+ if (plan.mode === 'suites') {
496
+ // Widen backstop: a source file changed that has no static test dependents.
497
+ // Run all PR suites (unit + integration + security) — a strict superset of
498
+ // the concretely-selected tests — so no integration/security match is lost.
499
+ for (const file of changed) {
500
+ const ext = path.extname(file);
501
+ const isSourceFile = ['.js', '.cjs', '.mjs', '.ts', '.cts', '.mts', '.json'].includes(ext);
502
+ if (!isSourceFile || file.startsWith('tests/') || shouldRunFullSuite([file])) continue;
503
+ const dependents = reverseIndex.get(file);
504
+ if (!dependents || dependents.size === 0) {
505
+ console.error(
506
+ `affected-tests: ${file} has no static test dependents; widening to PR suites (unit+integration+security)`,
507
+ );
508
+ }
509
+ }
510
+ for (const suite of plan.suites) {
511
+ runSuite(repoRoot, suite);
512
+ }
513
+ return;
514
+ }
515
+
516
+ if (plan.mode === 'suite') {
517
+ console.error('affected-tests: no affected tests found; running unit suite as smoke');
518
+ runSuite(repoRoot, plan.suite);
519
+ return;
520
+ }
521
+
522
+ // plan.mode === 'files'
523
+ runNodeTestFiles(repoRoot, plan.files);
524
+ }
525
+
526
+ module.exports = {
527
+ CRITICAL_PATHS,
528
+ PR_EXCLUDED_SUITES,
529
+ PR_FULL_SUITES,
530
+ buildForwardGraph,
531
+ buildReverseIndex,
532
+ buildTransitiveReverseIndex,
533
+ parseRelativeSpecifiers,
534
+ pickAffectedTests,
535
+ resolveBaseRef,
536
+ resolveRelativeDependency,
537
+ resolveRunPlan,
538
+ shouldRunFullSuite,
539
+ toPosixPath,
540
+ runAffectedTests,
541
+ };
@@ -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 };