@opengsd/gsd-core 1.2.0-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (503) hide show
  1. package/LICENSE +21 -0
  2. package/README.ja-JP.md +870 -0
  3. package/README.ko-KR.md +861 -0
  4. package/README.md +301 -0
  5. package/README.pt-BR.md +492 -0
  6. package/README.zh-CN.md +842 -0
  7. package/agents/gsd-advisor-researcher.md +127 -0
  8. package/agents/gsd-ai-researcher.md +133 -0
  9. package/agents/gsd-assumptions-analyzer.md +105 -0
  10. package/agents/gsd-code-fixer.md +668 -0
  11. package/agents/gsd-code-reviewer.md +387 -0
  12. package/agents/gsd-codebase-mapper.md +853 -0
  13. package/agents/gsd-debug-session-manager.md +314 -0
  14. package/agents/gsd-debugger.md +1452 -0
  15. package/agents/gsd-doc-classifier.md +168 -0
  16. package/agents/gsd-doc-synthesizer.md +204 -0
  17. package/agents/gsd-doc-verifier.md +217 -0
  18. package/agents/gsd-doc-writer.md +615 -0
  19. package/agents/gsd-domain-researcher.md +153 -0
  20. package/agents/gsd-eval-auditor.md +191 -0
  21. package/agents/gsd-eval-planner.md +154 -0
  22. package/agents/gsd-executor.md +772 -0
  23. package/agents/gsd-framework-selector.md +160 -0
  24. package/agents/gsd-integration-checker.md +470 -0
  25. package/agents/gsd-intel-updater.md +342 -0
  26. package/agents/gsd-nyquist-auditor.md +203 -0
  27. package/agents/gsd-pattern-mapper.md +335 -0
  28. package/agents/gsd-phase-researcher.md +928 -0
  29. package/agents/gsd-plan-checker.md +978 -0
  30. package/agents/gsd-planner.md +1218 -0
  31. package/agents/gsd-project-researcher.md +677 -0
  32. package/agents/gsd-research-synthesizer.md +255 -0
  33. package/agents/gsd-roadmapper.md +688 -0
  34. package/agents/gsd-security-auditor.md +155 -0
  35. package/agents/gsd-ui-auditor.md +495 -0
  36. package/agents/gsd-ui-checker.md +309 -0
  37. package/agents/gsd-ui-researcher.md +380 -0
  38. package/agents/gsd-user-profiler.md +171 -0
  39. package/agents/gsd-verifier.md +917 -0
  40. package/bin/install.js +10936 -0
  41. package/bin/lib/ui-safety-gate.cjs +107 -0
  42. package/commands/gsd/add-tests.md +42 -0
  43. package/commands/gsd/ai-integration-phase.md +37 -0
  44. package/commands/gsd/audit-fix.md +34 -0
  45. package/commands/gsd/audit-milestone.md +37 -0
  46. package/commands/gsd/audit-uat.md +24 -0
  47. package/commands/gsd/autonomous.md +46 -0
  48. package/commands/gsd/capture.md +62 -0
  49. package/commands/gsd/cleanup.md +24 -0
  50. package/commands/gsd/code-review.md +59 -0
  51. package/commands/gsd/complete-milestone.md +143 -0
  52. package/commands/gsd/config.md +56 -0
  53. package/commands/gsd/debug.md +52 -0
  54. package/commands/gsd/discuss-phase.md +76 -0
  55. package/commands/gsd/docs-update.md +49 -0
  56. package/commands/gsd/eval-review.md +33 -0
  57. package/commands/gsd/execute-phase.md +64 -0
  58. package/commands/gsd/explore.md +27 -0
  59. package/commands/gsd/extract-learnings.md +23 -0
  60. package/commands/gsd/fast.md +31 -0
  61. package/commands/gsd/forensics.md +57 -0
  62. package/commands/gsd/graphify.md +199 -0
  63. package/commands/gsd/health.md +31 -0
  64. package/commands/gsd/help.md +28 -0
  65. package/commands/gsd/import.md +41 -0
  66. package/commands/gsd/inbox.md +39 -0
  67. package/commands/gsd/ingest-docs.md +42 -0
  68. package/commands/gsd/manager.md +45 -0
  69. package/commands/gsd/map-codebase.md +83 -0
  70. package/commands/gsd/milestone-summary.md +51 -0
  71. package/commands/gsd/mvp-phase.md +45 -0
  72. package/commands/gsd/new-milestone.md +45 -0
  73. package/commands/gsd/new-project.md +47 -0
  74. package/commands/gsd/ns-context.md +23 -0
  75. package/commands/gsd/ns-ideate.md +24 -0
  76. package/commands/gsd/ns-manage.md +29 -0
  77. package/commands/gsd/ns-project.md +22 -0
  78. package/commands/gsd/ns-review.md +26 -0
  79. package/commands/gsd/ns-workflow.md +28 -0
  80. package/commands/gsd/pause-work.md +43 -0
  81. package/commands/gsd/phase.md +56 -0
  82. package/commands/gsd/plan-phase.md +62 -0
  83. package/commands/gsd/plan-review-convergence.md +59 -0
  84. package/commands/gsd/pr-branch.md +26 -0
  85. package/commands/gsd/profile-user.md +46 -0
  86. package/commands/gsd/progress.md +47 -0
  87. package/commands/gsd/quick.md +174 -0
  88. package/commands/gsd/resume-work.md +30 -0
  89. package/commands/gsd/review-backlog.md +63 -0
  90. package/commands/gsd/review.md +41 -0
  91. package/commands/gsd/secure-phase.md +36 -0
  92. package/commands/gsd/settings.md +29 -0
  93. package/commands/gsd/ship.md +24 -0
  94. package/commands/gsd/sketch.md +60 -0
  95. package/commands/gsd/spec-phase.md +63 -0
  96. package/commands/gsd/spike.md +57 -0
  97. package/commands/gsd/stats.md +19 -0
  98. package/commands/gsd/surface.md +155 -0
  99. package/commands/gsd/thread.md +24 -0
  100. package/commands/gsd/ui-phase.md +35 -0
  101. package/commands/gsd/ui-review.md +33 -0
  102. package/commands/gsd/ultraplan-phase.md +34 -0
  103. package/commands/gsd/undo.md +35 -0
  104. package/commands/gsd/update.md +48 -0
  105. package/commands/gsd/validate-phase.md +36 -0
  106. package/commands/gsd/verify-work.md +39 -0
  107. package/commands/gsd/workspace.md +52 -0
  108. package/commands/gsd/workstreams.md +70 -0
  109. package/get-shit-done/bin/check-latest-version.cjs +106 -0
  110. package/get-shit-done/bin/gsd-tools.cjs +1676 -0
  111. package/get-shit-done/bin/lib/active-workstream-store.cjs +302 -0
  112. package/get-shit-done/bin/lib/adr-parser.cjs +394 -0
  113. package/get-shit-done/bin/lib/agent-command-router.cjs +65 -0
  114. package/get-shit-done/bin/lib/artifacts.cjs +53 -0
  115. package/get-shit-done/bin/lib/audit.cjs +755 -0
  116. package/get-shit-done/bin/lib/check-command-router.cjs +333 -0
  117. package/get-shit-done/bin/lib/cjs-command-router-adapter.cjs +118 -0
  118. package/get-shit-done/bin/lib/clock.cjs +96 -0
  119. package/get-shit-done/bin/lib/clusters.cjs +135 -0
  120. package/get-shit-done/bin/lib/code-review-flags.cjs +74 -0
  121. package/get-shit-done/bin/lib/command-aliases.cjs +815 -0
  122. package/get-shit-done/bin/lib/command-arg-projection.cjs +62 -0
  123. package/get-shit-done/bin/lib/command-routing-hub.cjs +388 -0
  124. package/get-shit-done/bin/lib/commands.cjs +1188 -0
  125. package/get-shit-done/bin/lib/config-schema.cjs +31 -0
  126. package/get-shit-done/bin/lib/config.cjs +728 -0
  127. package/get-shit-done/bin/lib/configuration.cjs +248 -0
  128. package/get-shit-done/bin/lib/context-utilization.cjs +47 -0
  129. package/get-shit-done/bin/lib/core.cjs +2121 -0
  130. package/get-shit-done/bin/lib/decisions.cjs +116 -0
  131. package/get-shit-done/bin/lib/docs.cjs +270 -0
  132. package/get-shit-done/bin/lib/drift.cjs +388 -0
  133. package/get-shit-done/bin/lib/fallow-runner.cjs +109 -0
  134. package/get-shit-done/bin/lib/frontmatter.cjs +389 -0
  135. package/get-shit-done/bin/lib/gap-checker.cjs +205 -0
  136. package/get-shit-done/bin/lib/graphify.cjs +592 -0
  137. package/get-shit-done/bin/lib/gsd2-import.cjs +514 -0
  138. package/get-shit-done/bin/lib/init-command-router.cjs +58 -0
  139. package/get-shit-done/bin/lib/init.cjs +2112 -0
  140. package/get-shit-done/bin/lib/install-profiles.cjs +603 -0
  141. package/get-shit-done/bin/lib/installer-migration-authoring.cjs +117 -0
  142. package/get-shit-done/bin/lib/installer-migration-report.cjs +354 -0
  143. package/get-shit-done/bin/lib/installer-migrations/000-first-time-baseline.cjs +220 -0
  144. package/get-shit-done/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +41 -0
  145. package/get-shit-done/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +80 -0
  146. package/get-shit-done/bin/lib/installer-migrations.cjs +778 -0
  147. package/get-shit-done/bin/lib/intel.cjs +708 -0
  148. package/get-shit-done/bin/lib/learnings.cjs +421 -0
  149. package/get-shit-done/bin/lib/milestone.cjs +314 -0
  150. package/get-shit-done/bin/lib/model-catalog.cjs +212 -0
  151. package/get-shit-done/bin/lib/model-profiles.cjs +31 -0
  152. package/get-shit-done/bin/lib/observability/event.cjs +82 -0
  153. package/get-shit-done/bin/lib/observability/logger.cjs +174 -0
  154. package/get-shit-done/bin/lib/observability/redaction.cjs +50 -0
  155. package/get-shit-done/bin/lib/package-identity.cjs +31 -0
  156. package/get-shit-done/bin/lib/phase-command-router.cjs +191 -0
  157. package/get-shit-done/bin/lib/phase-lifecycle.cjs +80 -0
  158. package/get-shit-done/bin/lib/phase.cjs +1607 -0
  159. package/get-shit-done/bin/lib/phases-command-router.cjs +39 -0
  160. package/get-shit-done/bin/lib/plan-scan.cjs +97 -0
  161. package/get-shit-done/bin/lib/planning-workspace.cjs +238 -0
  162. package/get-shit-done/bin/lib/profile-output.cjs +1141 -0
  163. package/get-shit-done/bin/lib/profile-pipeline.cjs +539 -0
  164. package/get-shit-done/bin/lib/project-root.cjs +112 -0
  165. package/get-shit-done/bin/lib/prompt-budget.cjs +399 -0
  166. package/get-shit-done/bin/lib/review-reviewer-selection.cjs +125 -0
  167. package/get-shit-done/bin/lib/roadmap-command-router.cjs +28 -0
  168. package/get-shit-done/bin/lib/roadmap.cjs +650 -0
  169. package/get-shit-done/bin/lib/runtime-artifact-layout.cjs +301 -0
  170. package/get-shit-done/bin/lib/runtime-homes.cjs +222 -0
  171. package/get-shit-done/bin/lib/runtime-name-policy.cjs +83 -0
  172. package/get-shit-done/bin/lib/runtime-slash.cjs +112 -0
  173. package/get-shit-done/bin/lib/schema-detect.cjs +165 -0
  174. package/get-shit-done/bin/lib/secrets.cjs +32 -0
  175. package/get-shit-done/bin/lib/security.cjs +600 -0
  176. package/get-shit-done/bin/lib/semver-compare.cjs +35 -0
  177. package/get-shit-done/bin/lib/shell-command-projection.cjs +500 -0
  178. package/get-shit-done/bin/lib/state-command-router.cjs +252 -0
  179. package/get-shit-done/bin/lib/state-document.cjs +263 -0
  180. package/get-shit-done/bin/lib/state.cjs +2038 -0
  181. package/get-shit-done/bin/lib/surface.cjs +470 -0
  182. package/get-shit-done/bin/lib/task-command-router.cjs +81 -0
  183. package/get-shit-done/bin/lib/template.cjs +228 -0
  184. package/get-shit-done/bin/lib/uat.cjs +289 -0
  185. package/get-shit-done/bin/lib/update-context.cjs +209 -0
  186. package/get-shit-done/bin/lib/validate-command-router.cjs +83 -0
  187. package/get-shit-done/bin/lib/validate.cjs +92 -0
  188. package/get-shit-done/bin/lib/verify-command-router.cjs +40 -0
  189. package/get-shit-done/bin/lib/verify.cjs +1511 -0
  190. package/get-shit-done/bin/lib/workstream-inventory-builder.cjs +74 -0
  191. package/get-shit-done/bin/lib/workstream-inventory.cjs +146 -0
  192. package/get-shit-done/bin/lib/workstream-name-policy.cjs +94 -0
  193. package/get-shit-done/bin/lib/workstream.cjs +389 -0
  194. package/get-shit-done/bin/lib/worktree-safety.cjs +985 -0
  195. package/get-shit-done/bin/shared/config-defaults.manifest.json +97 -0
  196. package/get-shit-done/bin/shared/config-schema.manifest.json +175 -0
  197. package/get-shit-done/bin/shared/model-catalog.json +122 -0
  198. package/get-shit-done/bin/shared/runtime-aliases.manifest.json +75 -0
  199. package/get-shit-done/bin/verify-reapply-patches.cjs +352 -0
  200. package/get-shit-done/contexts/dev.md +21 -0
  201. package/get-shit-done/contexts/research.md +22 -0
  202. package/get-shit-done/contexts/review.md +23 -0
  203. package/get-shit-done/references/agent-contracts.md +79 -0
  204. package/get-shit-done/references/ai-evals.md +156 -0
  205. package/get-shit-done/references/ai-frameworks.md +186 -0
  206. package/get-shit-done/references/artifact-types.md +131 -0
  207. package/get-shit-done/references/autonomous-smart-discuss.md +277 -0
  208. package/get-shit-done/references/checkpoints.md +814 -0
  209. package/get-shit-done/references/common-bug-patterns.md +114 -0
  210. package/get-shit-done/references/context-budget.md +85 -0
  211. package/get-shit-done/references/continuation-format.md +253 -0
  212. package/get-shit-done/references/debugger-philosophy.md +76 -0
  213. package/get-shit-done/references/decimal-phase-calculation.md +64 -0
  214. package/get-shit-done/references/doc-conflict-engine.md +91 -0
  215. package/get-shit-done/references/domain-probes.md +125 -0
  216. package/get-shit-done/references/execute-mvp-tdd.md +81 -0
  217. package/get-shit-done/references/executor-examples.md +110 -0
  218. package/get-shit-done/references/few-shot-examples/plan-checker.md +73 -0
  219. package/get-shit-done/references/few-shot-examples/verifier.md +109 -0
  220. package/get-shit-done/references/gate-prompts.md +100 -0
  221. package/get-shit-done/references/gates.md +70 -0
  222. package/get-shit-done/references/git-integration.md +298 -0
  223. package/get-shit-done/references/git-planning-commit.md +40 -0
  224. package/get-shit-done/references/ios-scaffold.md +123 -0
  225. package/get-shit-done/references/mandatory-initial-read.md +2 -0
  226. package/get-shit-done/references/model-profile-resolution.md +38 -0
  227. package/get-shit-done/references/model-profiles.md +245 -0
  228. package/get-shit-done/references/mvp-concepts.md +49 -0
  229. package/get-shit-done/references/phase-argument-parsing.md +61 -0
  230. package/get-shit-done/references/planner-antipatterns.md +89 -0
  231. package/get-shit-done/references/planner-chunked.md +49 -0
  232. package/get-shit-done/references/planner-gap-closure.md +62 -0
  233. package/get-shit-done/references/planner-graphify-auto-update.md +67 -0
  234. package/get-shit-done/references/planner-human-verify-mode.md +57 -0
  235. package/get-shit-done/references/planner-interface-context.md +62 -0
  236. package/get-shit-done/references/planner-mvp-mode.md +53 -0
  237. package/get-shit-done/references/planner-reviews.md +39 -0
  238. package/get-shit-done/references/planner-revision.md +87 -0
  239. package/get-shit-done/references/planner-source-audit.md +73 -0
  240. package/get-shit-done/references/planning-config.md +471 -0
  241. package/get-shit-done/references/project-skills-discovery.md +19 -0
  242. package/get-shit-done/references/questioning.md +162 -0
  243. package/get-shit-done/references/revision-loop.md +97 -0
  244. package/get-shit-done/references/scout-codebase.md +51 -0
  245. package/get-shit-done/references/skeleton-template.md +48 -0
  246. package/get-shit-done/references/sketch-interactivity.md +41 -0
  247. package/get-shit-done/references/sketch-theme-system.md +94 -0
  248. package/get-shit-done/references/sketch-tooling.md +45 -0
  249. package/get-shit-done/references/sketch-variant-patterns.md +81 -0
  250. package/get-shit-done/references/spidr-splitting.md +69 -0
  251. package/get-shit-done/references/tdd.md +330 -0
  252. package/get-shit-done/references/thinking-models-debug.md +44 -0
  253. package/get-shit-done/references/thinking-models-execution.md +50 -0
  254. package/get-shit-done/references/thinking-models-planning.md +62 -0
  255. package/get-shit-done/references/thinking-models-research.md +50 -0
  256. package/get-shit-done/references/thinking-models-verification.md +55 -0
  257. package/get-shit-done/references/thinking-partner.md +96 -0
  258. package/get-shit-done/references/ui-brand.md +160 -0
  259. package/get-shit-done/references/universal-anti-patterns.md +63 -0
  260. package/get-shit-done/references/user-profiling.md +681 -0
  261. package/get-shit-done/references/user-story-template.md +58 -0
  262. package/get-shit-done/references/verification-overrides.md +227 -0
  263. package/get-shit-done/references/verification-patterns.md +612 -0
  264. package/get-shit-done/references/verify-mvp-mode.md +85 -0
  265. package/get-shit-done/references/workstream-flag.md +111 -0
  266. package/get-shit-done/references/worktree-path-safety.md +89 -0
  267. package/get-shit-done/templates/AI-SPEC.md +246 -0
  268. package/get-shit-done/templates/DEBUG.md +169 -0
  269. package/get-shit-done/templates/README.md +77 -0
  270. package/get-shit-done/templates/SECURITY.md +61 -0
  271. package/get-shit-done/templates/UAT.md +265 -0
  272. package/get-shit-done/templates/UI-SPEC.md +100 -0
  273. package/get-shit-done/templates/VALIDATION.md +76 -0
  274. package/get-shit-done/templates/claude-md.md +145 -0
  275. package/get-shit-done/templates/codebase/architecture.md +255 -0
  276. package/get-shit-done/templates/codebase/concerns.md +310 -0
  277. package/get-shit-done/templates/codebase/conventions.md +307 -0
  278. package/get-shit-done/templates/codebase/integrations.md +280 -0
  279. package/get-shit-done/templates/codebase/stack.md +186 -0
  280. package/get-shit-done/templates/codebase/structure.md +285 -0
  281. package/get-shit-done/templates/codebase/testing.md +480 -0
  282. package/get-shit-done/templates/config.json +62 -0
  283. package/get-shit-done/templates/context.md +352 -0
  284. package/get-shit-done/templates/continue-here.md +78 -0
  285. package/get-shit-done/templates/copilot-instructions.md +7 -0
  286. package/get-shit-done/templates/debug-subagent-prompt.md +91 -0
  287. package/get-shit-done/templates/dev-preferences.md +21 -0
  288. package/get-shit-done/templates/discovery.md +146 -0
  289. package/get-shit-done/templates/discussion-log.md +63 -0
  290. package/get-shit-done/templates/milestone-archive.md +123 -0
  291. package/get-shit-done/templates/milestone.md +115 -0
  292. package/get-shit-done/templates/phase-prompt.md +610 -0
  293. package/get-shit-done/templates/planner-subagent-prompt.md +117 -0
  294. package/get-shit-done/templates/project.md +186 -0
  295. package/get-shit-done/templates/requirements.md +231 -0
  296. package/get-shit-done/templates/research-project/ARCHITECTURE.md +204 -0
  297. package/get-shit-done/templates/research-project/FEATURES.md +147 -0
  298. package/get-shit-done/templates/research-project/PITFALLS.md +200 -0
  299. package/get-shit-done/templates/research-project/STACK.md +120 -0
  300. package/get-shit-done/templates/research-project/SUMMARY.md +170 -0
  301. package/get-shit-done/templates/research.md +592 -0
  302. package/get-shit-done/templates/retrospective.md +54 -0
  303. package/get-shit-done/templates/roadmap.md +202 -0
  304. package/get-shit-done/templates/spec.md +307 -0
  305. package/get-shit-done/templates/state.md +195 -0
  306. package/get-shit-done/templates/summary-complex.md +59 -0
  307. package/get-shit-done/templates/summary-minimal.md +41 -0
  308. package/get-shit-done/templates/summary-standard.md +48 -0
  309. package/get-shit-done/templates/summary.md +248 -0
  310. package/get-shit-done/templates/user-profile.md +146 -0
  311. package/get-shit-done/templates/user-setup.md +311 -0
  312. package/get-shit-done/templates/verification-report.md +322 -0
  313. package/get-shit-done/workflows/_runtime-launcher.snippet.sh +1 -0
  314. package/get-shit-done/workflows/add-backlog.md +91 -0
  315. package/get-shit-done/workflows/add-phase.md +113 -0
  316. package/get-shit-done/workflows/add-tests.md +355 -0
  317. package/get-shit-done/workflows/add-todo.md +161 -0
  318. package/get-shit-done/workflows/ai-integration-phase.md +295 -0
  319. package/get-shit-done/workflows/analyze-dependencies.md +96 -0
  320. package/get-shit-done/workflows/audit-fix.md +178 -0
  321. package/get-shit-done/workflows/audit-milestone.md +358 -0
  322. package/get-shit-done/workflows/audit-uat.md +110 -0
  323. package/get-shit-done/workflows/autonomous.md +795 -0
  324. package/get-shit-done/workflows/check-todos.md +180 -0
  325. package/get-shit-done/workflows/cleanup.md +155 -0
  326. package/get-shit-done/workflows/code-review-fix.md +502 -0
  327. package/get-shit-done/workflows/code-review.md +656 -0
  328. package/get-shit-done/workflows/complete-milestone.md +855 -0
  329. package/get-shit-done/workflows/debug.md +232 -0
  330. package/get-shit-done/workflows/diagnose-issues.md +241 -0
  331. package/get-shit-done/workflows/discovery-phase.md +291 -0
  332. package/get-shit-done/workflows/discuss-phase/modes/advisor.md +176 -0
  333. package/get-shit-done/workflows/discuss-phase/modes/all.md +28 -0
  334. package/get-shit-done/workflows/discuss-phase/modes/analyze.md +44 -0
  335. package/get-shit-done/workflows/discuss-phase/modes/auto.md +57 -0
  336. package/get-shit-done/workflows/discuss-phase/modes/batch.md +52 -0
  337. package/get-shit-done/workflows/discuss-phase/modes/chain.md +98 -0
  338. package/get-shit-done/workflows/discuss-phase/modes/default.md +141 -0
  339. package/get-shit-done/workflows/discuss-phase/modes/power.md +44 -0
  340. package/get-shit-done/workflows/discuss-phase/modes/text.md +55 -0
  341. package/get-shit-done/workflows/discuss-phase/templates/checkpoint.json +18 -0
  342. package/get-shit-done/workflows/discuss-phase/templates/context.md +136 -0
  343. package/get-shit-done/workflows/discuss-phase/templates/discussion-log.md +50 -0
  344. package/get-shit-done/workflows/discuss-phase-assumptions.md +675 -0
  345. package/get-shit-done/workflows/discuss-phase-power.md +291 -0
  346. package/get-shit-done/workflows/discuss-phase.md +499 -0
  347. package/get-shit-done/workflows/do.md +111 -0
  348. package/get-shit-done/workflows/docs-update.md +1162 -0
  349. package/get-shit-done/workflows/edit-phase.md +295 -0
  350. package/get-shit-done/workflows/eval-review.md +156 -0
  351. package/get-shit-done/workflows/execute-phase/steps/codebase-drift-gate.md +82 -0
  352. package/get-shit-done/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
  353. package/get-shit-done/workflows/execute-phase/steps/post-merge-gate.md +117 -0
  354. package/get-shit-done/workflows/execute-phase.md +1709 -0
  355. package/get-shit-done/workflows/execute-plan.md +526 -0
  356. package/get-shit-done/workflows/explore.md +144 -0
  357. package/get-shit-done/workflows/extract-learnings.md +243 -0
  358. package/get-shit-done/workflows/fast.md +124 -0
  359. package/get-shit-done/workflows/forensics.md +279 -0
  360. package/get-shit-done/workflows/graduation.md +196 -0
  361. package/get-shit-done/workflows/health.md +224 -0
  362. package/get-shit-done/workflows/help/modes/brief.md +22 -0
  363. package/get-shit-done/workflows/help/modes/default.md +50 -0
  364. package/get-shit-done/workflows/help/modes/full.md +784 -0
  365. package/get-shit-done/workflows/help/modes/topic.md +74 -0
  366. package/get-shit-done/workflows/help.md +24 -0
  367. package/get-shit-done/workflows/import.md +254 -0
  368. package/get-shit-done/workflows/inbox.md +387 -0
  369. package/get-shit-done/workflows/ingest-docs.md +339 -0
  370. package/get-shit-done/workflows/insert-phase.md +152 -0
  371. package/get-shit-done/workflows/list-phase-assumptions.md +178 -0
  372. package/get-shit-done/workflows/list-workspaces.md +57 -0
  373. package/get-shit-done/workflows/manager.md +393 -0
  374. package/get-shit-done/workflows/map-codebase.md +444 -0
  375. package/get-shit-done/workflows/milestone-summary.md +224 -0
  376. package/get-shit-done/workflows/mvp-phase.md +222 -0
  377. package/get-shit-done/workflows/new-milestone.md +635 -0
  378. package/get-shit-done/workflows/new-project.md +1555 -0
  379. package/get-shit-done/workflows/new-workspace.md +240 -0
  380. package/get-shit-done/workflows/next.md +299 -0
  381. package/get-shit-done/workflows/node-repair.md +92 -0
  382. package/get-shit-done/workflows/note.md +158 -0
  383. package/get-shit-done/workflows/pause-work.md +244 -0
  384. package/get-shit-done/workflows/plan-milestone-gaps.md +281 -0
  385. package/get-shit-done/workflows/plan-phase.md +1809 -0
  386. package/get-shit-done/workflows/plan-review-convergence.md +346 -0
  387. package/get-shit-done/workflows/plant-seed.md +230 -0
  388. package/get-shit-done/workflows/pr-branch.md +157 -0
  389. package/get-shit-done/workflows/profile-user.md +453 -0
  390. package/get-shit-done/workflows/progress.md +699 -0
  391. package/get-shit-done/workflows/quick.md +1039 -0
  392. package/get-shit-done/workflows/reapply-patches.md +426 -0
  393. package/get-shit-done/workflows/remove-phase.md +156 -0
  394. package/get-shit-done/workflows/remove-workspace.md +108 -0
  395. package/get-shit-done/workflows/resume-project.md +332 -0
  396. package/get-shit-done/workflows/review.md +623 -0
  397. package/get-shit-done/workflows/scan.md +105 -0
  398. package/get-shit-done/workflows/secure-phase.md +180 -0
  399. package/get-shit-done/workflows/session-report.md +146 -0
  400. package/get-shit-done/workflows/settings-advanced.md +620 -0
  401. package/get-shit-done/workflows/settings-integrations.md +312 -0
  402. package/get-shit-done/workflows/settings.md +552 -0
  403. package/get-shit-done/workflows/ship.md +356 -0
  404. package/get-shit-done/workflows/sketch-wrap-up.md +286 -0
  405. package/get-shit-done/workflows/sketch.md +361 -0
  406. package/get-shit-done/workflows/spec-phase.md +262 -0
  407. package/get-shit-done/workflows/spike-wrap-up.md +307 -0
  408. package/get-shit-done/workflows/spike.md +453 -0
  409. package/get-shit-done/workflows/stats.md +80 -0
  410. package/get-shit-done/workflows/sync-skills.md +182 -0
  411. package/get-shit-done/workflows/thread.md +222 -0
  412. package/get-shit-done/workflows/transition.md +694 -0
  413. package/get-shit-done/workflows/ui-phase.md +328 -0
  414. package/get-shit-done/workflows/ui-review.md +193 -0
  415. package/get-shit-done/workflows/ultraplan-phase.md +199 -0
  416. package/get-shit-done/workflows/undo.md +314 -0
  417. package/get-shit-done/workflows/update.md +443 -0
  418. package/get-shit-done/workflows/validate-phase.md +179 -0
  419. package/get-shit-done/workflows/verify-phase.md +544 -0
  420. package/get-shit-done/workflows/verify-work.md +781 -0
  421. package/hooks/dist/gsd-check-update-worker.js +95 -0
  422. package/hooks/dist/gsd-check-update.js +64 -0
  423. package/hooks/dist/gsd-context-monitor.js +195 -0
  424. package/hooks/dist/gsd-graphify-update.sh +158 -0
  425. package/hooks/dist/gsd-phase-boundary.sh +47 -0
  426. package/hooks/dist/gsd-prompt-guard.js +97 -0
  427. package/hooks/dist/gsd-read-guard.js +101 -0
  428. package/hooks/dist/gsd-read-injection-scanner.js +203 -0
  429. package/hooks/dist/gsd-session-state.sh +59 -0
  430. package/hooks/dist/gsd-statusline.js +548 -0
  431. package/hooks/dist/gsd-update-banner.js +134 -0
  432. package/hooks/dist/gsd-validate-commit.sh +57 -0
  433. package/hooks/dist/gsd-workflow-guard.js +166 -0
  434. package/hooks/dist/lib/git-cmd.js +150 -0
  435. package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
  436. package/hooks/gsd-check-update-worker.js +95 -0
  437. package/hooks/gsd-check-update.js +64 -0
  438. package/hooks/gsd-context-monitor.js +195 -0
  439. package/hooks/gsd-graphify-update.sh +158 -0
  440. package/hooks/gsd-phase-boundary.sh +47 -0
  441. package/hooks/gsd-prompt-guard.js +97 -0
  442. package/hooks/gsd-read-guard.js +101 -0
  443. package/hooks/gsd-read-injection-scanner.js +203 -0
  444. package/hooks/gsd-session-state.sh +59 -0
  445. package/hooks/gsd-statusline.js +548 -0
  446. package/hooks/gsd-update-banner.js +134 -0
  447. package/hooks/gsd-validate-commit.sh +57 -0
  448. package/hooks/gsd-workflow-guard.js +166 -0
  449. package/hooks/lib/git-cmd.js +150 -0
  450. package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
  451. package/hooks/managed-hooks-registry.cjs +34 -0
  452. package/package.json +102 -0
  453. package/scripts/affected-tests-lib.cjs +541 -0
  454. package/scripts/audit-workflow-script-paths.cjs +73 -0
  455. package/scripts/base64-scan.sh +339 -0
  456. package/scripts/build-hooks.js +236 -0
  457. package/scripts/changeset/README.md +129 -0
  458. package/scripts/changeset/cli.cjs +392 -0
  459. package/scripts/changeset/github-release-notes.cjs +199 -0
  460. package/scripts/changeset/lint.cjs +110 -0
  461. package/scripts/changeset/new.cjs +137 -0
  462. package/scripts/changeset/parse.cjs +114 -0
  463. package/scripts/changeset/render.cjs +34 -0
  464. package/scripts/changeset/serialize.cjs +130 -0
  465. package/scripts/check-alias-drift.cjs +108 -0
  466. package/scripts/check-env.cjs +302 -0
  467. package/scripts/check-npm-integrity.cjs +209 -0
  468. package/scripts/ci-guard-runner.cjs +16 -0
  469. package/scripts/ci-prepare-test-scope.cjs +46 -0
  470. package/scripts/ci-rebase-check.cjs +85 -0
  471. package/scripts/ci-test-scope.cjs +302 -0
  472. package/scripts/command-contract-helpers.cjs +64 -0
  473. package/scripts/diff-touches-shipped-paths.cjs +147 -0
  474. package/scripts/fix-slash-commands.cjs +147 -0
  475. package/scripts/gen-inventory-manifest.cjs +109 -0
  476. package/scripts/generate-package-identity.cjs +104 -0
  477. package/scripts/lint-command-contract.cjs +108 -0
  478. package/scripts/lint-descriptions.cjs +83 -0
  479. package/scripts/lint-docs-required.cjs +222 -0
  480. package/scripts/lint-no-source-grep-extras.cjs +81 -0
  481. package/scripts/lint-no-source-grep.cjs +174 -0
  482. package/scripts/lint-package-identity-drift.cjs +141 -0
  483. package/scripts/lint-pr-check-project-dir.cjs +98 -0
  484. package/scripts/lint-shared-module-handsync.cjs +388 -0
  485. package/scripts/lint-shell-command-projection-drift.cjs +57 -0
  486. package/scripts/lint-skill-deps.cjs +180 -0
  487. package/scripts/lint-test-file-count.allowlist.json +36 -0
  488. package/scripts/lint-test-file-count.cjs +190 -0
  489. package/scripts/pr-template-policy.cjs +268 -0
  490. package/scripts/prompt-injection-scan.sh +203 -0
  491. package/scripts/release-tarball-smoke.cjs +627 -0
  492. package/scripts/run-affected-tests.cjs +6 -0
  493. package/scripts/run-cross-platform-tests.cjs +63 -0
  494. package/scripts/run-tests.cjs +282 -0
  495. package/scripts/secret-scan-lint.sh +231 -0
  496. package/scripts/secret-scan.sh +358 -0
  497. package/scripts/setup-branch-protection.sh +236 -0
  498. package/scripts/shared-module-handsync-allowlist.json +183 -0
  499. package/scripts/strip-prose-atrefs.cjs +106 -0
  500. package/scripts/sync-rulesets.sh +34 -0
  501. package/scripts/sync-runtime-launcher.cjs +402 -0
  502. package/scripts/test-failure-reasons.cjs +34 -0
  503. package/scripts/workflow-policy.cjs +450 -0
