@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,373 @@
1
+ #!/usr/bin/env node
2
+ 'use strict';
3
+
4
+ const cp = require('node:child_process');
5
+ const fs = require('fs');
6
+ const path = require('path');
7
+ const { ExitError, runMain } = require('../lib/cli-exit.cjs');
8
+ const { classifyTitle } = require('./format-github-release-notes.cjs');
9
+
10
+ const DEFAULT_MAX_CONTENT = 1850;
11
+ const SECTION_LIMITS = Object.freeze([
12
+ ['Breaking', 4],
13
+ ['Security', 4],
14
+ ['Feature', 4],
15
+ ['Fix', 4],
16
+ ['Enhancement', 3],
17
+ ['Internal', 2],
18
+ ['New Contributors', 3],
19
+ ]);
20
+
21
+ function normalizeNewlines(value) {
22
+ return String(value || '').replace(/\r\n/g, '\n');
23
+ }
24
+
25
+ function releaseTag(release) {
26
+ return release.tagName || release.tag_name || release.name || 'release';
27
+ }
28
+
29
+ function releaseName(release) {
30
+ return release.name || releaseTag(release);
31
+ }
32
+
33
+ function releaseUrl(release) {
34
+ return release.url || release.html_url || '';
35
+ }
36
+
37
+ function releaseIsPrerelease(release) {
38
+ return release.isPrerelease === true || release.prerelease === true || /-/.test(releaseTag(release));
39
+ }
40
+
41
+ function loadPackageName(repoRoot = path.resolve(__dirname, '..', '..')) {
42
+ const pkgJson = JSON.parse(fs.readFileSync(path.join(repoRoot, 'package.json'), 'utf8'));
43
+ return pkgJson.name || 'package';
44
+ }
45
+
46
+ function normalizeHeading(value) {
47
+ const heading = value.trim().replace(/:$/, '');
48
+ const lower = heading.toLowerCase();
49
+
50
+ if (lower.includes('breaking')) return 'Breaking';
51
+ if (lower.includes('security')) return 'Security';
52
+ if (lower === 'feature' || lower === 'features') return 'Feature';
53
+ if (lower === 'fix' || lower === 'fixes' || lower === 'bug fixes') return 'Fix';
54
+ if (lower === 'enhancement' || lower === 'enhancements' || lower === 'improvements') return 'Enhancement';
55
+ if (lower === 'internal' || lower === 'maintenance') return 'Internal';
56
+ if (lower === 'new contributors') return 'New Contributors';
57
+ if (lower === "what's changed" || lower === 'whats changed') return "What's Changed";
58
+ return heading;
59
+ }
60
+
61
+ function collectSections(body) {
62
+ const sections = new Map();
63
+ let current = null;
64
+
65
+ for (const line of normalizeNewlines(body).split('\n')) {
66
+ const trimmed = line.trim();
67
+ const heading = trimmed.match(/^#{2,3}\s+(.+)$/);
68
+
69
+ if (heading) {
70
+ current = normalizeHeading(heading[1]);
71
+ continue;
72
+ }
73
+
74
+ const bullet = trimmed.match(/^([*-])\s+(.+)$/);
75
+ if (!bullet || !current) continue;
76
+
77
+ let section = current;
78
+ if (section === "What's Changed") {
79
+ section = classifyTitle(trimmed);
80
+ }
81
+
82
+ if (!sections.has(section)) sections.set(section, []);
83
+ sections.get(section).push(cleanBullet(trimmed));
84
+ }
85
+
86
+ return sections;
87
+ }
88
+
89
+ function cleanBullet(value) {
90
+ let text = String(value)
91
+ .replace(/^[*-]\s+/, '')
92
+ .replace(/\s+by\s+@[A-Za-z0-9-]+\s+in\s+https:\/\/github\.com\/[^/\s]+\/[^/\s]+\/pull\/(\d+)/g, ' (#$1)')
93
+ .replace(/\[([^\]]+)\]\((https:\/\/github\.com\/[^/\s]+\/[^/\s]+\/(?:pull|issues)\/(\d+))\)/g, '$1 (#$3)')
94
+ .replace(/\[([^\]]+)\]\((https?:\/\/[^)]+)\)/g, '$1')
95
+ .replace(/https:\/\/github\.com\/[^/\s]+\/[^/\s]+\/pull\/(\d+)/g, '#$1')
96
+ .replace(/https:\/\/github\.com\/[^/\s]+\/[^/\s]+\/issues\/(\d+)/g, '#$1')
97
+ .replace(/https:\/\/github\.com\/[^/\s]+\/[^/\s]+\/commit\/([0-9a-f]{7})[0-9a-f]*/gi, '$1')
98
+ .replace(/\*\*/g, '')
99
+ .replace(/\s+/g, ' ')
100
+ .trim();
101
+
102
+ text = text
103
+ .replace(/^\[codex\]\s+/i, '')
104
+ .replace(/^(?:feat|fix|docs|chore|ci|refactor|test)(?:\([^)]+\))?!?:\s+/i, '');
105
+ text = text.replace(/#(\d+)\s+\(#\1\)/g, '#$1');
106
+ text = text.replace(/\s+\(#(\d+)\)\s+\(#\1\)$/g, ' (#$1)');
107
+
108
+ return truncateAtWord(text, 220);
109
+ }
110
+
111
+ function truncateAtWord(value, maxLength) {
112
+ if (value.length <= maxLength) return value;
113
+ const splitAt = value.lastIndexOf(' ', maxLength - 3);
114
+ if (splitAt < 80) return `${value.slice(0, maxLength - 3)}...`;
115
+ return `${value.slice(0, splitAt)}...`;
116
+ }
117
+
118
+ function extractInstallCommand(body, packageName, release) {
119
+ const lines = normalizeNewlines(body).split('\n');
120
+ let inInstall = false;
121
+ let inFence = false;
122
+
123
+ for (const line of lines) {
124
+ const trimmed = line.trim();
125
+
126
+ if (/^##\s+Install\b/i.test(trimmed)) {
127
+ inInstall = true;
128
+ continue;
129
+ }
130
+
131
+ if (inInstall && /^##\s+/.test(trimmed)) break;
132
+ if (!inInstall) continue;
133
+
134
+ if (trimmed.startsWith('```')) {
135
+ inFence = !inFence;
136
+ continue;
137
+ }
138
+
139
+ if (inFence && /^npm\s+(?:i|install)\s+/.test(trimmed)) {
140
+ return trimmed;
141
+ }
142
+ }
143
+
144
+ const channel = releaseIsPrerelease(release) ? 'next' : 'latest';
145
+ return `npm i ${packageName}@${channel}`;
146
+ }
147
+
148
+ function buildDiscordReleasePayload({ release, packageName = loadPackageName(), maxContent = DEFAULT_MAX_CONTENT }) {
149
+ const tag = releaseTag(release);
150
+ const name = releaseName(release);
151
+ const url = releaseUrl(release);
152
+ const prerelease = releaseIsPrerelease(release);
153
+ const channel = prerelease ? 'next' : 'latest';
154
+ const body = release.body || '';
155
+ const sections = collectSections(body);
156
+ const footer = url ? `Full changelog: ${url}` : `Full changelog: ${tag}`;
157
+ const baseLines = [
158
+ `**${packageName} ${name}${prerelease ? ' pre-release' : ''} is out**`,
159
+ '',
160
+ 'Install:',
161
+ `\`${extractInstallCommand(body, packageName, release)}\``,
162
+ ];
163
+
164
+ const lines = baseLines.slice();
165
+ for (const [section, limit] of SECTION_LIMITS) {
166
+ appendSection(lines, section, sections.get(section) || [], limit, footer, maxContent);
167
+ }
168
+
169
+ if (lines.length === baseLines.length) {
170
+ appendLineGroup(lines, ['', 'No release notes were provided.'], footer, maxContent);
171
+ }
172
+
173
+ if (joinedLength(lines, footer) > maxContent) {
174
+ while (lines.length > baseLines.length && joinedLength(lines, footer) > maxContent) {
175
+ lines.pop();
176
+ }
177
+ }
178
+
179
+ const content = [...lines, '', footer].join('\n');
180
+ return {
181
+ username: 'GSD Releases',
182
+ content,
183
+ embeds: [
184
+ {
185
+ title: 'Release details',
186
+ url: url || undefined,
187
+ color: prerelease ? 0xd9a441 : 0x2f9e44,
188
+ fields: [
189
+ { name: 'Package', value: `\`${packageName}\``, inline: true },
190
+ { name: 'Channel', value: `\`${channel}\``, inline: true },
191
+ { name: 'Version', value: `\`${tag}\``, inline: true },
192
+ ],
193
+ },
194
+ ],
195
+ allowed_mentions: { parse: [] },
196
+ };
197
+ }
198
+
199
+ function appendSection(lines, section, bullets, limit, footer, maxContent) {
200
+ if (bullets.length === 0) return;
201
+
202
+ const label = sectionLabel(section);
203
+ const selected = [];
204
+ for (const bullet of bullets.slice(0, limit)) {
205
+ const candidate = ['', `**${label}**`, ...selected.map((item) => `- ${item}`), `- ${bullet}`];
206
+ if (joinedLength([...lines, ...candidate], footer) > maxContent) break;
207
+ selected.push(bullet);
208
+ }
209
+
210
+ if (selected.length === 0) return;
211
+
212
+ const group = ['', `**${label}**`, ...selected.map((item) => `- ${item}`)];
213
+ const remaining = bullets.length - selected.length;
214
+ if (remaining > 0) {
215
+ const moreLine = `- ...and ${remaining} more ${label.toLowerCase()}`;
216
+ if (joinedLength([...lines, ...group, moreLine], footer) <= maxContent) {
217
+ group.push(moreLine);
218
+ }
219
+ }
220
+
221
+ appendLineGroup(lines, group, footer, maxContent);
222
+ }
223
+
224
+ function appendLineGroup(lines, group, footer, maxContent) {
225
+ if (joinedLength([...lines, ...group], footer) <= maxContent) {
226
+ lines.push(...group);
227
+ }
228
+ }
229
+
230
+ function joinedLength(lines, footer) {
231
+ return [...lines, '', footer].join('\n').length;
232
+ }
233
+
234
+ function sectionLabel(section) {
235
+ if (section === 'Feature') return 'Features';
236
+ if (section === 'Fix') return 'Fixes';
237
+ if (section === 'Enhancement') return 'Enhancements';
238
+ return section;
239
+ }
240
+
241
+ function fetchRelease({ tag, repo, latest = false }) {
242
+ const args = ['release', 'view'];
243
+ if (!latest && tag) args.push(tag);
244
+ args.push('--json', 'tagName,name,isPrerelease,body,url');
245
+ if (repo) args.push('--repo', repo);
246
+ const raw = cp.execFileSync('gh', args, { encoding: 'utf8' });
247
+ return JSON.parse(raw);
248
+ }
249
+
250
+ async function postWebhook(webhookUrl, payload) {
251
+ const response = await fetch(webhookUrl, {
252
+ method: 'POST',
253
+ headers: { 'content-type': 'application/json' },
254
+ body: JSON.stringify(payload),
255
+ });
256
+
257
+ if (!response.ok) {
258
+ throw new ExitError(1, `Discord webhook failed with ${response.status}: ${await response.text()}`);
259
+ }
260
+ }
261
+
262
+ function parseArgs(argv) {
263
+ const opts = {
264
+ tag: null,
265
+ repo: null,
266
+ latest: false,
267
+ stdin: false,
268
+ post: false,
269
+ json: false,
270
+ allowMissingWebhook: false,
271
+ packageName: null,
272
+ maxContent: DEFAULT_MAX_CONTENT,
273
+ };
274
+
275
+ const args = argv.slice();
276
+ while (args.length > 0) {
277
+ const arg = args.shift();
278
+ if (arg === '--tag') {
279
+ opts.tag = requireValue(args, arg);
280
+ } else if (arg === '--repo') {
281
+ opts.repo = requireValue(args, arg);
282
+ } else if (arg === '--package') {
283
+ opts.packageName = requireValue(args, arg);
284
+ } else if (arg === '--max-content') {
285
+ opts.maxContent = Number(requireValue(args, arg));
286
+ } else if (arg === '--latest') {
287
+ opts.latest = true;
288
+ } else if (arg === '--stdin') {
289
+ opts.stdin = true;
290
+ } else if (arg === '--post') {
291
+ opts.post = true;
292
+ } else if (arg === '--json') {
293
+ opts.json = true;
294
+ } else if (arg === '--allow-missing-webhook') {
295
+ opts.allowMissingWebhook = true;
296
+ } else if (arg === '--help' || arg === '-h') {
297
+ process.stdout.write(
298
+ 'Usage: node scripts/release-notes/discord-release-summary.cjs [--tag <tag>|--latest|--stdin] [options]\n' +
299
+ '\n' +
300
+ 'Options:\n' +
301
+ ' --tag <tag> GitHub release tag to announce\n' +
302
+ ' --latest Announce the latest GitHub release\n' +
303
+ ' --stdin Read release JSON from stdin\n' +
304
+ ' --repo <owner/repo> Repository for gh release view\n' +
305
+ ' --package <name> Package name override\n' +
306
+ ' --post POST to DISCORD_WEBHOOK_URL\n' +
307
+ ' --allow-missing-webhook Warn and skip instead of failing when posting without a webhook\n' +
308
+ ' --json Print webhook payload JSON instead of content preview\n' +
309
+ ' --max-content <n> Max Discord content characters before webhook footer\n'
310
+ );
311
+ throw new ExitError(0);
312
+ } else {
313
+ throw new ExitError(2, `error: unknown argument ${arg}`);
314
+ }
315
+ }
316
+
317
+ if (!Number.isFinite(opts.maxContent) || opts.maxContent < 500 || opts.maxContent > 2000) {
318
+ throw new ExitError(2, 'error: --max-content must be between 500 and 2000');
319
+ }
320
+
321
+ return opts;
322
+ }
323
+
324
+ function requireValue(args, flag) {
325
+ const value = args.shift();
326
+ if (!value || value.startsWith('-')) {
327
+ throw new ExitError(2, `error: ${flag} requires a value`);
328
+ }
329
+ return value;
330
+ }
331
+
332
+ async function main() {
333
+ const opts = parseArgs(process.argv.slice(2));
334
+ let release;
335
+
336
+ if (opts.stdin) {
337
+ release = JSON.parse(fs.readFileSync('/dev/stdin', 'utf8'));
338
+ } else {
339
+ release = fetchRelease(opts);
340
+ }
341
+
342
+ const payload = buildDiscordReleasePayload({
343
+ release,
344
+ packageName: opts.packageName || loadPackageName(),
345
+ maxContent: opts.maxContent,
346
+ });
347
+
348
+ if (opts.post) {
349
+ const webhookUrl = process.env.DISCORD_WEBHOOK_URL;
350
+ if (!webhookUrl) {
351
+ if (opts.allowMissingWebhook) {
352
+ process.stderr.write('warning: DISCORD_WEBHOOK_URL is not set; skipping Discord announcement\n');
353
+ } else {
354
+ throw new ExitError(1, 'error: DISCORD_WEBHOOK_URL is not set');
355
+ }
356
+ } else {
357
+ await postWebhook(webhookUrl, payload);
358
+ }
359
+ }
360
+
361
+ process.stdout.write(opts.json ? `${JSON.stringify(payload, null, 2)}\n` : `${payload.content}\n`);
362
+ }
363
+
364
+ if (require.main === module) {
365
+ runMain(main);
366
+ }
367
+
368
+ module.exports = {
369
+ buildDiscordReleasePayload,
370
+ cleanBullet,
371
+ collectSections,
372
+ extractInstallCommand,
373
+ };
@@ -0,0 +1,261 @@
1
+ 'use strict';
2
+
3
+ const path = require('path');
4
+ const os = require('os');
5
+ const fs = require('fs');
6
+ const { execFileSync } = require('child_process');
7
+ const { runMain, ExitError } = require('../lib/cli-exit.cjs');
8
+
9
+ /**
10
+ * Classify a What's-Changed bullet line into 'Feature', 'Fix', or 'Enhancement'.
11
+ * @param {string} bulletLine - Full bullet line including the leading `* ` or `- ` marker.
12
+ * @returns {'Feature'|'Fix'|'Enhancement'}
13
+ */
14
+ function classifyTitle(bulletLine) {
15
+ // Strip leading `* ` or `- ` marker
16
+ const withoutMarker = bulletLine.replace(/^[*-]\s+/, '');
17
+
18
+ // Extract title = text before ` by @`
19
+ const byIdx = withoutMarker.indexOf(' by @');
20
+ const title = (byIdx !== -1 ? withoutMarker.slice(0, byIdx) : withoutMarker).trim();
21
+
22
+ if (/^feat(?:ure)?\s*(?:\(|!|:)/i.test(title)) return 'Feature';
23
+ if (/^fix\s*(?:\(|!|:)/i.test(title)) return 'Fix';
24
+ return 'Enhancement';
25
+ }
26
+
27
+ /**
28
+ * Reformat GitHub's auto-generated release notes into the repo's hand-curated format.
29
+ *
30
+ * @param {object} opts
31
+ * @param {string} opts.generatedBody - The raw GitHub-generated release body.
32
+ * @param {string} opts.version - Version string (e.g. "1.3.0-rc.1"), no leading "v".
33
+ * @param {boolean} opts.prerelease - Whether this is a pre-release.
34
+ * @param {string} opts.packageName - npm package name (e.g. "@therocketcode/gsd-core").
35
+ * @returns {string} Formatted release body (no trailing newline).
36
+ */
37
+ function formatReleaseNotes({ generatedBody, version, prerelease, packageName }) {
38
+ const lines = generatedBody.split('\n');
39
+
40
+ const featureBullets = [];
41
+ const fixBullets = [];
42
+ const enhancementBullets = [];
43
+ const newContributorBullets = [];
44
+ let fullChangelogLine = null;
45
+
46
+ let inWhatsChanged = false;
47
+ let inNewContributors = false;
48
+
49
+ for (const line of lines) {
50
+ const trimmed = line.trim();
51
+
52
+ // Detect section headings
53
+ if (trimmed === '## What\'s Changed') {
54
+ inWhatsChanged = true;
55
+ inNewContributors = false;
56
+ continue;
57
+ }
58
+
59
+ if (trimmed === '## New Contributors') {
60
+ inWhatsChanged = false;
61
+ inNewContributors = true;
62
+ continue;
63
+ }
64
+
65
+ // Full changelog line ends the What's Changed section
66
+ if (trimmed.startsWith('**Full Changelog**:')) {
67
+ inWhatsChanged = false;
68
+ inNewContributors = false;
69
+ fullChangelogLine = trimmed;
70
+ continue;
71
+ }
72
+
73
+ // Any other `##` heading ends current section
74
+ if (trimmed.startsWith('## ')) {
75
+ inWhatsChanged = false;
76
+ inNewContributors = false;
77
+ continue;
78
+ }
79
+
80
+ // Collect bullets
81
+ if (inWhatsChanged && (trimmed.startsWith('* ') || trimmed.startsWith('- '))) {
82
+ const category = classifyTitle(trimmed);
83
+ if (category === 'Feature') featureBullets.push(trimmed);
84
+ else if (category === 'Fix') fixBullets.push(trimmed);
85
+ else enhancementBullets.push(trimmed);
86
+ continue;
87
+ }
88
+
89
+ if (inNewContributors && (trimmed.startsWith('* ') || trimmed.startsWith('- '))) {
90
+ newContributorBullets.push(trimmed);
91
+ continue;
92
+ }
93
+ }
94
+
95
+ // Build Install block
96
+ let installBlock;
97
+ if (prerelease) {
98
+ installBlock = [
99
+ '## Install',
100
+ '',
101
+ 'This pre-release is published to npm under the `next` dist-tag.',
102
+ '',
103
+ '```bash',
104
+ `npm i ${packageName}@${version}`,
105
+ '# or',
106
+ `npm i ${packageName}@next`,
107
+ '```',
108
+ ].join('\n');
109
+ } else {
110
+ installBlock = [
111
+ '## Install',
112
+ '',
113
+ '```bash',
114
+ `npm i ${packageName}@${version}`,
115
+ '# or',
116
+ `npm i ${packageName}@latest`,
117
+ '```',
118
+ ].join('\n');
119
+ }
120
+
121
+ // Assemble groups (omit empty ones)
122
+ const groups = [];
123
+
124
+ // Group A: Install
125
+ groups.push(installBlock);
126
+
127
+ // Group B: What's Changed heading
128
+ groups.push('## What\'s Changed');
129
+
130
+ // Group C: Features
131
+ if (featureBullets.length > 0) {
132
+ groups.push('### Feature\n' + featureBullets.join('\n'));
133
+ }
134
+
135
+ // Group D: Enhancements
136
+ if (enhancementBullets.length > 0) {
137
+ groups.push('### Enhancement\n' + enhancementBullets.join('\n'));
138
+ }
139
+
140
+ // Group E: Fixes
141
+ if (fixBullets.length > 0) {
142
+ groups.push('### Fix\n' + fixBullets.join('\n'));
143
+ }
144
+
145
+ // Group F: New Contributors
146
+ if (newContributorBullets.length > 0) {
147
+ groups.push('## New Contributors\n' + newContributorBullets.join('\n'));
148
+ }
149
+
150
+ // Group G: Full Changelog
151
+ if (fullChangelogLine) {
152
+ groups.push(fullChangelogLine);
153
+ }
154
+
155
+ return groups.join('\n\n');
156
+ }
157
+
158
+ // CLI entry point
159
+ function main() {
160
+ try {
161
+ const argv = process.argv.slice(2);
162
+
163
+ let tag = null;
164
+ let repo = null;
165
+ let packageName = null;
166
+ let prerelease = null;
167
+ let useStdin = false;
168
+ let doApply = false;
169
+
170
+ for (let i = 0; i < argv.length; i++) {
171
+ const arg = argv[i];
172
+ if (arg === '--tag') {
173
+ tag = argv[++i];
174
+ } else if (arg === '--repo') {
175
+ repo = argv[++i];
176
+ } else if (arg === '--package') {
177
+ packageName = argv[++i];
178
+ } else if (arg === '--prerelease') {
179
+ prerelease = true;
180
+ } else if (arg === '--latest') {
181
+ prerelease = false;
182
+ } else if (arg === '--stdin') {
183
+ useStdin = true;
184
+ } else if (arg === '--apply') {
185
+ doApply = true;
186
+ }
187
+ }
188
+
189
+ // Derive version from tag
190
+ const version = tag ? tag.replace(/^v/, '') : null;
191
+
192
+ // Resolve package name if not provided
193
+ if (!packageName) {
194
+ const repoRoot = path.resolve(__dirname, '..', '..');
195
+ const pkgJson = JSON.parse(fs.readFileSync(path.join(repoRoot, 'package.json'), 'utf8'));
196
+ packageName = pkgJson.name;
197
+ }
198
+
199
+ let generatedBody;
200
+
201
+ if (useStdin) {
202
+ // Read from stdin
203
+ if (!version) {
204
+ throw new Error('--stdin mode requires --tag or --version to derive version');
205
+ }
206
+ if (prerelease === null) {
207
+ throw new Error('--stdin mode requires --prerelease or --latest');
208
+ }
209
+ generatedBody = fs.readFileSync('/dev/stdin', 'utf8');
210
+ } else {
211
+ // Fetch from gh
212
+ if (!tag) {
213
+ throw new Error('--tag <tag> is required');
214
+ }
215
+
216
+ const ghArgs = ['release', 'view', tag, '--json', 'body', '-q', '.body'];
217
+ if (repo) ghArgs.push('--repo', repo);
218
+
219
+ generatedBody = execFileSync('gh', ghArgs, { encoding: 'utf8' });
220
+
221
+ // Determine prerelease if not forced
222
+ if (prerelease === null) {
223
+ try {
224
+ const ghPreArgs = ['release', 'view', tag, '--json', 'isPrerelease', '-q', '.isPrerelease'];
225
+ if (repo) ghPreArgs.push('--repo', repo);
226
+ const result = execFileSync('gh', ghPreArgs, { encoding: 'utf8' }).trim();
227
+ prerelease = result === 'true';
228
+ } catch (_e) {
229
+ // Final fallback: check if tag contains `-` after version digits
230
+ prerelease = /-/.test(version);
231
+ }
232
+ }
233
+ }
234
+
235
+ const formatted = formatReleaseNotes({ generatedBody, version, prerelease, packageName });
236
+
237
+ if (doApply) {
238
+ const tmpFile = path.join(os.tmpdir(), `release-notes-${Date.now()}.md`);
239
+ fs.writeFileSync(tmpFile, formatted, 'utf8');
240
+ try {
241
+ const ghArgs = ['release', 'edit', tag, '--notes-file', tmpFile];
242
+ if (repo) ghArgs.push('--repo', repo);
243
+ execFileSync('gh', ghArgs, { encoding: 'utf8' });
244
+ process.stderr.write(`Release notes updated for ${tag}\n`);
245
+ } finally {
246
+ fs.unlinkSync(tmpFile);
247
+ }
248
+ } else {
249
+ process.stdout.write(formatted + '\n');
250
+ }
251
+ } catch (err) {
252
+ if (err instanceof ExitError) throw err;
253
+ throw new ExitError(1, err && err.message ? err.message : String(err));
254
+ }
255
+ }
256
+
257
+ if (require.main === module) {
258
+ runMain(main);
259
+ }
260
+
261
+ module.exports = { formatReleaseNotes, classifyTitle };