@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,470 @@
1
+ 'use strict';
2
+ /**
3
+ * Runtime surface module — ADR-0011 Phase 2 (Option B).
4
+ *
5
+ * Manages the runtime enable/disable surface state (the `.gsd-surface.json` marker in
6
+ * each runtime's config dir root (e.g., ~/.claude)) independently of the install-time profile marker
7
+ * (`.gsd-profile`). Runtime config locations are resolved by callers.
8
+ *
9
+ * Effective skill set = base profile ∪ explicitAdds − disabledClusters − explicitRemoves,
10
+ * then transitively closed via the manifest.
11
+ *
12
+ * Exports:
13
+ * readSurface(runtimeConfigDir)
14
+ * writeSurface(runtimeConfigDir, surfaceState)
15
+ * resolveSurface(runtimeConfigDir, manifest, clusterMap)
16
+ * applySurface(runtimeConfigDir, layout, manifest, clusterMap)
17
+ * listSurface(runtimeConfigDir, layout, manifest, clusterMap)
18
+ * pruneSkillDirs(skillsDir, retainedNames, prefix, manifest)
19
+ */
20
+
21
+ const fs = require('fs');
22
+ const path = require('path');
23
+ const os = require('os');
24
+ const { platformWriteSync } = require('./shell-command-projection.cjs');
25
+
26
+ const {
27
+ readActiveProfile,
28
+ resolveProfile,
29
+ stageSkillsForProfile,
30
+ stageAgentsForProfile,
31
+ loadSkillsManifest,
32
+ PROFILES,
33
+ } = require('./install-profiles.cjs');
34
+ const { CLUSTERS, allClusteredSkills } = require('./clusters.cjs');
35
+ const { findInstallSourceRoot } = require('./runtime-artifact-layout.cjs');
36
+
37
+ const SURFACE_FILE_NAME = '.gsd-surface.json';
38
+
39
+ // ---------------------------------------------------------------------------
40
+ // State IO
41
+ // ---------------------------------------------------------------------------
42
+
43
+ /**
44
+ * @typedef {Object} SurfaceState
45
+ * @property {string} baseProfile
46
+ * @property {string[]} disabledClusters
47
+ * @property {string[]} explicitAdds
48
+ * @property {string[]} explicitRemoves
49
+ */
50
+
51
+ /**
52
+ * Read the surface state from a runtime config directory.
53
+ *
54
+ * @param {string} runtimeConfigDir
55
+ * @returns {SurfaceState|null} null if file missing or corrupt
56
+ */
57
+ function readSurface(runtimeConfigDir) {
58
+ const filePath = path.join(runtimeConfigDir, SURFACE_FILE_NAME);
59
+ try {
60
+ const raw = fs.readFileSync(filePath, 'utf8');
61
+ const parsed = JSON.parse(raw);
62
+ // Structural validation — must have these fields with expected types
63
+ if (typeof parsed !== 'object' || parsed === null) return null;
64
+ if (typeof parsed.baseProfile !== 'string') return null;
65
+ if (!Array.isArray(parsed.disabledClusters)) return null;
66
+ if (!Array.isArray(parsed.explicitAdds)) return null;
67
+ if (!Array.isArray(parsed.explicitRemoves)) return null;
68
+ return {
69
+ baseProfile: parsed.baseProfile,
70
+ disabledClusters: parsed.disabledClusters,
71
+ explicitAdds: parsed.explicitAdds,
72
+ explicitRemoves: parsed.explicitRemoves,
73
+ };
74
+ } catch {
75
+ return null;
76
+ }
77
+ }
78
+
79
+ /**
80
+ * Write the surface state atomically via the platform seam (mkdir + tmp+rename).
81
+ *
82
+ * @param {string} runtimeConfigDir
83
+ * @param {SurfaceState} surfaceState
84
+ */
85
+ function writeSurface(runtimeConfigDir, surfaceState) {
86
+ platformWriteSync(path.join(runtimeConfigDir, SURFACE_FILE_NAME), JSON.stringify(surfaceState, null, 2) + '\n');
87
+ }
88
+
89
+ // ---------------------------------------------------------------------------
90
+ // Resolution
91
+ // ---------------------------------------------------------------------------
92
+
93
+ /**
94
+ * Expand cluster names to skill stems using the provided clusterMap.
95
+ *
96
+ * @param {string[]} clusterNames
97
+ * @param {Object} clusterMap CLUSTERS or override
98
+ * @returns {Set<string>}
99
+ */
100
+ function clustersToSkills(clusterNames, clusterMap) {
101
+ const result = new Set();
102
+ for (const name of clusterNames) {
103
+ const members = clusterMap[name];
104
+ if (members) {
105
+ for (const s of members) result.add(s);
106
+ }
107
+ }
108
+ return result;
109
+ }
110
+
111
+ /**
112
+ * Normalize manifest inputs to the skill-dependency Map shape expected by
113
+ * resolveProfile/computeClosure.
114
+ *
115
+ * Supports:
116
+ * - canonical Map<string, string[]>
117
+ * - legacy plain object map: { [stem]: string[] }
118
+ * - parsed gsd-file-manifest.json object ({ files: { ... } }) by rebuilding
119
+ * the dependency manifest from the install source tree
120
+ *
121
+ * @param {string} runtimeConfigDir
122
+ * @param {Map<string, string[]>|Object} manifest
123
+ * @returns {Map<string, string[]>}
124
+ */
125
+ function normalizeSkillManifest(runtimeConfigDir, manifest) {
126
+ if (manifest instanceof Map) {
127
+ return manifest;
128
+ }
129
+ if (!manifest || typeof manifest !== 'object') {
130
+ return new Map();
131
+ }
132
+
133
+ // Legacy/ad-hoc object map: stem -> requires[]
134
+ const arrayEntries = Object.entries(manifest).filter(([, value]) => Array.isArray(value));
135
+ if (arrayEntries.length > 0) {
136
+ return new Map(arrayEntries.map(([stem, deps]) => [stem, deps]));
137
+ }
138
+
139
+ // Parsed gsd-file-manifest.json shape: rebuild the dependency map from source.
140
+ if (manifest.files && typeof manifest.files === 'object') {
141
+ const srcCommandsDir = findInstallSourceRoot(runtimeConfigDir);
142
+ return loadSkillsManifest(srcCommandsDir);
143
+ }
144
+
145
+ return new Map();
146
+ }
147
+
148
+ /**
149
+ * Resolve the effective surface to a typed profile-like object.
150
+ * Shape: { name, skills: Set<string>|'*', agents: Set<string> }
151
+ *
152
+ * Resolution order:
153
+ * 1. Start with base profile resolved via resolveProfile()
154
+ * 2. Remove skills in disabled clusters
155
+ * 3. Add explicitAdds (and their transitive closure)
156
+ * 4. Remove explicitRemoves (only the stem itself, no cascade)
157
+ *
158
+ * @param {string} runtimeConfigDir
159
+ * @param {Map<string, string[]>} manifest
160
+ * @param {Object} [clusterMap] defaults to CLUSTERS
161
+ * @returns {{ name: string, skills: Set<string>, agents: Set<string> }}
162
+ */
163
+ function resolveSurface(runtimeConfigDir, manifest, clusterMap) {
164
+ const cm = clusterMap || CLUSTERS;
165
+ const skillManifest = normalizeSkillManifest(runtimeConfigDir, manifest);
166
+ const surface = readSurface(runtimeConfigDir);
167
+
168
+ // Determine base profile name: from surface state or from .gsd-profile marker
169
+ const baseProfileName = (surface && surface.baseProfile)
170
+ ? surface.baseProfile
171
+ : (readActiveProfile(runtimeConfigDir) || 'full');
172
+
173
+ // Resolve base profile
174
+ const baseResolved = resolveProfile({
175
+ modes: baseProfileName.split(',').map(s => s.trim()),
176
+ manifest: skillManifest,
177
+ });
178
+
179
+ // If full, we need to enumerate all skills from the manifest
180
+ let skills;
181
+ if (baseResolved.skills === '*') {
182
+ // Materialize all skill stems from manifest
183
+ skills = new Set();
184
+ for (const [key] of skillManifest) {
185
+ if (!key.startsWith('_calls_agents_')) skills.add(key);
186
+ }
187
+ } else {
188
+ skills = new Set(baseResolved.skills);
189
+ }
190
+
191
+ if (surface) {
192
+ // Step 2: remove disabled cluster members
193
+ const disabledSkills = clustersToSkills(surface.disabledClusters, cm);
194
+ for (const s of disabledSkills) skills.delete(s);
195
+
196
+ // Step 3: add explicitAdds with transitive closure
197
+ if (surface.explicitAdds.length > 0) {
198
+ const addSet = new Set(surface.explicitAdds);
199
+ // Compute closure of adds
200
+ const queue = [...addSet];
201
+ const visited = new Set(addSet);
202
+ while (queue.length > 0) {
203
+ const stem = queue.pop();
204
+ const deps = skillManifest.get(stem) || [];
205
+ for (const dep of deps) {
206
+ if (!visited.has(dep)) {
207
+ visited.add(dep);
208
+ queue.push(dep);
209
+ }
210
+ }
211
+ }
212
+ for (const s of visited) skills.add(s);
213
+ }
214
+
215
+ // Step 4: remove explicitRemoves (stem only, no cascade)
216
+ for (const s of surface.explicitRemoves) {
217
+ skills.delete(s);
218
+ }
219
+ }
220
+
221
+ // Derive agents from skills
222
+ const agents = new Set();
223
+ for (const skillStem of skills) {
224
+ const agentRefs = skillManifest.get(`_calls_agents_${skillStem}`) || [];
225
+ for (const agentStem of agentRefs) agents.add(agentStem);
226
+ }
227
+
228
+ const name = surface ? `surface:${surface.baseProfile}` : `profile:${baseProfileName}`;
229
+ return { name, skills, agents };
230
+ }
231
+
232
+ // ---------------------------------------------------------------------------
233
+ // Apply
234
+ // ---------------------------------------------------------------------------
235
+
236
+ /**
237
+ * Re-stage the active surface using the resolved layout.
238
+ * Iterates layout.kinds and syncs each artifact kind to its destination.
239
+ *
240
+ * @param {string} runtimeConfigDir
241
+ * @param {import('./runtime-artifact-layout.cjs').Layout} layout
242
+ * @param {Map<string, string[]>} manifest
243
+ * @param {Object} [clusterMap]
244
+ */
245
+ function applySurface(runtimeConfigDir, layout, manifest, clusterMap) {
246
+ if (path.resolve(runtimeConfigDir) !== path.resolve(layout.configDir)) {
247
+ throw new TypeError('applySurface runtimeConfigDir must match layout.configDir');
248
+ }
249
+ const skillManifest = normalizeSkillManifest(layout.configDir, manifest);
250
+ const resolved = resolveSurface(layout.configDir, skillManifest, clusterMap);
251
+ for (const kind of layout.kinds) {
252
+ const staged = kind.stage(resolved);
253
+ const dest = path.join(layout.configDir, kind.destSubpath);
254
+ _syncGsdDir(staged, dest, kind, skillManifest);
255
+ }
256
+ return resolved;
257
+ }
258
+
259
+ /**
260
+ * Prune GSD-managed skill directories from a skills directory.
261
+ *
262
+ * Removes every directory in `skillsDir` that is GSD-owned but NOT listed
263
+ * in `retainedNames`. User-owned dirs (not matching the GSD ownership criteria)
264
+ * are always preserved.
265
+ *
266
+ * Ownership criteria:
267
+ * - Non-empty prefix (e.g. 'gsd-'): dir name starts with that prefix AND
268
+ * appears in the manifest (manifest membership is required). Dirs that match
269
+ * the prefix but are NOT in the manifest are treated as user-owned and
270
+ * preserved — this prevents data loss for user-created gsd-* directories.
271
+ * A warning is written to stderr when such a dir is encountered.
272
+ * - Empty prefix (Hermes): dir name appears as a canonical skill stem in the
273
+ * manifest. User dirs not in the manifest are preserved.
274
+ * - Empty prefix without manifest, or manifest not a Map: conservative; no
275
+ * dirs are removed.
276
+ *
277
+ * This is the single point of truth for skill-dir pruning. Both _syncGsdDir
278
+ * (surface apply) and callers that need stand-alone pruning use this function.
279
+ *
280
+ * @param {string} skillsDir directory that contains the gsd-STEM sub-dirs
281
+ * @param {Set<string>} retainedNames set of directory names to keep (e.g. 'gsd-help')
282
+ * @param {string} prefix GSD dir prefix, e.g. 'gsd-' (or '' for Hermes)
283
+ * @param {Map<string, string[]>} [manifest] optional; required for Hermes empty-prefix case
284
+ * and for manifest-membership gate in prefixed case.
285
+ * Must be a Map; any other type is treated as missing.
286
+ */
287
+ function pruneSkillDirs(skillsDir, retainedNames, prefix, manifest) {
288
+ if (!fs.existsSync(skillsDir)) return;
289
+
290
+ // Finding 2: guard against callers passing a truthy non-Map as manifest.
291
+ // A non-Map manifest would throw on .keys(); treat it as absent and be conservative.
292
+ const safeManifest = (manifest instanceof Map) ? manifest : null;
293
+
294
+ // Build the canonical stem set from the manifest (used for both prefixed and Hermes paths).
295
+ // Deletion requires manifest membership — without a valid manifest, be conservative.
296
+ const canonicalStems = safeManifest
297
+ ? new Set([...safeManifest.keys()].filter(k => !k.startsWith('_calls_agents_')))
298
+ : null;
299
+
300
+ for (const entry of fs.readdirSync(skillsDir)) {
301
+ const entryPath = path.join(skillsDir, entry);
302
+ if (!fs.statSync(entryPath).isDirectory()) continue;
303
+
304
+ let isGsdOwned;
305
+ if (prefix !== '') {
306
+ if (!entry.startsWith(prefix)) {
307
+ // Does not match prefix at all — user-owned, preserve.
308
+ continue;
309
+ }
310
+ if (!canonicalStems) {
311
+ // No manifest available: cannot confirm ownership — preserve conservatively.
312
+ continue;
313
+ }
314
+ // Finding 1 fix: prefix match is necessary but NOT sufficient.
315
+ // The dir must also be in the manifest to be considered GSD-owned.
316
+ // A user-created gsd-* dir that isn't in the manifest is preserved with a warning.
317
+ if (!canonicalStems.has(entry.slice(prefix.length))) {
318
+ process.stderr.write(
319
+ `[gsd] Warning: ${entry} matches GSD prefix '${prefix}' but is not in the manifest — preserving (user-owned or unknown)\n`
320
+ );
321
+ continue;
322
+ }
323
+ isGsdOwned = true;
324
+ } else if (canonicalStems) {
325
+ // Hermes: GSD-owned iff the directory name appears in the canonical manifest.
326
+ isGsdOwned = canonicalStems.has(entry);
327
+ } else {
328
+ // No manifest available: be conservative, don't remove anything.
329
+ continue;
330
+ }
331
+
332
+ if (!isGsdOwned) continue; // Hermes path only: preserve user-owned dirs not in manifest
333
+ if (retainedNames.has(entry)) continue; // GSD-owned and in retain set
334
+ try {
335
+ fs.rmSync(entryPath, { recursive: true, force: true });
336
+ } catch (err) {
337
+ process.stderr.write(`surface: failed to prune ${entryPath}: ${err.message}\n`);
338
+ }
339
+ }
340
+ }
341
+
342
+ /**
343
+ * Sync destination directory from staged source.
344
+ *
345
+ * For 'commands' kind: iterate *.md files in destDir, remove if not in staged set.
346
+ * For 'agents' kind: same, but only remove files starting with 'gsd-' prefix.
347
+ * For 'skills' kind: iterate directories in destDir matching kind.prefix; add missing
348
+ * by copying recursively; remove dirs not in staged set. Preserves dirs not matching
349
+ * the prefix (user-owned skills). Pruning is delegated to pruneSkillDirs().
350
+ *
351
+ * For Hermes (empty prefix): uses manifest membership to discriminate GSD-owned vs
352
+ * user-owned dirs. GSD-owned = stem in manifest; removal targets = in manifest AND
353
+ * not in staged set. User-owned (not in manifest) are always preserved.
354
+ *
355
+ * @param {string} stagedDir source (staged temp dir or original)
356
+ * @param {string} destDir runtime destination
357
+ * @param {import('./runtime-artifact-layout.cjs').ArtifactKind|'commands'|'agents'} kind
358
+ * @param {Map<string, string[]>} [manifest] optional; required for Hermes empty-prefix removal
359
+ */
360
+ function _syncGsdDir(stagedDir, destDir, kind, manifest) {
361
+ if (!fs.existsSync(stagedDir)) return;
362
+ fs.mkdirSync(destDir, { recursive: true });
363
+
364
+ // Normalize: allow legacy string context for backward-compat with internal callers
365
+ const kindName = (typeof kind === 'string') ? kind : kind.kind;
366
+ const kindPrefix = (typeof kind === 'object' && kind !== null) ? kind.prefix : 'gsd-';
367
+
368
+ if (kindName === 'skills') {
369
+ // Skills kind: work with directories, not files.
370
+ // Each staged entry is a directory named ${prefix}${stem}.
371
+ const stagedDirs = new Set(
372
+ fs.readdirSync(stagedDir).filter(entry => {
373
+ return fs.statSync(path.join(stagedDir, entry)).isDirectory();
374
+ })
375
+ );
376
+
377
+ // Copy missing dirs from staged to dest (always overwrite to ensure content is current)
378
+ for (const dirName of stagedDirs) {
379
+ const destSubDir = path.join(destDir, dirName);
380
+ fs.cpSync(path.join(stagedDir, dirName), destSubDir, { recursive: true });
381
+ }
382
+
383
+ // Prune GSD-owned dirs that are no longer in the staged set.
384
+ // pruneSkillDirs() is the single point of truth for this logic.
385
+ pruneSkillDirs(destDir, stagedDirs, kindPrefix, manifest);
386
+ } else {
387
+ // commands / agents kind: work with .md files
388
+ const stagedFiles = new Set(
389
+ fs.readdirSync(stagedDir).filter(f => f.endsWith('.md'))
390
+ );
391
+
392
+ // Copy files from staged to dest (overwrite to keep content current)
393
+ for (const file of stagedFiles) {
394
+ fs.copyFileSync(path.join(stagedDir, file), path.join(destDir, file));
395
+ }
396
+
397
+ // Remove gsd-only files from dest that aren't in staged set
398
+ // For commands dir: all .md files are gsd skills
399
+ // For agents dir: only gsd-* files
400
+ const destEntries = fs.readdirSync(destDir).filter(f => f.endsWith('.md'));
401
+ for (const file of destEntries) {
402
+ if (kindName === 'agents' && !file.startsWith('gsd-')) continue;
403
+ if (!stagedFiles.has(file)) {
404
+ try { fs.unlinkSync(path.join(destDir, file)); } catch {}
405
+ }
406
+ }
407
+ }
408
+ }
409
+
410
+ // ---------------------------------------------------------------------------
411
+ // List
412
+ // ---------------------------------------------------------------------------
413
+
414
+ /**
415
+ * List the currently enabled and disabled skills with token cost.
416
+ *
417
+ * Token cost = sum of description lengths ÷ 4 (mirrors audit script).
418
+ * Descriptions are read from the install source (findInstallSourceRoot).
419
+ *
420
+ * @param {string} runtimeConfigDir
421
+ * @param {Map<string, string[]>} manifest
422
+ * @param {Object} [clusterMap]
423
+ * @returns {{ enabled: string[], disabled: string[], tokenCost: number }}
424
+ */
425
+ function listSurface(runtimeConfigDir, manifest, clusterMap) {
426
+ const skillManifest = normalizeSkillManifest(runtimeConfigDir, manifest);
427
+ const resolved = resolveSurface(runtimeConfigDir, skillManifest, clusterMap);
428
+
429
+ // All known stems from manifest (exclude _calls_agents_ meta keys)
430
+ const allStems = [];
431
+ for (const [key] of skillManifest) {
432
+ if (!key.startsWith('_calls_agents_')) allStems.push(key);
433
+ }
434
+
435
+ const enabledSet = resolved.skills instanceof Set ? resolved.skills : new Set(allStems);
436
+
437
+ const enabled = allStems.filter(s => enabledSet.has(s)).sort();
438
+ const disabled = allStems.filter(s => !enabledSet.has(s)).sort();
439
+
440
+ // Compute token cost by reading descriptions from the install source
441
+ const srcCommandsDir = findInstallSourceRoot(runtimeConfigDir);
442
+ let tokenCost = 0;
443
+ for (const stem of enabled) {
444
+ const filePath = path.join(srcCommandsDir, `${stem}.md`);
445
+ try {
446
+ const content = fs.readFileSync(filePath, 'utf8');
447
+ const descMatch = content.match(/^description:\s*(.+)$/m);
448
+ if (descMatch) {
449
+ tokenCost += Math.ceil(descMatch[1].trim().length / 4);
450
+ }
451
+ } catch {}
452
+ }
453
+
454
+ return { enabled, disabled, tokenCost };
455
+ }
456
+
457
+ // ---------------------------------------------------------------------------
458
+ // Exports
459
+ // ---------------------------------------------------------------------------
460
+
461
+ module.exports = {
462
+ readSurface,
463
+ writeSurface,
464
+ resolveSurface,
465
+ applySurface,
466
+ listSurface,
467
+ // Exported for testing and for callers that need stand-alone pruning
468
+ pruneSkillDirs,
469
+ _syncGsdDir,
470
+ };
@@ -0,0 +1,81 @@
1
+ 'use strict';
2
+
3
+ const fs = require('fs');
4
+ const path = require('path');
5
+ const { output, error, ERROR_REASON } = require('./core.cjs');
6
+
7
+ function isBehaviorAddingTaskContent(content) {
8
+ const tddTrue = /\btdd\s*=\s*["']true["']/i.test(content);
9
+
10
+ const behaviorMatch = content.match(/<behavior>([\s\S]*?)<\/behavior>/i);
11
+ const hasBehaviorBlock = Boolean(behaviorMatch && behaviorMatch[1].trim().length > 0);
12
+
13
+ const filesMatch = content.match(/<files>([\s\S]*?)<\/files>/i);
14
+ let hasSourceFiles = false;
15
+ if (filesMatch) {
16
+ const fileLines = filesMatch[1]
17
+ .split(/[\n,]/)
18
+ .map((line) => line.trim().replace(/^[-*]\s*/, ''))
19
+ .filter(Boolean);
20
+ hasSourceFiles = fileLines.some((file) =>
21
+ !/\.md$/i.test(file) &&
22
+ !/\.json$/i.test(file) &&
23
+ !/\.test\.[^.]+$/i.test(file) &&
24
+ !/\.spec\.[^.]+$/i.test(file) &&
25
+ !/(^|[\\/])tests?[\\/]/i.test(file) &&
26
+ !/\.(yml|yaml|toml|ini|cfg|conf|properties)$/i.test(file) &&
27
+ !/(^|[\\/])\.env(\..+)?$/i.test(file)
28
+ );
29
+ }
30
+
31
+ const isBehaviorAdding = tddTrue && hasBehaviorBlock && hasSourceFiles;
32
+ const missing = [];
33
+ if (!tddTrue) missing.push('tdd="true" frontmatter absent');
34
+ if (!hasBehaviorBlock) missing.push('<behavior> block missing or empty');
35
+ if (!hasSourceFiles) missing.push('<files> has no non-test source file');
36
+
37
+ return {
38
+ is_behavior_adding: isBehaviorAdding,
39
+ checks: {
40
+ tdd_true: tddTrue,
41
+ has_behavior_block: hasBehaviorBlock,
42
+ has_source_files: hasSourceFiles,
43
+ },
44
+ reason: isBehaviorAdding ? null : `Not behavior-adding: ${missing.join('; ')}`,
45
+ };
46
+ }
47
+
48
+ function routeTaskCommand({ args, cwd, raw }) {
49
+ const subcommand = args[1];
50
+ if (subcommand !== 'is-behavior-adding') {
51
+ error('Unknown task subcommand. Available: is-behavior-adding', ERROR_REASON.SDK_UNKNOWN_COMMAND);
52
+ }
53
+
54
+ let content = null;
55
+ if (args[2] === '--task-content') {
56
+ content = args[3] || null;
57
+ } else if (args[2]) {
58
+ const projectRoot = path.resolve(cwd || process.cwd());
59
+ const requestedPath = args[2];
60
+ const resolvedTaskPath = path.resolve(projectRoot, requestedPath);
61
+ const rel = path.relative(projectRoot, resolvedTaskPath);
62
+ if (rel === '..' || rel.startsWith(`..${path.sep}`)) {
63
+ error(`Task file is outside project scope: ${requestedPath}`, ERROR_REASON.USAGE);
64
+ }
65
+ if (!fs.existsSync(resolvedTaskPath)) {
66
+ error(`Task file not found: ${requestedPath}`, ERROR_REASON.USAGE);
67
+ }
68
+ content = fs.readFileSync(resolvedTaskPath, 'utf-8');
69
+ }
70
+
71
+ if (!content) {
72
+ error('Usage: task.is-behavior-adding <plan-file-path> | --task-content "<xml>"', ERROR_REASON.USAGE);
73
+ }
74
+
75
+ output(isBehaviorAddingTaskContent(content), raw);
76
+ }
77
+
78
+ module.exports = {
79
+ isBehaviorAddingTaskContent,
80
+ routeTaskCommand,
81
+ };