@@ -0,0 +1,603 @@
1
+ /**
2
+ * Skill Surface Budget Module — single source of truth for which skills/agents
3
+ * are written to the runtime config dirs (ADR-0011).
4
+ *
5
+ * Background: every installed `gsd-*` skill costs eager system-prompt tokens
6
+ * because runtimes (Claude Code, opencode, etc.) enumerate skill descriptions
7
+ * in `<available_skills>` on every turn. With 66 skills + 33 agents GSD alone
8
+ * consumes ~60% of the default 1%-of-context skill-listing budget, causing
9
+ * dropped skills when users stack multiple plugins (#3408).
10
+ *
11
+ * Profile model: three named profiles replace the old minimal/full binary:
12
+ * - core — eight skills covering the main project loop (includes surface for ADR-0011 expand contract)
13
+ * - standard — core + phase management and workspace skills
14
+ * - full — all skills (previous default, '*' sentinel)
15
+ * Profiles compose: --profile=core,audit resolves to union(closure(core), closure(audit)).
16
+ * Back-compat aliases: --minimal / --core-only both map to --profile=core.
17
+ *
18
+ * This module owns:
19
+ * - PROFILES map: named profile → base skill set (or '*' sentinel for full)
20
+ * - loadSkillsManifest: parse requires: frontmatter from commands/gsd/*.md
21
+ * - resolveProfile: compute transitive closure of profile base over manifest
22
+ * - stageSkillsForProfile / stageAgentsForProfile: filesystem staging
23
+ * - readActiveProfile / writeActiveProfile: .gsd-profile marker persistence
24
+ * - resolveEffectiveProfile: explicit flag > .gsd-profile marker > full
25
+ * - mostRestrictiveProfile: resolve multi-runtime disagreement to smallest set
26
+ *
27
+ * Companion module (Phase 2): get-shit-done/bin/lib/surface.cjs owns the
28
+ * runtime /gsd:surface command. It reuses stageSkillsForProfile and
29
+ * stageAgentsForProfile from this module for cluster-level enable/disable
30
+ * without reinstall, persisting state in <runtimeConfigDir>/.gsd-surface.json.
31
+ *
32
+ * Legacy back-compat exports (deprecated, kept for existing callers):
33
+ * - MINIMAL_SKILL_ALLOWLIST — derived from PROFILES.core
34
+ * - isMinimalMode(mode) — returns true for 'minimal'
35
+ * - shouldInstallSkill(name, mode|resolvedProfile) — overloaded
36
+ * - stageSkillsForMode(srcDir, mode) — wraps stageSkillsForProfile
37
+ */
38
+
39
+ 'use strict';
40
+
41
+ const fs = require('fs');
42
+ const path = require('path');
43
+ const os = require('os');
44
+ const { platformWriteSync } = require('./shell-command-projection.cjs');
45
+
46
+ // ---------------------------------------------------------------------------
47
+ // Profile definitions
48
+ // ---------------------------------------------------------------------------
49
+
50
+ /**
51
+ * PROFILES maps profile name → base skill set (array) or '*' sentinel (full).
52
+ *
53
+ * The effective set for any profile is CLOSURE(base, requires: manifest).
54
+ * standard is a superset of core; full is the identity (all skills).
55
+ *
56
+ * Composition: --profile=core,audit resolves to union(closure(core), closure(audit)).
57
+ */
58
+ const PROFILES = Object.freeze({
59
+ core: Object.freeze([
60
+ 'new-project',
61
+ 'discuss-phase',
62
+ 'plan-phase',
63
+ 'execute-phase',
64
+ 'phase',
65
+ 'help',
66
+ 'update',
67
+ 'surface',
68
+ ]),
69
+ standard: Object.freeze([
70
+ // Core loop
71
+ 'new-project',
72
+ 'discuss-phase',
73
+ 'plan-phase',
74
+ 'execute-phase',
75
+ 'help',
76
+ 'update',
77
+ 'surface',
78
+ // Phase management (hot nodes from audit — required by 38+ skills)
79
+ 'phase',
80
+ 'review',
81
+ 'config',
82
+ 'progress',
83
+ // Workspace / state
84
+ 'resume-work',
85
+ 'pause-work',
86
+ 'workspace',
87
+ ]),
88
+ full: '*',
89
+ });
90
+
91
+ // ---------------------------------------------------------------------------
92
+ // Manifest parsing
93
+ // ---------------------------------------------------------------------------
94
+
95
+ /**
96
+ * Parse the requires: field from YAML frontmatter.
97
+ * Handles: "requires: [a, b, c]" (flow style) and absent field.
98
+ * Returns string[] — empty array if no requires: field.
99
+ *
100
+ * No external YAML parser dependency — hand-parse the single line
101
+ * since GSD enforces flow-style arrays for requires:.
102
+ *
103
+ * @param {string} content full file content
104
+ * @returns {string[]}
105
+ */
106
+ function parseRequires(content) {
107
+ const fmMatch = content.match(/^---\r?\n([\s\S]*?)\r?\n---/m);
108
+ if (!fmMatch) return [];
109
+ const fm = fmMatch[1];
110
+ const line = fm.match(/^requires:\s*(.+)$/m);
111
+ if (!line) return [];
112
+ const val = line[1].trim();
113
+ // Flow-style: [a, b, c]
114
+ if (val.startsWith('[') && val.endsWith(']')) {
115
+ const inner = val.slice(1, -1).trim();
116
+ if (!inner) return [];
117
+ return inner.split(',').map((s) => s.trim()).filter(Boolean);
118
+ }
119
+ // Single bare value (not currently used, but defensive)
120
+ return val ? [val] : [];
121
+ }
122
+
123
+ /**
124
+ * Parse agent references from a skill file's body text.
125
+ * Scans the full content for `gsd-<stem>` patterns that correspond to
126
+ * real agent files. Returns all unique `gsd-*` stems found in the body.
127
+ *
128
+ * The caller is responsible for filtering by which agents actually exist —
129
+ * this function returns all syntactically valid `gsd-*` matches.
130
+ *
131
+ * @param {string} content full file content
132
+ * @returns {string[]} deduplicated agent stems like ['gsd-planner', 'gsd-executor']
133
+ */
134
+ function parseCallsAgents(content) {
135
+ // Match word-boundary gsd-<stem> patterns; stems are lowercase letters and hyphens.
136
+ // We use a regex that matches `gsd-` followed by one or more lowercase-alpha-or-hyphen chars.
137
+ // This catches `gsd-planner`, `gsd-plan-checker`, etc. in prose and code.
138
+ const matches = content.match(/\bgsd-[a-z][a-z-]*/g);
139
+ if (!matches) return [];
140
+ // Deduplicate
141
+ return [...new Set(matches)];
142
+ }
143
+
144
+ /**
145
+ * Load the requires: dependency graph from a commands/gsd directory.
146
+ * Also derives calls_agents for each skill by scanning the body text for
147
+ * `gsd-*` agent name references. Agent stems are stored under the special
148
+ * key `_calls_agents_<stem>` so they don't conflict with skill stems.
149
+ *
150
+ * @param {string} commandsDir absolute path to commands/gsd/
151
+ * @returns {Map<string, string[]>} stem → [required stem, ...] plus _calls_agents_<stem> entries
152
+ */
153
+ function loadSkillsManifest(commandsDir) {
154
+ const manifest = new Map();
155
+ if (!fs.existsSync(commandsDir)) return manifest;
156
+ const entries = fs.readdirSync(commandsDir, { withFileTypes: true });
157
+ for (const entry of entries) {
158
+ if (!entry.isFile()) continue;
159
+ if (!entry.name.endsWith('.md')) continue;
160
+ const stem = entry.name.slice(0, -3);
161
+ try {
162
+ const content = fs.readFileSync(path.join(commandsDir, entry.name), 'utf8');
163
+ manifest.set(stem, parseRequires(content));
164
+ // Derive agent references from body text
165
+ const agentRefs = parseCallsAgents(content);
166
+ manifest.set(`_calls_agents_${stem}`, agentRefs);
167
+ } catch {
168
+ manifest.set(stem, []);
169
+ manifest.set(`_calls_agents_${stem}`, []);
170
+ }
171
+ }
172
+ return manifest;
173
+ }
174
+
175
+ // ---------------------------------------------------------------------------
176
+ // Profile resolution (transitive closure)
177
+ // ---------------------------------------------------------------------------
178
+
179
+ /**
180
+ * Compute the transitive closure of a set of skill stems over the manifest.
181
+ *
182
+ * @param {Iterable<string>} base initial set of stems
183
+ * @param {Map<string, string[]>} manifest skill → [required stems]
184
+ * @returns {Set<string>}
185
+ */
186
+ function computeClosure(base, manifest) {
187
+ const closed = new Set(base);
188
+ const queue = [...closed];
189
+ while (queue.length > 0) {
190
+ const stem = queue.pop();
191
+ const deps = manifest.get(stem) || [];
192
+ for (const dep of deps) {
193
+ if (!closed.has(dep)) {
194
+ closed.add(dep);
195
+ queue.push(dep);
196
+ }
197
+ }
198
+ }
199
+ return closed;
200
+ }
201
+
202
+ /**
203
+ * Resolve a profile (or composed profiles) to a typed result object.
204
+ *
205
+ * @param {object} opts
206
+ * @param {string[]} [opts.modes=['full']] profile names to resolve and union
207
+ * @param {Map<string, string[]>} [opts.manifest] parsed requires: graph
208
+ * @param {object} [opts._profilesOverride] for testing — override PROFILES
209
+ * @returns {{ name: string, skills: Set<string>|'*', agents: Set<string> }}
210
+ */
211
+ function resolveProfile({ modes, manifest, _profilesOverride } = {}) {
212
+ const profiles = _profilesOverride || PROFILES;
213
+ const activeModes = (modes && modes.length > 0) ? modes : ['full'];
214
+ const normalizedModes = activeModes
215
+ .flatMap((mode) => String(mode).split(','))
216
+ .map((mode) => mode.trim())
217
+ .filter(Boolean);
218
+ const modesToResolve = normalizedModes.length > 0 ? normalizedModes : ['full'];
219
+
220
+ // If any mode is 'full', the result is the full sentinel
221
+ if (modesToResolve.includes('full')) {
222
+ return { name: 'full', skills: '*', agents: new Set() };
223
+ }
224
+
225
+ const validModes = modesToResolve.filter((mode) => Object.prototype.hasOwnProperty.call(profiles, mode));
226
+ if (validModes.length === 0) {
227
+ // Invalid/corrupt marker fallback: avoid empty installs by defaulting to full.
228
+ return { name: 'full', skills: '*', agents: new Set() };
229
+ }
230
+
231
+ const man = manifest || new Map();
232
+ const unionSkills = new Set();
233
+
234
+ for (const mode of validModes) {
235
+ const base = profiles[mode];
236
+ if (base === '*') {
237
+ // This profile is full — sentinel short-circuit
238
+ return { name: 'full', skills: '*', agents: new Set() };
239
+ }
240
+ const closure = computeClosure(base, man);
241
+ for (const s of closure) unionSkills.add(s);
242
+ }
243
+
244
+ // Derive agents: union of all agent names referenced in the body text of
245
+ // every skill in unionSkills. Agent names are stored in the manifest under
246
+ // _calls_agents_<stem> keys (populated by loadSkillsManifest).
247
+ const unionAgents = new Set();
248
+ for (const skillStem of unionSkills) {
249
+ const agentRefs = man.get(`_calls_agents_${skillStem}`) || [];
250
+ for (const agentStem of agentRefs) {
251
+ unionAgents.add(agentStem);
252
+ }
253
+ }
254
+
255
+ const name = validModes.length === 1 ? validModes[0] : validModes.join(',');
256
+ return { name, skills: unionSkills, agents: unionAgents };
257
+ }
258
+
259
+ // ---------------------------------------------------------------------------
260
+ // Staging — skills
261
+ // ---------------------------------------------------------------------------
262
+
263
+ // Stage dirs created during this process — cleaned up on exit.
264
+ // 13 runtime dispatch sites in install.js can each call stageSkillsForMode,
265
+ // so accumulating them in a single set avoids leaks without forcing each
266
+ // site to track its own cleanup handle.
267
+ const STAGED_DIRS = new Set();
268
+ let exitHandlerRegistered = false;
269
+
270
+ function cleanupStagedSkills() {
271
+ for (const dir of STAGED_DIRS) {
272
+ try {
273
+ fs.rmSync(dir, { recursive: true, force: true });
274
+ } catch {
275
+ // Best-effort: missing dir or permission error shouldn't crash a
276
+ // successful install. The OS reaps tmpdir eventually.
277
+ }
278
+ }
279
+ STAGED_DIRS.clear();
280
+ }
281
+
282
+ // Signals we register a cleanup handler for in addition to the natural
283
+ // 'exit' event. `process.on('exit')` does NOT fire on these — an installer
284
+ // is exactly the kind of process users abort mid-run, so without explicit
285
+ // signal handling Ctrl+C would leave staged tmp dirs behind.
286
+ const CLEANUP_SIGNALS = ['SIGINT', 'SIGTERM', 'SIGHUP'];
287
+
288
+ function ensureExitCleanup() {
289
+ if (exitHandlerRegistered) return;
290
+ exitHandlerRegistered = true;
291
+ process.on('exit', cleanupStagedSkills);
292
+ for (const sig of CLEANUP_SIGNALS) {
293
+ // `once` so re-raising the signal below isn't intercepted by us a second
294
+ // time — the OS-default handler should take over and exit with the right
295
+ // status code (so CI sees the abort, scripts see 130 for SIGINT, etc.).
296
+ process.once(sig, () => {
297
+ cleanupStagedSkills();
298
+ process.kill(process.pid, sig);
299
+ });
300
+ }
301
+ }
302
+
303
+ /**
304
+ * Stage a filtered copy of commands/gsd for a resolved profile.
305
+ * In full mode (skills === '*') returns srcDir unchanged (no-op).
306
+ *
307
+ * @param {string} srcDir absolute path to commands/gsd
308
+ * @param {{ skills: Set<string>|'*' }} resolvedProfile
309
+ * @returns {string} path to staged dir (or srcDir for full)
310
+ */
311
+ function stageSkillsForProfile(srcDir, resolvedProfile) {
312
+ if (resolvedProfile.skills === '*') return srcDir;
313
+ if (!fs.existsSync(srcDir)) return srcDir;
314
+
315
+ const stageDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gsd-profile-skills-'));
316
+ try {
317
+ const entries = fs.readdirSync(srcDir, { withFileTypes: true });
318
+ for (const entry of entries) {
319
+ if (!entry.isFile()) continue;
320
+ if (!entry.name.endsWith('.md')) continue;
321
+ const stem = entry.name.slice(0, -3);
322
+ if (!resolvedProfile.skills.has(stem)) continue;
323
+ fs.copyFileSync(
324
+ path.join(srcDir, entry.name),
325
+ path.join(stageDir, entry.name),
326
+ );
327
+ }
328
+ } catch (err) {
329
+ try { fs.rmSync(stageDir, { recursive: true, force: true }); } catch {}
330
+ throw err;
331
+ }
332
+ STAGED_DIRS.add(stageDir);
333
+ ensureExitCleanup();
334
+ return stageDir;
335
+ }
336
+
337
+ /**
338
+ * Stage a filtered copy of the agents directory for a resolved profile.
339
+ * For 'full', returns srcAgentsDir unchanged.
340
+ * For tiered profiles, copies only agents whose full stem (e.g. 'gsd-planner')
341
+ * is in resolvedProfile.agents — which is populated by resolveProfile() from
342
+ * the _calls_agents_* entries in the manifest.
343
+ *
344
+ * @param {string} srcAgentsDir absolute path to agents/
345
+ * @param {{ agents: Set<string>, skills: Set<string>|'*' }} resolvedProfile
346
+ * @returns {string} path to staged dir (or srcAgentsDir for full)
347
+ */
348
+ function stageAgentsForProfile(srcAgentsDir, resolvedProfile) {
349
+ if (resolvedProfile.skills === '*') return srcAgentsDir;
350
+ if (!fs.existsSync(srcAgentsDir)) return srcAgentsDir;
351
+
352
+ const stageDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gsd-profile-agents-'));
353
+ try {
354
+ if (resolvedProfile.agents instanceof Set && resolvedProfile.agents.size > 0) {
355
+ const entries = fs.readdirSync(srcAgentsDir, { withFileTypes: true });
356
+ for (const entry of entries) {
357
+ if (!entry.isFile()) continue;
358
+ if (!entry.name.endsWith('.md')) continue;
359
+ // Agent stem is the full filename without extension, e.g. "gsd-planner"
360
+ const stem = entry.name.slice(0, -3);
361
+ if (!resolvedProfile.agents.has(stem)) continue;
362
+ fs.copyFileSync(
363
+ path.join(srcAgentsDir, entry.name),
364
+ path.join(stageDir, entry.name),
365
+ );
366
+ }
367
+ }
368
+ // If agents is empty Set, we produce an empty stageDir (no agents for this profile)
369
+ } catch (err) {
370
+ try { fs.rmSync(stageDir, { recursive: true, force: true }); } catch {}
371
+ throw err;
372
+ }
373
+ STAGED_DIRS.add(stageDir);
374
+ ensureExitCleanup();
375
+ return stageDir;
376
+ }
377
+
378
+ function stageSkillsForRuntimeAsSkills(srcCommandsDir, resolvedProfile, converter, prefix) {
379
+ if (!fs.existsSync(srcCommandsDir)) return srcCommandsDir;
380
+
381
+ const stageDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gsd-profile-runtime-skills-'));
382
+ try {
383
+ const entries = fs.readdirSync(srcCommandsDir, { withFileTypes: true });
384
+ for (const entry of entries) {
385
+ if (!entry.isFile()) continue;
386
+ if (!entry.name.endsWith('.md')) continue;
387
+ const stem = entry.name.slice(0, -3);
388
+ if (resolvedProfile.skills !== '*' && !resolvedProfile.skills.has(stem)) continue;
389
+ const content = fs.readFileSync(path.join(srcCommandsDir, entry.name), 'utf8');
390
+ const skillName = `${prefix}${stem}`;
391
+ const converted = converter(content, skillName);
392
+ const destDir = path.join(stageDir, skillName);
393
+ fs.mkdirSync(destDir, { recursive: true });
394
+ fs.writeFileSync(path.join(destDir, 'SKILL.md'), converted);
395
+ }
396
+ } catch (err) {
397
+ try { fs.rmSync(stageDir, { recursive: true, force: true }); } catch {}
398
+ throw err;
399
+ }
400
+ STAGED_DIRS.add(stageDir);
401
+ ensureExitCleanup();
402
+ return stageDir;
403
+ }
404
+
405
+ // ---------------------------------------------------------------------------
406
+ // Profile marker persistence
407
+ // ---------------------------------------------------------------------------
408
+
409
+ const PROFILE_MARKER_NAME = '.gsd-profile';
410
+
411
+ /**
412
+ * Read the active profile from a runtime config directory.
413
+ *
414
+ * @param {string} runtimeConfigDir absolute path (e.g. ~/.claude/skills)
415
+ * @returns {string|null} profile name (e.g. 'core', 'standard', 'core,audit') or null
416
+ */
417
+ function readActiveProfile(runtimeConfigDir) {
418
+ const markerPath = path.join(runtimeConfigDir, PROFILE_MARKER_NAME);
419
+ try {
420
+ const raw = fs.readFileSync(markerPath, 'utf8').trim();
421
+ if (!raw) return null;
422
+ // Validate that it looks like a profile name (alphanumeric + hyphens + commas)
423
+ if (!/^[a-z0-9,_-]+$/i.test(raw)) return null;
424
+ return raw;
425
+ } catch {
426
+ return null;
427
+ }
428
+ }
429
+
430
+ /**
431
+ * Persist the active profile to a runtime config directory.
432
+ *
433
+ * @param {string} runtimeConfigDir absolute path (e.g. ~/.claude/skills)
434
+ * @param {string} profileName e.g. 'core', 'standard', 'full'
435
+ */
436
+ function writeActiveProfile(runtimeConfigDir, profileName) {
437
+ platformWriteSync(path.join(runtimeConfigDir, PROFILE_MARKER_NAME), profileName + '\n');
438
+ }
439
+
440
+ // ---------------------------------------------------------------------------
441
+ // Profile resolution helpers for install / update flows
442
+ // ---------------------------------------------------------------------------
443
+
444
+ /**
445
+ * Rank ordering for profiles (lower index = more restrictive / smaller skill set).
446
+ * Unknown profiles default to the permissive end (treated as 'full').
447
+ */
448
+ const PROFILE_RANK = Object.freeze(['core', 'standard', 'full']);
449
+
450
+ /**
451
+ * Given an array of profile names (one per runtime), return the most-restrictive
452
+ * profile — i.e. the one with the smallest effective skill set.
453
+ *
454
+ * Ordering (most to least restrictive): core < standard < full.
455
+ * Composed profiles (e.g. 'core,audit') and unknown profiles are treated as
456
+ * 'full' for this comparison.
457
+ *
458
+ * @param {string[]} profileNames
459
+ * @returns {string}
460
+ */
461
+ function mostRestrictiveProfile(profileNames) {
462
+ if (!profileNames || profileNames.length === 0) return 'full';
463
+ // Initialize with the least-restrictive rank (one past the end of PROFILE_RANK)
464
+ let bestRank = PROFILE_RANK.length;
465
+ let bestName = 'full';
466
+ for (const name of profileNames) {
467
+ const rank = PROFILE_RANK.indexOf(name);
468
+ // Unknown/composed profiles are treated as the permissive 'full' rank.
469
+ const effectiveRank = rank === -1 ? PROFILE_RANK.indexOf('full') : rank;
470
+ if (effectiveRank < bestRank) {
471
+ bestRank = effectiveRank;
472
+ bestName = rank === -1 ? 'full' : name;
473
+ }
474
+ }
475
+ return bestName;
476
+ }
477
+
478
+ /**
479
+ * Resolve the effective profile name for an install() run.
480
+ *
481
+ * Priority:
482
+ * 1. Explicit flag (requestedProfileName != null) → use it as-is.
483
+ * 2. Marker exists in targetDir and is not 'full' → use marker.
484
+ * 3. Else → 'full' (back-compat for fresh non-interactive installs).
485
+ *
486
+ * This is the single source-of-truth for the "which profile should this
487
+ * install() invocation use?" question. Extracted so it can be unit-tested
488
+ * independently of the bin/install.js megafile.
489
+ *
490
+ * @param {object} opts
491
+ * @param {string|null} opts.requestedProfileName explicit flag value (or null)
492
+ * @param {string} opts.targetDir runtime config dir (e.g. ~/.claude)
493
+ * @returns {string} profile name, e.g. 'core', 'standard', 'full'
494
+ */
495
+ function resolveEffectiveProfile({ requestedProfileName, targetDir }) {
496
+ // 1. Explicit flag overrides everything
497
+ if (requestedProfileName != null) return requestedProfileName;
498
+ // 2. Marker-driven (gsd update path)
499
+ const marker = readActiveProfile(targetDir);
500
+ if (marker && marker !== 'full') return marker;
501
+ // 3. Default
502
+ return 'full';
503
+ }
504
+
505
+ // ---------------------------------------------------------------------------
506
+ // Back-compat shims (deprecated — use profile-based API instead)
507
+ // ---------------------------------------------------------------------------
508
+
509
+ /**
510
+ * @deprecated Use PROFILES.core instead.
511
+ * Preserved for callers in install.js and existing tests.
512
+ */
513
+ const MINIMAL_SKILL_ALLOWLIST = Object.freeze([...PROFILES.core]);
514
+
515
+ const MINIMAL_ALLOWLIST_SET = new Set(MINIMAL_SKILL_ALLOWLIST);
516
+
517
+ /**
518
+ * @deprecated Use resolveProfile({ modes: ['core'] }) instead.
519
+ */
520
+ function isMinimalMode(mode) {
521
+ return mode === 'minimal' || mode === 'core-only';
522
+ }
523
+
524
+ /**
525
+ * Overloaded for back-compat.
526
+ * - If resolvedProfileOrMode is a string: legacy mode check (full/minimal)
527
+ * - If resolvedProfileOrMode is an object with .skills: new profile API
528
+ *
529
+ * @deprecated String-mode form; use resolvedProfile object form instead.
530
+ */
531
+ function shouldInstallSkill(skillBaseName, resolvedProfileOrMode) {
532
+ if (typeof resolvedProfileOrMode === 'object' && resolvedProfileOrMode !== null) {
533
+ const { skills } = resolvedProfileOrMode;
534
+ if (skills === '*') return true;
535
+ return skills instanceof Set && skills.has(skillBaseName);
536
+ }
537
+ // Legacy string mode
538
+ const mode = resolvedProfileOrMode;
539
+ if (!isMinimalMode(mode)) return true;
540
+ return MINIMAL_ALLOWLIST_SET.has(skillBaseName);
541
+ }
542
+
543
+ /**
544
+ * Stage a filtered copy of the source commands/gsd directory.
545
+ * Back-compat wrapper: maps 'minimal' → core profile, 'full' → full.
546
+ *
547
+ * @deprecated Use stageSkillsForProfile with a resolved profile instead.
548
+ * @param {string} srcDir absolute path to commands/gsd
549
+ * @param {string} mode 'full' | 'minimal'
550
+ * @returns {string} path to use (original or staged tmp)
551
+ */
552
+ function stageSkillsForMode(srcDir, mode) {
553
+ if (!isMinimalMode(mode)) return srcDir;
554
+ if (!fs.existsSync(srcDir)) return srcDir;
555
+
556
+ const stageDir = fs.mkdtempSync(path.join(os.tmpdir(), 'gsd-minimal-skills-'));
557
+ try {
558
+ const entries = fs.readdirSync(srcDir, { withFileTypes: true });
559
+ for (const entry of entries) {
560
+ if (!entry.isFile()) continue;
561
+ if (!entry.name.endsWith('.md')) continue;
562
+ const baseName = entry.name.replace(/\.md$/, '');
563
+ if (!shouldInstallSkill(baseName, mode)) continue;
564
+ fs.copyFileSync(
565
+ path.join(srcDir, entry.name),
566
+ path.join(stageDir, entry.name),
567
+ );
568
+ }
569
+ } catch (err) {
570
+ try { fs.rmSync(stageDir, { recursive: true, force: true }); } catch {}
571
+ throw err;
572
+ }
573
+ STAGED_DIRS.add(stageDir);
574
+ ensureExitCleanup();
575
+ return stageDir;
576
+ }
577
+
578
+ // ---------------------------------------------------------------------------
579
+ // Exports
580
+ // ---------------------------------------------------------------------------
581
+
582
+ module.exports = {
583
+ // New profile API (ADR-0011)
584
+ PROFILES,
585
+ PROFILE_RANK,
586
+ loadSkillsManifest,
587
+ resolveProfile,
588
+ resolveEffectiveProfile,
589
+ mostRestrictiveProfile,
590
+ stageSkillsForProfile,
591
+ stageAgentsForProfile,
592
+ stageSkillsForRuntimeAsSkills,
593
+ STAGED_DIRS,
594
+ readActiveProfile,
595
+ writeActiveProfile,
596
+ // Shared internals
597
+ cleanupStagedSkills,
598
+ // Back-compat / deprecated
599
+ MINIMAL_SKILL_ALLOWLIST,
600
+ isMinimalMode,
601
+ shouldInstallSkill,
602
+ stageSkillsForMode,
603
+ };