@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,278 @@
1
+ #!/usr/bin/env node
2
+ 'use strict';
3
+
4
+ // ---------------------------------------------------------------------------
5
+ // Constants
6
+ // ---------------------------------------------------------------------------
7
+
8
+ const POSSIBLE_DUPLICATE_LABEL = 'possible-duplicate';
9
+ const HUMAN_REVIEW_LABEL = 'needs-maintainer-review';
10
+ const CHALLENGE_MARKER = '<!-- gsd-dedupe-challenge -->';
11
+ const DEFAULT_WINDOW_HOURS = 24;
12
+ const DEFAULT_THRESHOLD = 0.6;
13
+ const DEFAULT_MAX_CANDIDATES = 5;
14
+ const MIN_TOKEN_LENGTH = 3;
15
+
16
+ const EXEMPT_LABELS = [
17
+ 'priority: critical',
18
+ 'pinned',
19
+ 'confirmed-bug',
20
+ 'confirmed',
21
+ 'fix-pending',
22
+ 'needs-maintainer-review',
23
+ ];
24
+
25
+ const STOPWORDS = new Set([
26
+ 'the', 'a', 'an', 'and', 'or', 'but', 'if', 'then', 'is', 'are', 'was',
27
+ 'be', 'to', 'of', 'in', 'on', 'for', 'with', 'as', 'at', 'by', 'from',
28
+ 'this', 'that', 'it', 'its', 'not', 'no', 'when', 'what', 'why', 'how',
29
+ 'does', 'do', 'doing', 'did', 'can', 'will', 'would', 'should',
30
+ 'i', 'we', 'you', 'your', 'my', 'me',
31
+ 'issue', 'bug', 'error', 'problem', 'feature', 'request',
32
+ 'help', 'support', 'please', 'question',
33
+ 'after', 'before', 'into', 'only', 'then', 'than', 'them', 'they',
34
+ 'use', 'used', 'using',
35
+ ]);
36
+
37
+ // ---------------------------------------------------------------------------
38
+ // tokenize(title) -> string[]
39
+ //
40
+ // Lowercase the title, replace any non-[a-z0-9] run with a space, split on
41
+ // whitespace, drop tokens shorter than MIN_TOKEN_LENGTH, drop STOPWORDS, and
42
+ // dedupe while preserving stable first-occurrence order.
43
+ //
44
+ // Non-string, null, or empty input returns []. Must not throw on any input.
45
+ // ---------------------------------------------------------------------------
46
+
47
+ function tokenize(title) {
48
+ if (typeof title !== 'string' || !title) return [];
49
+
50
+ const normalized = title.toLowerCase().replace(/[^a-z0-9]+/g, ' ').trim();
51
+ if (!normalized) return [];
52
+
53
+ const seen = new Set();
54
+ const result = [];
55
+
56
+ for (const token of normalized.split(' ')) {
57
+ if (!token || token.length < MIN_TOKEN_LENGTH) continue;
58
+ if (STOPWORDS.has(token)) continue;
59
+ if (seen.has(token)) continue;
60
+ seen.add(token);
61
+ result.push(token);
62
+ }
63
+
64
+ return result;
65
+ }
66
+
67
+ // ---------------------------------------------------------------------------
68
+ // diceSimilarity(aTokens, bTokens) -> number 0..1
69
+ //
70
+ // Sørensen–Dice over the token sets: 2 * |A ∩ B| / (|A| + |B|).
71
+ // Both inputs are treated as sets (duplicates ignored). Empty either side -> 0.
72
+ // Identical sets -> 1.
73
+ // ---------------------------------------------------------------------------
74
+
75
+ function diceSimilarity(aTokens, bTokens) {
76
+ const setA = new Set(aTokens);
77
+ const setB = new Set(bTokens);
78
+
79
+ if (setA.size === 0 || setB.size === 0) return 0;
80
+
81
+ let intersection = 0;
82
+ for (const token of setA) {
83
+ if (setB.has(token)) intersection += 1;
84
+ }
85
+
86
+ return (2 * intersection) / (setA.size + setB.size);
87
+ }
88
+
89
+ // ---------------------------------------------------------------------------
90
+ // scoreCandidates(newTitle, candidates, opts) -> [{number, title, score}]
91
+ //
92
+ // opts: { threshold=DEFAULT_THRESHOLD, limit=DEFAULT_MAX_CANDIDATES, excludeNumber }
93
+ //
94
+ // Tokenizes newTitle once. If no tokens -> []. Filters out null/garbage
95
+ // candidates, those missing a number, and the excluded number. Scores each
96
+ // using diceSimilarity. Keeps score >= threshold. Sorts DESC by score,
97
+ // tie-break ASC by number. Caps to limit.
98
+ // ---------------------------------------------------------------------------
99
+
100
+ function scoreCandidates(newTitle, candidates, opts) {
101
+ const threshold = (opts && opts.threshold != null) ? opts.threshold : DEFAULT_THRESHOLD;
102
+ const limit = (opts && opts.limit != null) ? opts.limit : DEFAULT_MAX_CANDIDATES;
103
+ const excludeNumber = opts && opts.excludeNumber;
104
+
105
+ const newTokens = tokenize(newTitle);
106
+ if (newTokens.length === 0) return [];
107
+
108
+ const scored = [];
109
+
110
+ const safeCandidates = Array.isArray(candidates) ? candidates : [];
111
+ for (const candidate of safeCandidates) {
112
+ if (!candidate || typeof candidate !== 'object') continue;
113
+ if (!(typeof candidate.number === 'number' && Number.isFinite(candidate.number))) continue;
114
+ if (candidate.number === excludeNumber) continue;
115
+
116
+ const score = diceSimilarity(newTokens, tokenize(candidate.title));
117
+ if (score < threshold) continue;
118
+
119
+ scored.push({ number: candidate.number, title: candidate.title, score });
120
+ }
121
+
122
+ scored.sort((a, b) => {
123
+ if (Math.abs(a.score - b.score) > 1e-9) return b.score - a.score;
124
+ return a.number - b.number;
125
+ });
126
+
127
+ return scored.slice(0, limit);
128
+ }
129
+
130
+ // ---------------------------------------------------------------------------
131
+ // renderChallengeComment(candidates, opts) -> string
132
+ //
133
+ // opts: { windowHours=DEFAULT_WINDOW_HOURS }
134
+ //
135
+ // Deterministic. Must start with CHALLENGE_MARKER on its own line. Must list
136
+ // each candidate as a line with #<number>, title, and percentage similarity.
137
+ // Must mention windowHours and the 👎 veto.
138
+ // ---------------------------------------------------------------------------
139
+
140
+ function renderChallengeComment(candidates, opts) {
141
+ const windowHours = (opts && opts.windowHours != null) ? opts.windowHours : DEFAULT_WINDOW_HOURS;
142
+
143
+ const lines = [CHALLENGE_MARKER, ''];
144
+
145
+ lines.push('**Possible duplicate detected.** This issue may already be reported:');
146
+ lines.push('');
147
+
148
+ for (const candidate of candidates) {
149
+ const pct = Math.round(candidate.score * 100);
150
+ lines.push(`- #${candidate.number} — ${candidate.title} (similarity ${pct}%)`);
151
+ }
152
+
153
+ lines.push('');
154
+ lines.push(
155
+ `If this is **not** a duplicate, react with 👎 on this comment to veto and keep the issue open. ` +
156
+ `If no response is received within ${windowHours} hours, this issue may be closed as a duplicate.`,
157
+ );
158
+
159
+ return lines.join('\n');
160
+ }
161
+
162
+ // ---------------------------------------------------------------------------
163
+ // isChallengeComment(body) -> boolean
164
+ //
165
+ // True iff body is a string containing CHALLENGE_MARKER.
166
+ // ---------------------------------------------------------------------------
167
+
168
+ function isChallengeComment(body) {
169
+ if (typeof body !== 'string') return false;
170
+ return body.includes(CHALLENGE_MARKER);
171
+ }
172
+
173
+ // ---------------------------------------------------------------------------
174
+ // hasExemptLabel(labels) -> boolean
175
+ //
176
+ // labels may be an array of strings or array of {name}. True if any name is
177
+ // in EXEMPT_LABELS.
178
+ // ---------------------------------------------------------------------------
179
+
180
+ function hasExemptLabel(labels) {
181
+ if (!Array.isArray(labels)) return false;
182
+ const exemptSet = new Set(EXEMPT_LABELS);
183
+ for (const label of labels) {
184
+ const name = typeof label === 'string' ? label : (label && label.name);
185
+ if (name && exemptSet.has(name)) return true;
186
+ }
187
+ return false;
188
+ }
189
+
190
+ // ---------------------------------------------------------------------------
191
+ // toMs(value) -> number
192
+ //
193
+ // Coerce a Date, ISO string, or ms-number to milliseconds since epoch.
194
+ // ---------------------------------------------------------------------------
195
+
196
+ function toMs(value) {
197
+ if (value instanceof Date) return value.getTime();
198
+ if (typeof value === 'string') return new Date(value).getTime();
199
+ return Number(value);
200
+ }
201
+
202
+ // ---------------------------------------------------------------------------
203
+ // shouldClose(input) -> {close: boolean, reason: string}
204
+ //
205
+ // input: { now, labels, challengeComment, laterUserComments, windowHours=DEFAULT_WINDOW_HOURS }
206
+ //
207
+ // Decision order (returns first match):
208
+ // 1. hasExemptLabel(labels) -> {close:false, reason:'exempt-label'}
209
+ // 2. !challengeComment -> {close:false, reason:'no-challenge-comment'}
210
+ // 3. challengeComment.downvoted -> {close:false, reason:'vetoed'}
211
+ // 4. laterUserComments > 0 -> {close:false, reason:'reporter-responded'}
212
+ // 5. ageHours < windowHours -> {close:false, reason:'within-window'}
213
+ // 6. else -> {close:true, reason:'duplicate-no-response'}
214
+ // ---------------------------------------------------------------------------
215
+
216
+ function shouldClose(input) {
217
+ const {
218
+ now,
219
+ labels = [],
220
+ challengeComment,
221
+ laterUserComments = 0,
222
+ } = input;
223
+ const windowHours = (input.windowHours != null) ? input.windowHours : DEFAULT_WINDOW_HOURS;
224
+
225
+ if (hasExemptLabel(labels)) {
226
+ return { close: false, reason: 'exempt-label' };
227
+ }
228
+
229
+ if (!challengeComment) {
230
+ return { close: false, reason: 'no-challenge-comment' };
231
+ }
232
+
233
+ if (challengeComment.downvoted) {
234
+ return { close: false, reason: 'vetoed' };
235
+ }
236
+
237
+ if (laterUserComments > 0) {
238
+ return { close: false, reason: 'reporter-responded' };
239
+ }
240
+
241
+ const nowMs = toMs(now);
242
+ const createdMs = toMs(challengeComment.createdAt);
243
+
244
+ if (!Number.isFinite(nowMs) || !Number.isFinite(createdMs)) {
245
+ return { close: false, reason: 'invalid-timestamp' };
246
+ }
247
+
248
+ const ageHours = (nowMs - createdMs) / 3600000;
249
+
250
+ if (ageHours < windowHours) {
251
+ return { close: false, reason: 'within-window' };
252
+ }
253
+
254
+ return { close: true, reason: 'duplicate-no-response' };
255
+ }
256
+
257
+ // ---------------------------------------------------------------------------
258
+ // Exports
259
+ // ---------------------------------------------------------------------------
260
+
261
+ module.exports = {
262
+ POSSIBLE_DUPLICATE_LABEL,
263
+ HUMAN_REVIEW_LABEL,
264
+ CHALLENGE_MARKER,
265
+ DEFAULT_WINDOW_HOURS,
266
+ DEFAULT_THRESHOLD,
267
+ DEFAULT_MAX_CANDIDATES,
268
+ MIN_TOKEN_LENGTH,
269
+ EXEMPT_LABELS,
270
+ STOPWORDS,
271
+ tokenize,
272
+ diceSimilarity,
273
+ scoreCandidates,
274
+ renderChallengeComment,
275
+ isChallengeComment,
276
+ hasExemptLabel,
277
+ shouldClose,
278
+ };
@@ -0,0 +1,136 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * @file allowlist-ratchet.cjs
5
+ *
6
+ * Reusable "better than a count ratchet" primitives for CI guards.
7
+ *
8
+ * ## Motivation (issue #597)
9
+ *
10
+ * A count ratchet (`assert(offenders.length <= N)`) has a masking blind spot:
11
+ * fixing one offender and introducing a new one keeps the count constant, so a
12
+ * novel defect slips through green. These helpers enforce on IDENTITY instead,
13
+ * making every individual offender visible and requiring monotonic progress
14
+ * toward zero.
15
+ *
16
+ * ## Design
17
+ *
18
+ * Both functions are pure (no I/O, no global state). The `fail` callback is
19
+ * injected by the caller so the same logic can be used with `node:assert.fail`,
20
+ * a custom throw, or a message-collector in unit tests.
21
+ */
22
+
23
+ /**
24
+ * Assert that `current` offenders are all within the known allowlist, and that
25
+ * every entry in the allowlist still offends (forcing the allowlist to shrink
26
+ * as defects are fixed).
27
+ *
28
+ * Fails when:
29
+ * - Any id in `current` is NOT in `known` → novel offender introduced.
30
+ * - Any id in `known` is NOT in `current` → stale allowlist entry must be
31
+ * pruned so the guard ratchets toward zero (the ratchet-DOWN direction).
32
+ *
33
+ * ## Masking blind spot this prevents (issue #597)
34
+ *
35
+ * A count ratchet (`assert(count <= N)`) allows one offender to be silently
36
+ * replaced by another while the count stays at N. By asserting on identity
37
+ * instead, every new offender is caught by name, and every fixed offender
38
+ * forces the allowlist to shrink.
39
+ *
40
+ * @param {object} opts
41
+ * @param {string} opts.label - Human-readable name for the guard (used
42
+ * in failure messages).
43
+ * @param {Iterable<string>} opts.current - The offending ids found in the
44
+ * current run.
45
+ * @param {Iterable<string>} opts.known - The allowlisted ids (baseline).
46
+ * @param {function(string): void} opts.fail - Callback invoked with a
47
+ * descriptive message on any violation.
48
+ * Pass `require('node:assert').fail`, a
49
+ * custom thrower, or a collector. The
50
+ * function is NOT imported here so callers
51
+ * control the failure mode.
52
+ * @param {string} [opts.pruneHint] - Optional hint appended to the stale-
53
+ * entry failure message (e.g. the name of
54
+ * the allowlist file to edit).
55
+ * @returns {{ novel: string[], stale: string[] }} Sorted arrays of novel ids
56
+ * (in current but not known) and stale ids (in known but not current).
57
+ */
58
+ function assertWithinAllowlist({ label, current, known, fail, pruneHint }) {
59
+ const currentSet = new Set(current);
60
+ const knownSet = new Set(known);
61
+
62
+ const novel = [...currentSet].filter((id) => !knownSet.has(id)).sort();
63
+ const stale = [...knownSet].filter((id) => !currentSet.has(id)).sort();
64
+
65
+ if (novel.length > 0) {
66
+ const list = novel.map((id) => ` - ${id}`).join('\n');
67
+ fail(
68
+ `[${label}] ${novel.length} NEW offender(s) introduced — fix at the source; do not just add to the allowlist.\n${list}`
69
+ );
70
+ }
71
+
72
+ if (stale.length > 0) {
73
+ const list = stale.map((id) => ` - ${id}`).join('\n');
74
+ const hint = pruneHint ? `\n(${pruneHint})` : '';
75
+ fail(
76
+ `[${label}] ${stale.length} allowlisted id(s) no longer offend and MUST be pruned so the guard ratchets toward zero.${hint}\n${list}`
77
+ );
78
+ }
79
+
80
+ return { novel, stale };
81
+ }
82
+
83
+ /**
84
+ * Assert that an artifact's measured maximum stays within a declared ceiling,
85
+ * and that the ceiling itself does not creep above the high-water mark (budgets
86
+ * may only decrease, not increase over time).
87
+ *
88
+ * Fails when:
89
+ * - `actualMax > ceiling` → regression: artifact exceeds budget.
90
+ * - `ceiling - actualMax > grace` → ceiling sits too far above the measured
91
+ * value; tighten it toward `actualMax`.
92
+ *
93
+ * ## Masking blind spot this prevents (issue #597)
94
+ *
95
+ * A plain `assert(size <= ceiling)` with a ceiling set generously high allows
96
+ * the artifact to grow unchecked as long as it stays under the ceiling. The
97
+ * `grace` band forces the ceiling to stay close to the high-water mark,
98
+ * ensuring that any upward creep is immediately visible.
99
+ *
100
+ * @param {object} opts
101
+ * @param {string} opts.label - Human-readable name for the guard (used
102
+ * in failure messages).
103
+ * @param {number} opts.actualMax - The measured value (e.g. bundle size in
104
+ * bytes, line count).
105
+ * @param {number} opts.ceiling - The declared budget ceiling.
106
+ * @param {number} opts.grace - Maximum allowed slack (`ceiling -
107
+ * actualMax`) before the ceiling is
108
+ * considered too loose.
109
+ * @param {function(string): void} opts.fail - Callback invoked with a
110
+ * descriptive message on any violation.
111
+ * @returns {{ ok: boolean, slack: number }} Whether both checks passed and the
112
+ * current slack value.
113
+ */
114
+ function assertTightCeiling({ label, actualMax, ceiling, grace, fail }) {
115
+ const slack = ceiling - actualMax;
116
+ let ok = true;
117
+
118
+ if (actualMax > ceiling) {
119
+ ok = false;
120
+ fail(
121
+ `[${label}] Regression: artifact value ${actualMax} exceeds budget ceiling ${ceiling}. ` +
122
+ `Raise the ceiling to at most ${actualMax} only if the increase is justified.`
123
+ );
124
+ } else if (slack > grace) {
125
+ ok = false;
126
+ fail(
127
+ `[${label}] Ceiling ${ceiling} sits too far above the high-water mark ${actualMax} ` +
128
+ `(slack ${slack} > grace ${grace}). Tighten the ceiling toward ${actualMax}. ` +
129
+ `Budgets may only decrease.`
130
+ );
131
+ }
132
+
133
+ return { ok, slack };
134
+ }
135
+
136
+ module.exports = { assertWithinAllowlist, assertTightCeiling };
@@ -0,0 +1,56 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Error that carries a process exit code. CLI logic throws this instead of
5
+ * calling process.exit() (banned by n/no-process-exit); runMain() translates it
6
+ * into process.exitCode at the entrypoint.
7
+ *
8
+ * @param {number} code exit code (default 1)
9
+ * @param {string} [message] optional human message; when set and code != 0 it is
10
+ * written to stderr by runMain before the process exits.
11
+ */
12
+ class ExitError extends Error {
13
+ constructor(code = 1, message) {
14
+ super(message === undefined ? `process exit ${code}` : message);
15
+ this.name = 'ExitError';
16
+ this.code = code;
17
+ // Whether runMain should print this.message to stderr (only when a real
18
+ // message was provided, not the synthetic default).
19
+ this.hasUserMessage = message !== undefined;
20
+ }
21
+ }
22
+
23
+ /**
24
+ * Run a CLI main function and translate its outcome into process.exitCode
25
+ * (never process.exit(), so n/no-process-exit stays satisfied). Supports sync or
26
+ * async main.
27
+ * - main returns a number -> process.exitCode = that number
28
+ * - main throws/rejects ExitError -> process.exitCode = err.code, and if
29
+ * err.hasUserMessage && err.code !== 0, err.message is written to stderr
30
+ * - main throws/rejects anything else -> the stack is written to stderr and
31
+ * process.exitCode = 1
32
+ * Letting the event loop drain (vs process.exit) means buffered stdout/stderr is
33
+ * flushed and process.on('exit') cleanup handlers still fire.
34
+ *
35
+ * @param {() => (number|void|Promise<number|void>)} main
36
+ */
37
+ function runMain(main) {
38
+ Promise.resolve()
39
+ .then(() => main())
40
+ .then((code) => {
41
+ if (typeof code === 'number') process.exitCode = code;
42
+ })
43
+ .catch((err) => {
44
+ if (err instanceof ExitError) {
45
+ if (err.hasUserMessage && err.code !== 0) {
46
+ process.stderr.write(`${err.message}\n`);
47
+ }
48
+ process.exitCode = err.code;
49
+ return;
50
+ }
51
+ process.stderr.write(`${err && err.stack ? err.stack : String(err)}\n`);
52
+ process.exitCode = 1;
53
+ });
54
+ }
55
+
56
+ module.exports = { ExitError, runMain };
@@ -0,0 +1,114 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * lint-command-contract.cjs (ADR-0002)
4
+ *
5
+ * Enforces the commands/gsd/*.md contract across all 65 command files:
6
+ *
7
+ * 1. name: present, non-empty, matches gsd: or gsd- prefix
8
+ * 2. description: present, non-empty
9
+ * 3. allowed-tools: block present, non-empty, all entries from CANONICAL_TOOLS
10
+ * 4. execution_context @-refs: every @-reference resolves to an existing file on disk
11
+ * 5. execution_context @-refs: each appears on its own line (no trailing prose)
12
+ *
13
+ * Exit 0 = clean. Exit 1 = violations (with diagnostics).
14
+ */
15
+
16
+ 'use strict';
17
+
18
+ const fs = require('fs');
19
+ const path = require('path');
20
+
21
+ const ROOT = path.join(__dirname, '..');
22
+ const COMMANDS_DIR = path.join(ROOT, 'commands', 'gsd');
23
+ const GSD_ROOT = path.join(ROOT, 'gsd-core');
24
+
25
+ const {
26
+ CANONICAL_TOOLS,
27
+ parseFrontmatter,
28
+ executionContextRefs: extractExecutionContextRefs,
29
+ } = require('./command-contract-helpers.cjs');
30
+
31
+ const { runMain } = require('./lib/cli-exit.cjs');
32
+
33
+ // ─── check one file ───────────────────────────────────────────────────────────
34
+
35
+ function check(filePath) {
36
+ const content = fs.readFileSync(filePath, 'utf-8');
37
+ const rel = path.relative(ROOT, filePath);
38
+ const fm = parseFrontmatter(content);
39
+ const violations = [];
40
+
41
+ // 1. name: present + gsd: / gsd- prefix
42
+ if (!fm.name || !fm.name.trim()) {
43
+ violations.push('name: field missing or empty');
44
+ } else if (!/^gsd[:-]/.test(fm.name.trim())) {
45
+ violations.push(`name: must start with "gsd:" or "gsd-", got "${fm.name.trim()}"`);
46
+ }
47
+
48
+ // 2. description: present + non-empty
49
+ if (!fm.description || !fm.description.trim()) {
50
+ violations.push('description: field missing or empty');
51
+ }
52
+
53
+ // 3. allowed-tools: present + non-empty + all entries canonical
54
+ if (!fm['allowed-tools'] || !fm['allowed-tools'].trim()) {
55
+ violations.push('allowed-tools: block missing or empty');
56
+ } else {
57
+ const tools = fm['allowed-tools'].split('\n').map(t => t.trim()).filter(Boolean);
58
+ for (const tool of tools) {
59
+ const valid =
60
+ CANONICAL_TOOLS.has(tool) ||
61
+ (tool.startsWith('mcp__context7__') && CANONICAL_TOOLS.has('mcp__context7__*'));
62
+ if (!valid) violations.push(`allowed-tools: unknown tool "${tool}"`);
63
+ }
64
+ }
65
+
66
+ // 4+5. execution_context @-refs resolve + no trailing prose
67
+ const refs = extractExecutionContextRefs(content);
68
+ for (const { token, normalized, trailingProse } of refs) {
69
+ const absPath = path.join(GSD_ROOT, normalized);
70
+ if (!fs.existsSync(absPath)) {
71
+ violations.push(`execution_context: @-ref "${normalized}" does not exist on disk`);
72
+ }
73
+ if (trailingProse) {
74
+ violations.push(`execution_context: @-ref "${token}" has trailing prose on the same line`);
75
+ }
76
+ }
77
+
78
+ if (violations.length === 0) return null;
79
+ return { file: rel, violations };
80
+ }
81
+
82
+ // ─── run ─────────────────────────────────────────────────────────────────────
83
+
84
+ function main() {
85
+ const commandFiles = fs
86
+ .readdirSync(COMMANDS_DIR)
87
+ .filter(f => f.endsWith('.md'))
88
+ .map(f => path.join(COMMANDS_DIR, f));
89
+
90
+ const results = commandFiles.map(check).filter(Boolean);
91
+
92
+ if (results.length === 0) {
93
+ console.log(
94
+ `ok lint-command-contract: ${commandFiles.length} command files checked, 0 violations`,
95
+ );
96
+ return 0;
97
+ }
98
+
99
+ const total = results.reduce((n, r) => n + r.violations.length, 0);
100
+ process.stderr.write(
101
+ `\nERROR lint-command-contract: ${total} violation(s) across ${results.length} file(s)\n\n`,
102
+ );
103
+ for (const r of results) {
104
+ process.stderr.write(` ${r.file}\n`);
105
+ for (const v of r.violations) {
106
+ process.stderr.write(` - ${v}\n`);
107
+ }
108
+ process.stderr.write('\n');
109
+ }
110
+ process.stderr.write('See docs/adr/0002-command-contract-validation-module.md for the contract spec.\n\n');
111
+ return 1;
112
+ }
113
+
114
+ runMain(main);
@@ -0,0 +1,87 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * lint-descriptions.cjs
4
+ *
5
+ * Enforces the 100-char description budget for commands/gsd/*.md files.
6
+ *
7
+ * Usage:
8
+ * node scripts/lint-descriptions.cjs [file.md ...]
9
+ *
10
+ * If no args are given, scans commands/gsd/ automatically.
11
+ * Exits 1 if any description exceeds 100 chars; exits 0 if all pass.
12
+ */
13
+
14
+ 'use strict';
15
+
16
+ const fs = require('fs');
17
+ const path = require('path');
18
+ const { ExitError, runMain } = require('./lib/cli-exit.cjs');
19
+
20
+ const MAX_LENGTH = 100;
21
+ const COMMANDS_DIR = path.join(__dirname, '..', 'commands', 'gsd');
22
+
23
+ /**
24
+ * Parse the description field from frontmatter in a .md file.
25
+ * Returns null if no description is found.
26
+ */
27
+ function parseDescription(content) {
28
+ const fmMatch = content.match(/^---\r?\n([\s\S]*?)\r?\n---/);
29
+ if (!fmMatch) return null;
30
+ const fm = fmMatch[1];
31
+
32
+ const quoted = fm.match(/^description:\s+"((?:[^"\\]|\\.)*)"\s*$/m);
33
+ if (quoted) return quoted[1];
34
+
35
+ const plain = fm.match(/^description:\s+(.+)$/m);
36
+ if (plain) return plain[1].trim();
37
+
38
+ return null;
39
+ }
40
+
41
+ function getFiles() {
42
+ if (process.argv.length > 2) {
43
+ return process.argv.slice(2);
44
+ }
45
+ return fs.readdirSync(COMMANDS_DIR)
46
+ .filter(f => f.endsWith('.md'))
47
+ .map(f => path.join(COMMANDS_DIR, f));
48
+ }
49
+
50
+ function main() {
51
+ const files = getFiles();
52
+ const violations = [];
53
+
54
+ for (const filePath of files) {
55
+ let content;
56
+ try {
57
+ content = fs.readFileSync(filePath, 'utf-8');
58
+ } catch (err) {
59
+ throw new ExitError(1, `ERROR: Cannot read file: ${filePath}\n ${err.message}`);
60
+ }
61
+
62
+ const description = parseDescription(content);
63
+ if (description === null) continue;
64
+
65
+ if (description.length > MAX_LENGTH) {
66
+ violations.push({ filePath, length: description.length, description });
67
+ }
68
+ }
69
+
70
+ if (violations.length === 0) {
71
+ const checked = files.length;
72
+ process.stdout.write(`ok lint-descriptions: ${checked} file(s) checked, 0 violations\n`);
73
+ return 0;
74
+ }
75
+
76
+ process.stderr.write(`\nERROR lint-descriptions: ${violations.length} violation(s) found\n\n`);
77
+ for (const v of violations) {
78
+ const preview = v.description.length > 120 ? v.description.slice(0, 117) + '...' : v.description;
79
+ process.stderr.write(` ${v.filePath}\n`);
80
+ process.stderr.write(` Length : ${v.length} (max ${MAX_LENGTH})\n`);
81
+ process.stderr.write(` Desc : ${preview}\n\n`);
82
+ }
83
+ process.stderr.write(`Trim descriptions to <= ${MAX_LENGTH} chars. Flag docs belong in argument-hint:.\n\n`);
84
+ return 1;
85
+ }
86
+
87
+ runMain(main);