claude-code-pilot 2.0.0 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (455) hide show
  1. package/bin/install.js +267 -250
  2. package/manifest.json +5 -18
  3. package/package.json +5 -7
  4. package/src/agents/build-error-resolver.md +114 -0
  5. package/src/agents/ccp-advisor-researcher.md +104 -0
  6. package/src/agents/ccp-assumptions-analyzer.md +105 -0
  7. package/{gsd/agents/gsd-codebase-mapper.md → src/agents/ccp-codebase-mapper.md} +7 -7
  8. package/{gsd/agents/gsd-debugger.md → src/agents/ccp-debugger.md} +125 -8
  9. package/{gsd/agents/gsd-executor.md → src/agents/ccp-executor.md} +31 -20
  10. package/{gsd/agents/gsd-integration-checker.md → src/agents/ccp-integration-checker.md} +2 -2
  11. package/{gsd/agents/gsd-nyquist-auditor.md → src/agents/ccp-nyquist-auditor.md} +3 -3
  12. package/{gsd/agents/gsd-phase-researcher.md → src/agents/ccp-phase-researcher.md} +127 -13
  13. package/{gsd/agents/gsd-plan-checker.md → src/agents/ccp-plan-checker.md} +57 -21
  14. package/{gsd/agents/gsd-planner.md → src/agents/ccp-planner.md} +61 -23
  15. package/{gsd/agents/gsd-project-researcher.md → src/agents/ccp-project-researcher.md} +33 -6
  16. package/{gsd/agents/gsd-research-synthesizer.md → src/agents/ccp-research-synthesizer.md} +11 -11
  17. package/{gsd/agents/gsd-roadmapper.md → src/agents/ccp-roadmapper.md} +39 -10
  18. package/src/agents/ccp-ui-auditor.md +439 -0
  19. package/src/agents/ccp-ui-checker.md +300 -0
  20. package/src/agents/ccp-ui-researcher.md +357 -0
  21. package/{gsd/agents/gsd-verifier.md → src/agents/ccp-verifier.md} +81 -15
  22. package/src/agents/cpp-build-resolver.md +90 -0
  23. package/src/agents/cpp-reviewer.md +72 -0
  24. package/src/agents/database-reviewer.md +91 -0
  25. package/src/agents/docs-lookup.md +68 -0
  26. package/src/agents/flutter-reviewer.md +243 -0
  27. package/src/agents/go-build-resolver.md +94 -0
  28. package/src/agents/go-reviewer.md +76 -0
  29. package/src/agents/java-build-resolver.md +153 -0
  30. package/src/agents/java-reviewer.md +92 -0
  31. package/src/agents/kotlin-build-resolver.md +118 -0
  32. package/src/agents/kotlin-reviewer.md +159 -0
  33. package/src/agents/planner.md +212 -0
  34. package/src/agents/python-reviewer.md +98 -0
  35. package/src/agents/pytorch-build-resolver.md +120 -0
  36. package/src/agents/refactor-cleaner.md +85 -0
  37. package/src/agents/rust-build-resolver.md +148 -0
  38. package/src/agents/rust-reviewer.md +94 -0
  39. package/src/agents/typescript-reviewer.md +112 -0
  40. package/src/available-rules/README.md +80 -0
  41. package/src/available-rules/cpp/coding-style.md +44 -0
  42. package/src/available-rules/cpp/hooks.md +39 -0
  43. package/src/available-rules/cpp/patterns.md +51 -0
  44. package/src/available-rules/cpp/security.md +51 -0
  45. package/src/available-rules/cpp/testing.md +44 -0
  46. package/src/available-rules/csharp/coding-style.md +72 -0
  47. package/src/available-rules/csharp/hooks.md +25 -0
  48. package/src/available-rules/csharp/patterns.md +50 -0
  49. package/src/available-rules/csharp/security.md +58 -0
  50. package/src/available-rules/csharp/testing.md +46 -0
  51. package/src/available-rules/java/coding-style.md +114 -0
  52. package/src/available-rules/java/hooks.md +18 -0
  53. package/src/available-rules/java/patterns.md +146 -0
  54. package/src/available-rules/java/security.md +100 -0
  55. package/src/available-rules/java/testing.md +131 -0
  56. package/src/available-rules/kotlin/hooks.md +17 -0
  57. package/src/available-rules/rust/coding-style.md +151 -0
  58. package/src/available-rules/rust/hooks.md +16 -0
  59. package/src/available-rules/rust/patterns.md +168 -0
  60. package/src/available-rules/rust/security.md +141 -0
  61. package/src/available-rules/rust/testing.md +154 -0
  62. package/src/commands/aside.md +164 -0
  63. package/src/commands/build-fix.md +62 -0
  64. package/src/commands/ccp/add-backlog.md +76 -0
  65. package/{gsd/commands-gsd → src/commands/ccp}/add-phase.md +3 -3
  66. package/{gsd/commands-gsd → src/commands/ccp}/add-tests.md +5 -5
  67. package/{gsd/commands-gsd → src/commands/ccp}/add-todo.md +4 -4
  68. package/{gsd/commands-gsd → src/commands/ccp}/audit-milestone.md +3 -3
  69. package/src/commands/ccp/audit-uat.md +24 -0
  70. package/src/commands/ccp/autonomous.md +41 -0
  71. package/{gsd/commands-gsd → src/commands/ccp}/check-todos.md +3 -3
  72. package/{gsd/commands-gsd → src/commands/ccp}/cleanup.md +3 -3
  73. package/{gsd/commands-gsd → src/commands/ccp}/complete-milestone.md +9 -9
  74. package/{gsd/commands-gsd → src/commands/ccp}/debug.md +14 -9
  75. package/src/commands/ccp/discuss-phase.md +64 -0
  76. package/src/commands/ccp/do.md +30 -0
  77. package/src/commands/ccp/execute-phase.md +59 -0
  78. package/src/commands/ccp/fast.md +30 -0
  79. package/src/commands/ccp/forensics.md +56 -0
  80. package/{gsd/commands-gsd → src/commands/ccp}/health.md +3 -3
  81. package/{gsd/commands-gsd → src/commands/ccp}/help.md +5 -5
  82. package/{gsd/commands-gsd → src/commands/ccp}/insert-phase.md +3 -3
  83. package/{gsd/commands-gsd → src/commands/ccp}/list-phase-assumptions.md +2 -2
  84. package/src/commands/ccp/manager.md +39 -0
  85. package/{gsd/commands-gsd → src/commands/ccp}/map-codebase.md +7 -7
  86. package/src/commands/ccp/milestone-summary.md +51 -0
  87. package/{gsd/commands-gsd → src/commands/ccp}/new-milestone.md +8 -8
  88. package/{gsd/commands-gsd → src/commands/ccp}/new-project.md +8 -8
  89. package/src/commands/ccp/next.md +24 -0
  90. package/src/commands/ccp/note.md +34 -0
  91. package/{gsd/commands-gsd → src/commands/ccp}/pause-work.md +3 -3
  92. package/{gsd/commands-gsd → src/commands/ccp}/plan-milestone-gaps.md +5 -5
  93. package/{gsd/commands-gsd → src/commands/ccp}/plan-phase.md +9 -7
  94. package/src/commands/ccp/plant-seed.md +28 -0
  95. package/src/commands/ccp/pr-branch.md +25 -0
  96. package/{gsd/commands-gsd → src/commands/ccp}/progress.md +3 -3
  97. package/{gsd/commands-gsd → src/commands/ccp}/quick.md +10 -8
  98. package/{gsd/commands-gsd → src/commands/ccp}/remove-phase.md +3 -3
  99. package/{gsd/commands-gsd → src/commands/ccp}/research-phase.md +17 -12
  100. package/{gsd/commands-gsd → src/commands/ccp}/resume-work.md +3 -3
  101. package/src/commands/ccp/review-backlog.md +61 -0
  102. package/src/commands/ccp/session-report.md +19 -0
  103. package/src/commands/ccp/set-profile.md +12 -0
  104. package/{gsd/commands-gsd → src/commands/ccp}/settings.md +5 -5
  105. package/src/commands/ccp/ship.md +23 -0
  106. package/src/commands/ccp/stats.md +18 -0
  107. package/src/commands/ccp/thread.md +127 -0
  108. package/src/commands/ccp/ui-phase.md +34 -0
  109. package/src/commands/ccp/ui-review.md +32 -0
  110. package/{gsd/commands-gsd → src/commands/ccp}/update.md +5 -5
  111. package/{gsd/commands-gsd → src/commands/ccp}/validate-phase.md +3 -3
  112. package/{gsd/commands-gsd → src/commands/ccp}/verify-work.md +5 -5
  113. package/src/commands/code-review.md +40 -0
  114. package/src/commands/context-budget.md +29 -0
  115. package/src/commands/cpp-build.md +173 -0
  116. package/src/commands/cpp-review.md +132 -0
  117. package/src/commands/cpp-test.md +251 -0
  118. package/src/commands/docs.md +31 -0
  119. package/src/commands/e2e.md +364 -0
  120. package/src/commands/eval.md +120 -0
  121. package/{ecc → src}/commands/evolve.md +2 -2
  122. package/src/commands/go-build.md +183 -0
  123. package/src/commands/go-review.md +148 -0
  124. package/src/commands/go-test.md +268 -0
  125. package/src/commands/gradle-build.md +70 -0
  126. package/src/commands/harness-audit.md +71 -0
  127. package/src/commands/kotlin-build.md +174 -0
  128. package/src/commands/kotlin-review.md +140 -0
  129. package/src/commands/kotlin-test.md +312 -0
  130. package/src/commands/orchestrate.md +231 -0
  131. package/src/commands/plan.md +114 -0
  132. package/src/commands/prompt-optimize.md +38 -0
  133. package/src/commands/prune.md +25 -0
  134. package/src/commands/python-review.md +297 -0
  135. package/{ecc → src}/commands/quality-gate.md +1 -1
  136. package/src/commands/refactor-clean.md +80 -0
  137. package/src/commands/rules-distill.md +11 -0
  138. package/src/commands/rust-build.md +187 -0
  139. package/src/commands/rust-review.md +142 -0
  140. package/src/commands/rust-test.md +308 -0
  141. package/{ecc → src}/commands/sessions.md +10 -10
  142. package/src/commands/setup-pm.md +80 -0
  143. package/{kit → src}/commands/setup.md +45 -19
  144. package/src/commands/skill-create.md +172 -0
  145. package/src/commands/skill-health.md +51 -0
  146. package/src/commands/tdd.md +328 -0
  147. package/src/commands/test-coverage.md +69 -0
  148. package/src/commands/update-codemaps.md +72 -0
  149. package/src/commands/update-docs.md +84 -0
  150. package/{gsd/hooks/gsd-context-monitor.js → src/hooks/ccp-context-monitor.js} +3 -3
  151. package/src/hooks/ccp-prompt-guard.js +96 -0
  152. package/{gsd/hooks/gsd-statusline.js → src/hooks/ccp-statusline.js} +7 -7
  153. package/src/hooks/ccp-workflow-guard.js +94 -0
  154. package/src/hooks/config-protection.js +141 -0
  155. package/{kit → src}/hooks/kit-check-update.js +7 -4
  156. package/src/hooks/mcp-health-check.js +620 -0
  157. package/{ecc/scripts → src}/hooks/run-with-flags-shell.sh +1 -1
  158. package/{ecc/scripts → src}/hooks/run-with-flags.js +74 -13
  159. package/src/hooks/session-end-marker.js +29 -0
  160. package/{ecc/scripts → src}/hooks/session-end.js +83 -40
  161. package/{ecc/scripts → src}/hooks/session-start.js +75 -9
  162. package/{ecc/scripts → src}/lib/hook-flags.js +8 -4
  163. package/{ecc/scripts → src}/lib/project-detect.js +2 -1
  164. package/{ecc/scripts → src}/lib/session-manager.d.ts +5 -1
  165. package/{ecc/scripts → src}/lib/session-manager.js +202 -92
  166. package/{ecc/scripts → src}/lib/utils.d.ts +23 -1
  167. package/{ecc/scripts → src}/lib/utils.js +91 -3
  168. package/{gsd/get-shit-done/bin/gsd-tools.cjs → src/pilot/bin/ccp-tools.cjs} +257 -86
  169. package/{gsd/get-shit-done → src/pilot}/bin/lib/commands.cjs +1 -1
  170. package/src/pilot/bin/lib/config.cjs +444 -0
  171. package/src/pilot/bin/lib/core.cjs +1190 -0
  172. package/src/pilot/bin/lib/init.cjs +1281 -0
  173. package/src/pilot/bin/lib/model-profiles.cjs +67 -0
  174. package/{gsd/get-shit-done → src/pilot}/bin/lib/phase.cjs +2 -2
  175. package/src/pilot/bin/lib/security.cjs +382 -0
  176. package/{gsd/get-shit-done → src/pilot}/bin/lib/state.cjs +1 -1
  177. package/src/pilot/bin/lib/uat.cjs +282 -0
  178. package/{gsd/get-shit-done → src/pilot}/bin/lib/verify.cjs +10 -10
  179. package/{gsd/get-shit-done → src/pilot}/references/continuation-format.md +16 -16
  180. package/{gsd/get-shit-done → src/pilot}/references/decimal-phase-calculation.md +5 -5
  181. package/{gsd/get-shit-done → src/pilot}/references/git-integration.md +5 -5
  182. package/{gsd/get-shit-done → src/pilot}/references/git-planning-commit.md +4 -4
  183. package/src/pilot/references/mcp-servers.json +153 -0
  184. package/{gsd/get-shit-done → src/pilot}/references/model-profile-resolution.md +2 -2
  185. package/{gsd/get-shit-done → src/pilot}/references/model-profiles.md +20 -20
  186. package/{gsd/get-shit-done → src/pilot}/references/phase-argument-parsing.md +4 -4
  187. package/{gsd/get-shit-done → src/pilot}/references/planning-config.md +15 -15
  188. package/{gsd/get-shit-done → src/pilot}/references/ui-brand.md +5 -5
  189. package/{gsd/get-shit-done → src/pilot}/references/verification-patterns.md +1 -1
  190. package/{gsd/get-shit-done → src/pilot}/templates/DEBUG.md +1 -1
  191. package/{gsd/get-shit-done → src/pilot}/templates/UAT.md +3 -3
  192. package/src/pilot/templates/UI-SPEC.md +100 -0
  193. package/{gsd/get-shit-done → src/pilot}/templates/VALIDATION.md +1 -1
  194. package/src/pilot/templates/claude-md.md +122 -0
  195. package/{gsd/get-shit-done → src/pilot}/templates/codebase/architecture.md +2 -2
  196. package/{gsd/get-shit-done → src/pilot}/templates/codebase/structure.md +13 -13
  197. package/{gsd/get-shit-done → src/pilot}/templates/context.md +4 -4
  198. package/src/pilot/templates/copilot-instructions.md +7 -0
  199. package/{gsd/get-shit-done → src/pilot}/templates/debug-subagent-prompt.md +4 -4
  200. package/src/pilot/templates/dev-preferences.md +21 -0
  201. package/{gsd/get-shit-done → src/pilot}/templates/discovery.md +2 -2
  202. package/src/pilot/templates/discussion-log.md +63 -0
  203. package/{gsd/get-shit-done → src/pilot}/templates/phase-prompt.md +12 -12
  204. package/{gsd/get-shit-done → src/pilot}/templates/planner-subagent-prompt.md +7 -7
  205. package/{gsd/get-shit-done → src/pilot}/templates/project.md +1 -1
  206. package/{gsd/get-shit-done → src/pilot}/templates/research.md +2 -2
  207. package/{gsd/get-shit-done → src/pilot}/templates/state.md +2 -2
  208. package/{gsd/get-shit-done → src/pilot}/templates/summary-complex.md +1 -1
  209. package/{gsd/get-shit-done → src/pilot}/workflows/add-phase.md +11 -11
  210. package/{gsd/get-shit-done → src/pilot}/workflows/add-tests.md +15 -15
  211. package/{gsd/get-shit-done → src/pilot}/workflows/add-todo.md +7 -7
  212. package/{gsd/get-shit-done → src/pilot}/workflows/audit-milestone.md +24 -16
  213. package/src/pilot/workflows/audit-uat.md +109 -0
  214. package/src/pilot/workflows/autonomous.md +891 -0
  215. package/{gsd/get-shit-done → src/pilot}/workflows/check-todos.md +10 -10
  216. package/{gsd/get-shit-done → src/pilot}/workflows/cleanup.md +3 -3
  217. package/{gsd/get-shit-done → src/pilot}/workflows/complete-milestone.md +19 -16
  218. package/{gsd/get-shit-done → src/pilot}/workflows/diagnose-issues.md +9 -4
  219. package/{gsd/get-shit-done → src/pilot}/workflows/discovery-phase.md +8 -8
  220. package/src/pilot/workflows/discuss-phase-assumptions.md +653 -0
  221. package/{gsd/get-shit-done → src/pilot}/workflows/discuss-phase.md +407 -49
  222. package/src/pilot/workflows/do.md +104 -0
  223. package/src/pilot/workflows/execute-phase.md +821 -0
  224. package/{gsd/get-shit-done → src/pilot}/workflows/execute-plan.md +79 -28
  225. package/src/pilot/workflows/fast.md +105 -0
  226. package/src/pilot/workflows/forensics.md +265 -0
  227. package/{gsd/get-shit-done → src/pilot}/workflows/health.md +34 -11
  228. package/src/pilot/workflows/help.md +775 -0
  229. package/{gsd/get-shit-done → src/pilot}/workflows/insert-phase.md +10 -10
  230. package/{gsd/get-shit-done → src/pilot}/workflows/list-phase-assumptions.md +4 -4
  231. package/src/pilot/workflows/manager.md +362 -0
  232. package/{gsd/get-shit-done → src/pilot}/workflows/map-codebase.md +27 -17
  233. package/src/pilot/workflows/milestone-summary.md +223 -0
  234. package/{gsd/get-shit-done → src/pilot}/workflows/new-milestone.md +135 -33
  235. package/{gsd/get-shit-done → src/pilot}/workflows/new-project.md +152 -79
  236. package/src/pilot/workflows/next.md +97 -0
  237. package/src/pilot/workflows/node-repair.md +92 -0
  238. package/src/pilot/workflows/note.md +156 -0
  239. package/src/pilot/workflows/pause-work.md +177 -0
  240. package/{gsd/get-shit-done → src/pilot}/workflows/plan-milestone-gaps.md +10 -11
  241. package/src/pilot/workflows/plan-phase.md +859 -0
  242. package/src/pilot/workflows/plant-seed.md +169 -0
  243. package/src/pilot/workflows/pr-branch.md +129 -0
  244. package/{gsd/get-shit-done → src/pilot}/workflows/progress.md +95 -34
  245. package/{gsd/get-shit-done → src/pilot}/workflows/quick.md +33 -21
  246. package/{gsd/get-shit-done → src/pilot}/workflows/remove-phase.md +14 -14
  247. package/{gsd/get-shit-done → src/pilot}/workflows/research-phase.md +18 -10
  248. package/{gsd/get-shit-done → src/pilot}/workflows/resume-project.md +37 -18
  249. package/src/pilot/workflows/session-report.md +146 -0
  250. package/{gsd/get-shit-done → src/pilot}/workflows/set-profile.md +7 -7
  251. package/{gsd/get-shit-done → src/pilot}/workflows/settings.md +75 -22
  252. package/src/pilot/workflows/ship.md +228 -0
  253. package/src/pilot/workflows/stats.md +60 -0
  254. package/{gsd/get-shit-done → src/pilot}/workflows/transition.md +57 -17
  255. package/src/pilot/workflows/ui-phase.md +302 -0
  256. package/src/pilot/workflows/ui-review.md +165 -0
  257. package/{gsd/get-shit-done → src/pilot}/workflows/update.md +88 -58
  258. package/{gsd/get-shit-done → src/pilot}/workflows/validate-phase.md +24 -17
  259. package/{gsd/get-shit-done → src/pilot}/workflows/verify-phase.md +26 -15
  260. package/{gsd/get-shit-done → src/pilot}/workflows/verify-work.md +89 -37
  261. package/{ecc → src}/rules/common/agents.md +1 -0
  262. package/{ecc → src}/rules/common/coding-style.md +21 -0
  263. package/src/skills/agentic-engineering/SKILL.md +63 -0
  264. package/src/skills/ai-first-engineering/SKILL.md +51 -0
  265. package/src/skills/ai-regression-testing/SKILL.md +385 -0
  266. package/src/skills/api-design/SKILL.md +523 -0
  267. package/src/skills/architecture-decision-records/SKILL.md +179 -0
  268. package/src/skills/backend-patterns/SKILL.md +598 -0
  269. package/src/skills/benchmark/SKILL.md +87 -0
  270. package/src/skills/blueprint/SKILL.md +90 -0
  271. package/src/skills/browser-qa/SKILL.md +81 -0
  272. package/src/skills/claude-api/SKILL.md +337 -0
  273. package/src/skills/codebase-onboarding/SKILL.md +233 -0
  274. package/src/skills/coding-standards/SKILL.md +530 -0
  275. package/src/skills/context-budget/SKILL.md +135 -0
  276. package/{ecc → src}/skills/continuous-learning-v2/SKILL.md +2 -2
  277. package/{ecc → src}/skills/continuous-learning-v2/agents/observer-loop.sh +1 -1
  278. package/src/skills/cpp-coding-standards/SKILL.md +723 -0
  279. package/src/skills/cpp-testing/SKILL.md +324 -0
  280. package/src/skills/database-migrations/SKILL.md +429 -0
  281. package/src/skills/deep-research/SKILL.md +155 -0
  282. package/src/skills/deployment-patterns/SKILL.md +427 -0
  283. package/src/skills/django-patterns/SKILL.md +734 -0
  284. package/src/skills/django-security/SKILL.md +593 -0
  285. package/src/skills/django-tdd/SKILL.md +729 -0
  286. package/src/skills/django-verification/SKILL.md +469 -0
  287. package/src/skills/docker-patterns/SKILL.md +364 -0
  288. package/src/skills/documentation-lookup/SKILL.md +90 -0
  289. package/src/skills/e2e-testing/SKILL.md +326 -0
  290. package/src/skills/exa-search/SKILL.md +103 -0
  291. package/src/skills/frontend-patterns/SKILL.md +642 -0
  292. package/src/skills/golang-patterns/SKILL.md +674 -0
  293. package/src/skills/golang-testing/SKILL.md +720 -0
  294. package/src/skills/java-coding-standards/SKILL.md +147 -0
  295. package/src/skills/jpa-patterns/SKILL.md +151 -0
  296. package/src/skills/kotlin-coroutines-flows/SKILL.md +284 -0
  297. package/src/skills/kotlin-exposed-patterns/SKILL.md +719 -0
  298. package/src/skills/kotlin-ktor-patterns/SKILL.md +689 -0
  299. package/src/skills/kotlin-patterns/SKILL.md +711 -0
  300. package/src/skills/kotlin-testing/SKILL.md +824 -0
  301. package/src/skills/laravel-patterns/SKILL.md +415 -0
  302. package/src/skills/laravel-security/SKILL.md +285 -0
  303. package/src/skills/laravel-tdd/SKILL.md +283 -0
  304. package/src/skills/laravel-verification/SKILL.md +179 -0
  305. package/src/skills/mcp-server-patterns/SKILL.md +67 -0
  306. package/src/skills/perl-patterns/SKILL.md +504 -0
  307. package/src/skills/perl-testing/SKILL.md +475 -0
  308. package/src/skills/postgres-patterns/SKILL.md +147 -0
  309. package/src/skills/prompt-optimizer/SKILL.md +397 -0
  310. package/src/skills/python-patterns/SKILL.md +750 -0
  311. package/src/skills/python-testing/SKILL.md +816 -0
  312. package/src/skills/rust-patterns/SKILL.md +499 -0
  313. package/src/skills/rust-testing/SKILL.md +500 -0
  314. package/src/skills/safety-guard/SKILL.md +69 -0
  315. package/src/skills/search-first/SKILL.md +161 -0
  316. package/src/skills/security-review/SKILL.md +495 -0
  317. package/src/skills/security-review/cloud-infrastructure-security.md +361 -0
  318. package/src/skills/security-scan/SKILL.md +165 -0
  319. package/src/skills/springboot-patterns/SKILL.md +314 -0
  320. package/src/skills/springboot-security/SKILL.md +272 -0
  321. package/src/skills/springboot-tdd/SKILL.md +158 -0
  322. package/src/skills/springboot-verification/SKILL.md +231 -0
  323. package/src/skills/tdd-workflow/SKILL.md +410 -0
  324. package/ecc/scripts/hooks/session-end-marker.js +0 -15
  325. package/gsd/LICENSE +0 -21
  326. package/gsd/commands-gsd/discuss-phase.md +0 -90
  327. package/gsd/commands-gsd/execute-phase.md +0 -41
  328. package/gsd/commands-gsd/join-discord.md +0 -18
  329. package/gsd/commands-gsd/reapply-patches.md +0 -123
  330. package/gsd/commands-gsd/set-profile.md +0 -34
  331. package/gsd/get-shit-done/bin/lib/config.cjs +0 -169
  332. package/gsd/get-shit-done/bin/lib/core.cjs +0 -492
  333. package/gsd/get-shit-done/bin/lib/init.cjs +0 -710
  334. package/gsd/get-shit-done/workflows/execute-phase.md +0 -459
  335. package/gsd/get-shit-done/workflows/help.md +0 -489
  336. package/gsd/get-shit-done/workflows/pause-work.md +0 -122
  337. package/gsd/get-shit-done/workflows/plan-phase.md +0 -560
  338. package/gsd/hooks/gsd-check-update.js +0 -81
  339. package/kit/CLAUDE.md +0 -43
  340. package/kit/commands/kit/update.md +0 -46
  341. package/kit/mcp.json +0 -10
  342. package/kit/rules/code-style.md +0 -24
  343. /package/{ecc → src}/agents/architect.md +0 -0
  344. /package/{ecc → src}/agents/code-reviewer.md +0 -0
  345. /package/{ecc → src}/agents/doc-updater.md +0 -0
  346. /package/{ecc → src}/agents/e2e-runner.md +0 -0
  347. /package/{ecc → src}/agents/security-reviewer.md +0 -0
  348. /package/{ecc → src}/agents/tdd-guide.md +0 -0
  349. /package/{ecc/rules → src/available-rules}/golang/coding-style.md +0 -0
  350. /package/{ecc/rules → src/available-rules}/golang/hooks.md +0 -0
  351. /package/{ecc/rules → src/available-rules}/golang/patterns.md +0 -0
  352. /package/{ecc/rules → src/available-rules}/golang/security.md +0 -0
  353. /package/{ecc/rules → src/available-rules}/golang/testing.md +0 -0
  354. /package/{ecc/rules → src/available-rules}/kotlin/coding-style.md +0 -0
  355. /package/{ecc/rules → src/available-rules}/kotlin/patterns.md +0 -0
  356. /package/{ecc/rules → src/available-rules}/kotlin/security.md +0 -0
  357. /package/{ecc/rules → src/available-rules}/kotlin/testing.md +0 -0
  358. /package/{ecc/rules → src/available-rules}/perl/coding-style.md +0 -0
  359. /package/{ecc/rules → src/available-rules}/perl/hooks.md +0 -0
  360. /package/{ecc/rules → src/available-rules}/perl/patterns.md +0 -0
  361. /package/{ecc/rules → src/available-rules}/perl/security.md +0 -0
  362. /package/{ecc/rules → src/available-rules}/perl/testing.md +0 -0
  363. /package/{ecc/rules → src/available-rules}/php/coding-style.md +0 -0
  364. /package/{ecc/rules → src/available-rules}/php/hooks.md +0 -0
  365. /package/{ecc/rules → src/available-rules}/php/patterns.md +0 -0
  366. /package/{ecc/rules → src/available-rules}/php/security.md +0 -0
  367. /package/{ecc/rules → src/available-rules}/php/testing.md +0 -0
  368. /package/{ecc/rules → src/available-rules}/python/coding-style.md +0 -0
  369. /package/{ecc/rules → src/available-rules}/python/hooks.md +0 -0
  370. /package/{ecc/rules → src/available-rules}/python/patterns.md +0 -0
  371. /package/{ecc/rules → src/available-rules}/python/security.md +0 -0
  372. /package/{ecc/rules → src/available-rules}/python/testing.md +0 -0
  373. /package/{ecc/rules → src/available-rules}/swift/coding-style.md +0 -0
  374. /package/{ecc/rules → src/available-rules}/swift/hooks.md +0 -0
  375. /package/{ecc/rules → src/available-rules}/swift/patterns.md +0 -0
  376. /package/{ecc/rules → src/available-rules}/swift/security.md +0 -0
  377. /package/{ecc/rules → src/available-rules}/swift/testing.md +0 -0
  378. /package/{ecc/rules → src/available-rules}/typescript/coding-style.md +0 -0
  379. /package/{ecc/rules → src/available-rules}/typescript/hooks.md +0 -0
  380. /package/{ecc/rules → src/available-rules}/typescript/patterns.md +0 -0
  381. /package/{ecc/rules → src/available-rules}/typescript/security.md +0 -0
  382. /package/{ecc/rules → src/available-rules}/typescript/testing.md +0 -0
  383. /package/{ecc → src}/commands/checkpoint.md +0 -0
  384. /package/{ecc → src}/commands/learn.md +0 -0
  385. /package/{ecc → src}/commands/model-route.md +0 -0
  386. /package/{ecc → src}/commands/resume-session.md +0 -0
  387. /package/{ecc → src}/commands/save-session.md +0 -0
  388. /package/{kit → src}/commands/setup-refresh.md +0 -0
  389. /package/{kit → src}/commands/tool-guide.md +0 -0
  390. /package/{ecc → src}/commands/verify.md +0 -0
  391. /package/{ecc → src}/contexts/dev.md +0 -0
  392. /package/{ecc → src}/contexts/research.md +0 -0
  393. /package/{ecc → src}/contexts/review.md +0 -0
  394. /package/{ecc → src}/examples/CLAUDE.md +0 -0
  395. /package/{ecc → src}/examples/django-api-CLAUDE.md +0 -0
  396. /package/{ecc → src}/examples/go-microservice-CLAUDE.md +0 -0
  397. /package/{ecc → src}/examples/rust-api-CLAUDE.md +0 -0
  398. /package/{ecc → src}/examples/saas-nextjs-CLAUDE.md +0 -0
  399. /package/{ecc → src}/examples/user-CLAUDE.md +0 -0
  400. /package/{ecc/scripts → src}/hooks/check-hook-enabled.js +0 -0
  401. /package/{ecc/scripts → src}/hooks/evaluate-session.js +0 -0
  402. /package/{ecc/scripts → src}/hooks/pre-compact.js +0 -0
  403. /package/{ecc/scripts → src}/hooks/suggest-compact.js +0 -0
  404. /package/{ecc/scripts → src}/lib/package-manager.d.ts +0 -0
  405. /package/{ecc/scripts → src}/lib/package-manager.js +0 -0
  406. /package/{ecc/scripts → src}/lib/resolve-formatter.js +0 -0
  407. /package/{ecc/scripts → src}/lib/session-aliases.d.ts +0 -0
  408. /package/{ecc/scripts → src}/lib/session-aliases.js +0 -0
  409. /package/{ecc/scripts → src}/lib/shell-split.js +0 -0
  410. /package/{gsd/get-shit-done → src/pilot}/bin/lib/frontmatter.cjs +0 -0
  411. /package/{gsd/get-shit-done → src/pilot}/bin/lib/milestone.cjs +0 -0
  412. /package/{gsd/get-shit-done → src/pilot}/bin/lib/roadmap.cjs +0 -0
  413. /package/{gsd/get-shit-done → src/pilot}/bin/lib/template.cjs +0 -0
  414. /package/{gsd/get-shit-done → src/pilot}/references/checkpoints.md +0 -0
  415. /package/{gsd/get-shit-done → src/pilot}/references/questioning.md +0 -0
  416. /package/{gsd/get-shit-done → src/pilot}/references/tdd.md +0 -0
  417. /package/{gsd/get-shit-done → src/pilot}/templates/codebase/concerns.md +0 -0
  418. /package/{gsd/get-shit-done → src/pilot}/templates/codebase/conventions.md +0 -0
  419. /package/{gsd/get-shit-done → src/pilot}/templates/codebase/integrations.md +0 -0
  420. /package/{gsd/get-shit-done → src/pilot}/templates/codebase/stack.md +0 -0
  421. /package/{gsd/get-shit-done → src/pilot}/templates/codebase/testing.md +0 -0
  422. /package/{gsd/get-shit-done → src/pilot}/templates/config.json +0 -0
  423. /package/{gsd/get-shit-done → src/pilot}/templates/continue-here.md +0 -0
  424. /package/{gsd/get-shit-done → src/pilot}/templates/milestone-archive.md +0 -0
  425. /package/{gsd/get-shit-done → src/pilot}/templates/milestone.md +0 -0
  426. /package/{gsd/get-shit-done → src/pilot}/templates/requirements.md +0 -0
  427. /package/{gsd/get-shit-done → src/pilot}/templates/research-project/ARCHITECTURE.md +0 -0
  428. /package/{gsd/get-shit-done → src/pilot}/templates/research-project/FEATURES.md +0 -0
  429. /package/{gsd/get-shit-done → src/pilot}/templates/research-project/PITFALLS.md +0 -0
  430. /package/{gsd/get-shit-done → src/pilot}/templates/research-project/STACK.md +0 -0
  431. /package/{gsd/get-shit-done → src/pilot}/templates/research-project/SUMMARY.md +0 -0
  432. /package/{gsd/get-shit-done → src/pilot}/templates/retrospective.md +0 -0
  433. /package/{gsd/get-shit-done → src/pilot}/templates/roadmap.md +0 -0
  434. /package/{gsd/get-shit-done → src/pilot}/templates/summary-minimal.md +0 -0
  435. /package/{gsd/get-shit-done → src/pilot}/templates/summary-standard.md +0 -0
  436. /package/{gsd/get-shit-done → src/pilot}/templates/summary.md +0 -0
  437. /package/{gsd/get-shit-done → src/pilot}/templates/user-setup.md +0 -0
  438. /package/{gsd/get-shit-done → src/pilot}/templates/verification-report.md +0 -0
  439. /package/{ecc → src}/rules/common/development-workflow.md +0 -0
  440. /package/{ecc → src}/rules/common/git-workflow.md +0 -0
  441. /package/{ecc → src}/rules/common/hooks.md +0 -0
  442. /package/{ecc → src}/rules/common/patterns.md +0 -0
  443. /package/{ecc → src}/rules/common/performance.md +0 -0
  444. /package/{ecc → src}/rules/common/security.md +0 -0
  445. /package/{ecc → src}/rules/common/testing.md +0 -0
  446. /package/{ecc → src}/skills/continuous-learning-v2/agents/observer.md +0 -0
  447. /package/{ecc → src}/skills/continuous-learning-v2/agents/start-observer.sh +0 -0
  448. /package/{ecc → src}/skills/continuous-learning-v2/config.json +0 -0
  449. /package/{ecc → src}/skills/continuous-learning-v2/hooks/observe.sh +0 -0
  450. /package/{ecc → src}/skills/continuous-learning-v2/scripts/detect-project.sh +0 -0
  451. /package/{ecc → src}/skills/continuous-learning-v2/scripts/instinct-cli.py +0 -0
  452. /package/{ecc → src}/skills/continuous-learning-v2/scripts/test_parse_instinct.py +0 -0
  453. /package/{ecc → src}/skills/strategic-compact/SKILL.md +0 -0
  454. /package/{ecc → src}/skills/strategic-compact/suggest-compact.sh +0 -0
  455. /package/{ecc/skills/verification-loop-SKILL.md → src/skills/verification-loop/SKILL.md} +0 -0
@@ -0,0 +1,620 @@
1
+ #!/usr/bin/env node
2
+ // Registration: auto (PreToolUse on mcp_*, PostToolUseFailure on mcp_*)
3
+ 'use strict';
4
+
5
+ /**
6
+ * MCP health-check hook.
7
+ *
8
+ * Compatible with Claude Code's existing hook events:
9
+ * - PreToolUse: probe MCP server health before MCP tool execution
10
+ * - PostToolUseFailure: mark unhealthy servers, attempt reconnect, and re-probe
11
+ *
12
+ * The hook persists health state outside the conversation context so it
13
+ * survives compaction and later turns.
14
+ */
15
+
16
+ const fs = require('fs');
17
+ const os = require('os');
18
+ const path = require('path');
19
+ const http = require('http');
20
+ const https = require('https');
21
+ const { spawn, spawnSync } = require('child_process');
22
+
23
+ const MAX_STDIN = 1024 * 1024;
24
+ const DEFAULT_TTL_MS = 2 * 60 * 1000;
25
+ const DEFAULT_TIMEOUT_MS = 5000;
26
+ const DEFAULT_BACKOFF_MS = 30 * 1000;
27
+ const MAX_BACKOFF_MS = 10 * 60 * 1000;
28
+ const HEALTHY_HTTP_CODES = new Set([200, 201, 202, 204, 301, 302, 303, 304, 307, 308, 405]);
29
+ const RECONNECT_STATUS_CODES = new Set([401, 403, 429, 503]);
30
+ const FAILURE_PATTERNS = [
31
+ { code: 401, pattern: /\b401\b|unauthori[sz]ed|auth(?:entication)?\s+(?:failed|expired|invalid)/i },
32
+ { code: 403, pattern: /\b403\b|forbidden|permission denied/i },
33
+ { code: 429, pattern: /\b429\b|rate limit|too many requests/i },
34
+ { code: 503, pattern: /\b503\b|service unavailable|overloaded|temporarily unavailable/i },
35
+ { code: 'transport', pattern: /ECONNREFUSED|ENOTFOUND|EAI_AGAIN|timed? out|socket hang up|connection (?:failed|lost|reset|closed)/i }
36
+ ];
37
+
38
+ function envNumber(name, fallback) {
39
+ const value = Number(process.env[name]);
40
+ return Number.isFinite(value) && value >= 0 ? value : fallback;
41
+ }
42
+
43
+ function stateFilePath() {
44
+ if (process.env.ECC_MCP_HEALTH_STATE_PATH) {
45
+ return path.resolve(process.env.ECC_MCP_HEALTH_STATE_PATH);
46
+ }
47
+ return path.join(os.homedir(), '.claude', 'mcp-health-cache.json');
48
+ }
49
+
50
+ function configPaths() {
51
+ if (process.env.ECC_MCP_CONFIG_PATH) {
52
+ return process.env.ECC_MCP_CONFIG_PATH
53
+ .split(path.delimiter)
54
+ .map(entry => entry.trim())
55
+ .filter(Boolean)
56
+ .map(entry => path.resolve(entry));
57
+ }
58
+
59
+ const cwd = process.cwd();
60
+ const home = os.homedir();
61
+
62
+ return [
63
+ path.join(cwd, '.claude.json'),
64
+ path.join(cwd, '.claude', 'settings.json'),
65
+ path.join(home, '.claude.json'),
66
+ path.join(home, '.claude', 'settings.json')
67
+ ];
68
+ }
69
+
70
+ function readJsonFile(filePath) {
71
+ try {
72
+ return JSON.parse(fs.readFileSync(filePath, 'utf8'));
73
+ } catch {
74
+ return null;
75
+ }
76
+ }
77
+
78
+ function loadState(filePath) {
79
+ const state = readJsonFile(filePath);
80
+ if (!state || typeof state !== 'object' || Array.isArray(state)) {
81
+ return { version: 1, servers: {} };
82
+ }
83
+
84
+ if (!state.servers || typeof state.servers !== 'object' || Array.isArray(state.servers)) {
85
+ state.servers = {};
86
+ }
87
+
88
+ return state;
89
+ }
90
+
91
+ function saveState(filePath, state) {
92
+ try {
93
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
94
+ fs.writeFileSync(filePath, JSON.stringify(state, null, 2));
95
+ } catch {
96
+ // Never block the hook on state persistence errors.
97
+ }
98
+ }
99
+
100
+ function readRawStdin() {
101
+ return new Promise(resolve => {
102
+ let raw = '';
103
+ let truncated = /^(1|true|yes)$/i.test(String(process.env.CCP_HOOK_INPUT_TRUNCATED || ''));
104
+ process.stdin.setEncoding('utf8');
105
+ process.stdin.on('data', chunk => {
106
+ if (raw.length < MAX_STDIN) {
107
+ const remaining = MAX_STDIN - raw.length;
108
+ raw += chunk.substring(0, remaining);
109
+ if (chunk.length > remaining) {
110
+ truncated = true;
111
+ }
112
+ } else {
113
+ truncated = true;
114
+ }
115
+ });
116
+ process.stdin.on('end', () => resolve({ raw, truncated }));
117
+ process.stdin.on('error', () => resolve({ raw, truncated }));
118
+ });
119
+ }
120
+
121
+ function safeParse(raw) {
122
+ try {
123
+ return raw.trim() ? JSON.parse(raw) : {};
124
+ } catch {
125
+ return {};
126
+ }
127
+ }
128
+
129
+ function extractMcpTarget(input) {
130
+ const toolName = String(input.tool_name || input.name || '');
131
+ const explicitServer = input.server
132
+ || input.mcp_server
133
+ || input.tool_input?.server
134
+ || input.tool_input?.mcp_server
135
+ || input.tool_input?.connector
136
+ || null;
137
+ const explicitTool = input.tool
138
+ || input.mcp_tool
139
+ || input.tool_input?.tool
140
+ || input.tool_input?.mcp_tool
141
+ || null;
142
+
143
+ if (explicitServer) {
144
+ return {
145
+ server: String(explicitServer),
146
+ tool: explicitTool ? String(explicitTool) : toolName
147
+ };
148
+ }
149
+
150
+ if (!toolName.startsWith('mcp__')) {
151
+ return null;
152
+ }
153
+
154
+ const segments = toolName.slice(5).split('__');
155
+ if (segments.length < 2 || !segments[0]) {
156
+ return null;
157
+ }
158
+
159
+ return {
160
+ server: segments[0],
161
+ tool: segments.slice(1).join('__')
162
+ };
163
+ }
164
+
165
+ function extractMcpTargetFromRaw(raw) {
166
+ const toolNameMatch = raw.match(/"(?:tool_name|name)"\s*:\s*"([^"]+)"/);
167
+ const serverMatch = raw.match(/"(?:server|mcp_server|connector)"\s*:\s*"([^"]+)"/);
168
+ const toolMatch = raw.match(/"(?:tool|mcp_tool)"\s*:\s*"([^"]+)"/);
169
+
170
+ return extractMcpTarget({
171
+ tool_name: toolNameMatch ? toolNameMatch[1] : '',
172
+ server: serverMatch ? serverMatch[1] : undefined,
173
+ tool: toolMatch ? toolMatch[1] : undefined
174
+ });
175
+ }
176
+
177
+ function resolveServerConfig(serverName) {
178
+ for (const filePath of configPaths()) {
179
+ const data = readJsonFile(filePath);
180
+ const server = data?.mcpServers?.[serverName]
181
+ || data?.mcp_servers?.[serverName]
182
+ || null;
183
+
184
+ if (server && typeof server === 'object' && !Array.isArray(server)) {
185
+ return {
186
+ config: server,
187
+ source: filePath
188
+ };
189
+ }
190
+ }
191
+
192
+ return null;
193
+ }
194
+
195
+ function markHealthy(state, serverName, now, details = {}) {
196
+ state.servers[serverName] = {
197
+ status: 'healthy',
198
+ checkedAt: now,
199
+ expiresAt: now + envNumber('ECC_MCP_HEALTH_TTL_MS', DEFAULT_TTL_MS),
200
+ failureCount: 0,
201
+ lastError: null,
202
+ lastFailureCode: null,
203
+ nextRetryAt: now,
204
+ lastRestoredAt: now,
205
+ ...details
206
+ };
207
+ }
208
+
209
+ function markUnhealthy(state, serverName, now, failureCode, errorMessage) {
210
+ const previous = state.servers[serverName] || {};
211
+ const failureCount = Number(previous.failureCount || 0) + 1;
212
+ const backoffBase = envNumber('ECC_MCP_HEALTH_BACKOFF_MS', DEFAULT_BACKOFF_MS);
213
+ const nextRetryDelay = Math.min(backoffBase * (2 ** Math.max(failureCount - 1, 0)), MAX_BACKOFF_MS);
214
+
215
+ state.servers[serverName] = {
216
+ status: 'unhealthy',
217
+ checkedAt: now,
218
+ expiresAt: now,
219
+ failureCount,
220
+ lastError: errorMessage || null,
221
+ lastFailureCode: failureCode || null,
222
+ nextRetryAt: now + nextRetryDelay,
223
+ lastRestoredAt: previous.lastRestoredAt || null
224
+ };
225
+ }
226
+
227
+ function failureSummary(input) {
228
+ const output = input.tool_output;
229
+ const pieces = [
230
+ typeof input.error === 'string' ? input.error : '',
231
+ typeof input.message === 'string' ? input.message : '',
232
+ typeof input.tool_response === 'string' ? input.tool_response : '',
233
+ typeof output === 'string' ? output : '',
234
+ typeof output?.output === 'string' ? output.output : '',
235
+ typeof output?.stderr === 'string' ? output.stderr : '',
236
+ typeof input.tool_input?.error === 'string' ? input.tool_input.error : ''
237
+ ].filter(Boolean);
238
+
239
+ return pieces.join('\n');
240
+ }
241
+
242
+ function detectFailureCode(text) {
243
+ const summary = String(text || '');
244
+ for (const entry of FAILURE_PATTERNS) {
245
+ if (entry.pattern.test(summary)) {
246
+ return entry.code;
247
+ }
248
+ }
249
+ return null;
250
+ }
251
+
252
+ function requestHttp(urlString, headers, timeoutMs) {
253
+ return new Promise(resolve => {
254
+ let settled = false;
255
+ let timedOut = false;
256
+
257
+ const url = new URL(urlString);
258
+ const client = url.protocol === 'https:' ? https : http;
259
+
260
+ const req = client.request(
261
+ url,
262
+ {
263
+ method: 'GET',
264
+ headers,
265
+ },
266
+ res => {
267
+ if (settled) return;
268
+ settled = true;
269
+ res.resume();
270
+ resolve({
271
+ ok: HEALTHY_HTTP_CODES.has(res.statusCode),
272
+ statusCode: res.statusCode,
273
+ reason: `HTTP ${res.statusCode}`
274
+ });
275
+ }
276
+ );
277
+
278
+ req.setTimeout(timeoutMs, () => {
279
+ timedOut = true;
280
+ req.destroy(new Error('timeout'));
281
+ });
282
+
283
+ req.on('error', error => {
284
+ if (settled) return;
285
+ settled = true;
286
+ resolve({
287
+ ok: false,
288
+ statusCode: null,
289
+ reason: timedOut ? 'request timed out' : error.message
290
+ });
291
+ });
292
+
293
+ req.end();
294
+ });
295
+ }
296
+
297
+ function probeCommandServer(serverName, config) {
298
+ return new Promise(resolve => {
299
+ const command = config.command;
300
+ const args = Array.isArray(config.args) ? config.args.map(arg => String(arg)) : [];
301
+ const timeoutMs = envNumber('ECC_MCP_HEALTH_TIMEOUT_MS', DEFAULT_TIMEOUT_MS);
302
+ const mergedEnv = {
303
+ ...process.env,
304
+ ...(config.env && typeof config.env === 'object' && !Array.isArray(config.env) ? config.env : {})
305
+ };
306
+
307
+ let stderr = '';
308
+ let done = false;
309
+
310
+ function finish(result) {
311
+ if (done) return;
312
+ done = true;
313
+ resolve(result);
314
+ }
315
+
316
+ let child;
317
+ try {
318
+ child = spawn(command, args, {
319
+ env: mergedEnv,
320
+ cwd: process.cwd(),
321
+ stdio: ['pipe', 'ignore', 'pipe']
322
+ });
323
+ } catch (error) {
324
+ finish({
325
+ ok: false,
326
+ statusCode: null,
327
+ reason: error.message
328
+ });
329
+ return;
330
+ }
331
+
332
+ child.stderr.on('data', chunk => {
333
+ if (stderr.length < 4000) {
334
+ const remaining = 4000 - stderr.length;
335
+ stderr += String(chunk).slice(0, remaining);
336
+ }
337
+ });
338
+
339
+ child.on('error', error => {
340
+ finish({
341
+ ok: false,
342
+ statusCode: null,
343
+ reason: error.message
344
+ });
345
+ });
346
+
347
+ child.on('exit', (code, signal) => {
348
+ finish({
349
+ ok: false,
350
+ statusCode: code,
351
+ reason: stderr.trim() || `process exited before handshake (${signal || code || 'unknown'})`
352
+ });
353
+ });
354
+
355
+ const timer = setTimeout(() => {
356
+ try {
357
+ child.kill('SIGTERM');
358
+ } catch {
359
+ // ignore
360
+ }
361
+
362
+ setTimeout(() => {
363
+ try {
364
+ child.kill('SIGKILL');
365
+ } catch {
366
+ // ignore
367
+ }
368
+ }, 200).unref?.();
369
+
370
+ finish({
371
+ ok: true,
372
+ statusCode: null,
373
+ reason: `${serverName} accepted a new stdio process`
374
+ });
375
+ }, timeoutMs);
376
+
377
+ if (typeof timer.unref === 'function') {
378
+ timer.unref();
379
+ }
380
+ });
381
+ }
382
+
383
+ async function probeServer(serverName, resolvedConfig) {
384
+ const config = resolvedConfig.config;
385
+
386
+ if (config.type === 'http' || config.url) {
387
+ const result = await requestHttp(config.url, config.headers || {}, envNumber('ECC_MCP_HEALTH_TIMEOUT_MS', DEFAULT_TIMEOUT_MS));
388
+
389
+ return {
390
+ ok: result.ok,
391
+ failureCode: RECONNECT_STATUS_CODES.has(result.statusCode) ? result.statusCode : null,
392
+ reason: result.reason,
393
+ source: resolvedConfig.source
394
+ };
395
+ }
396
+
397
+ if (config.command) {
398
+ const result = await probeCommandServer(serverName, config);
399
+
400
+ return {
401
+ ok: result.ok,
402
+ failureCode: RECONNECT_STATUS_CODES.has(result.statusCode) ? result.statusCode : null,
403
+ reason: result.reason,
404
+ source: resolvedConfig.source
405
+ };
406
+ }
407
+
408
+ return {
409
+ ok: false,
410
+ failureCode: null,
411
+ reason: 'unsupported MCP server config',
412
+ source: resolvedConfig.source
413
+ };
414
+ }
415
+
416
+ function reconnectCommand(serverName) {
417
+ const key = `ECC_MCP_RECONNECT_${String(serverName).toUpperCase().replace(/[^A-Z0-9]/g, '_')}`;
418
+ const command = process.env[key] || process.env.ECC_MCP_RECONNECT_COMMAND || '';
419
+ if (!command.trim()) {
420
+ return null;
421
+ }
422
+
423
+ return command.includes('{server}')
424
+ ? command.replace(/\{server\}/g, serverName)
425
+ : command;
426
+ }
427
+
428
+ function attemptReconnect(serverName) {
429
+ const command = reconnectCommand(serverName);
430
+ if (!command) {
431
+ return { attempted: false, success: false, reason: 'no reconnect command configured' };
432
+ }
433
+
434
+ const result = spawnSync(command, {
435
+ shell: true,
436
+ env: process.env,
437
+ cwd: process.cwd(),
438
+ encoding: 'utf8',
439
+ timeout: envNumber('ECC_MCP_RECONNECT_TIMEOUT_MS', DEFAULT_TIMEOUT_MS)
440
+ });
441
+
442
+ if (result.error) {
443
+ return { attempted: true, success: false, reason: result.error.message };
444
+ }
445
+
446
+ if (result.status !== 0) {
447
+ return {
448
+ attempted: true,
449
+ success: false,
450
+ reason: (result.stderr || result.stdout || `reconnect exited ${result.status}`).trim()
451
+ };
452
+ }
453
+
454
+ return { attempted: true, success: true, reason: 'reconnect command completed' };
455
+ }
456
+
457
+ function shouldFailOpen() {
458
+ return /^(1|true|yes)$/i.test(String(process.env.ECC_MCP_HEALTH_FAIL_OPEN || ''));
459
+ }
460
+
461
+ function emitLogs(logs) {
462
+ for (const line of logs) {
463
+ process.stderr.write(`${line}\n`);
464
+ }
465
+ }
466
+
467
+ async function handlePreToolUse(rawInput, input, target, statePathValue, now) {
468
+ const logs = [];
469
+ const state = loadState(statePathValue);
470
+ const previous = state.servers[target.server] || {};
471
+
472
+ if (previous.status === 'healthy' && Number(previous.expiresAt || 0) > now) {
473
+ return { rawInput, exitCode: 0, logs };
474
+ }
475
+
476
+ if (previous.status === 'unhealthy' && Number(previous.nextRetryAt || 0) > now) {
477
+ logs.push(
478
+ `[MCPHealthCheck] ${target.server} is marked unhealthy until ${new Date(previous.nextRetryAt).toISOString()}; skipping ${target.tool || 'tool'}`
479
+ );
480
+ return { rawInput, exitCode: shouldFailOpen() ? 0 : 2, logs };
481
+ }
482
+
483
+ const resolvedConfig = resolveServerConfig(target.server);
484
+ if (!resolvedConfig) {
485
+ logs.push(`[MCPHealthCheck] No MCP config found for ${target.server}; skipping preflight probe`);
486
+ return { rawInput, exitCode: 0, logs };
487
+ }
488
+
489
+ const probe = await probeServer(target.server, resolvedConfig);
490
+ if (probe.ok) {
491
+ markHealthy(state, target.server, now, { source: resolvedConfig.source });
492
+ saveState(statePathValue, state);
493
+
494
+ if (previous.status === 'unhealthy') {
495
+ logs.push(`[MCPHealthCheck] ${target.server} connection restored`);
496
+ }
497
+
498
+ return { rawInput, exitCode: 0, logs };
499
+ }
500
+
501
+ let reconnect = { attempted: false, success: false, reason: 'probe failed' };
502
+ if (probe.failureCode || previous.status === 'unhealthy') {
503
+ reconnect = attemptReconnect(target.server);
504
+ if (reconnect.success) {
505
+ const reprobe = await probeServer(target.server, resolvedConfig);
506
+ if (reprobe.ok) {
507
+ markHealthy(state, target.server, now, {
508
+ source: resolvedConfig.source,
509
+ restoredBy: 'reconnect-command'
510
+ });
511
+ saveState(statePathValue, state);
512
+ logs.push(`[MCPHealthCheck] ${target.server} connection restored after reconnect`);
513
+ return { rawInput, exitCode: 0, logs };
514
+ }
515
+ probe.reason = `${probe.reason}; reconnect reprobe failed: ${reprobe.reason}`;
516
+ }
517
+ }
518
+
519
+ markUnhealthy(state, target.server, now, probe.failureCode, probe.reason);
520
+ saveState(statePathValue, state);
521
+
522
+ const reconnectSuffix = reconnect.attempted
523
+ ? ` Reconnect attempt: ${reconnect.success ? 'ok' : reconnect.reason}.`
524
+ : '';
525
+ logs.push(
526
+ `[MCPHealthCheck] ${target.server} is unavailable (${probe.reason}). Blocking ${target.tool || 'tool'} so Claude can fall back to non-MCP tools.${reconnectSuffix}`
527
+ );
528
+
529
+ return { rawInput, exitCode: shouldFailOpen() ? 0 : 2, logs };
530
+ }
531
+
532
+ async function handlePostToolUseFailure(rawInput, input, target, statePathValue, now) {
533
+ const logs = [];
534
+ const summary = failureSummary(input);
535
+ const failureCode = detectFailureCode(summary);
536
+
537
+ if (!failureCode) {
538
+ return { rawInput, exitCode: 0, logs };
539
+ }
540
+
541
+ const state = loadState(statePathValue);
542
+ markUnhealthy(state, target.server, now, failureCode, summary.slice(0, 500));
543
+ saveState(statePathValue, state);
544
+
545
+ logs.push(`[MCPHealthCheck] ${target.server} reported ${failureCode}; marking server unhealthy and attempting reconnect`);
546
+
547
+ const reconnect = attemptReconnect(target.server);
548
+ if (!reconnect.attempted) {
549
+ logs.push(`[MCPHealthCheck] ${target.server} reconnect skipped: ${reconnect.reason}`);
550
+ return { rawInput, exitCode: 0, logs };
551
+ }
552
+
553
+ if (!reconnect.success) {
554
+ logs.push(`[MCPHealthCheck] ${target.server} reconnect failed: ${reconnect.reason}`);
555
+ return { rawInput, exitCode: 0, logs };
556
+ }
557
+
558
+ const resolvedConfig = resolveServerConfig(target.server);
559
+ if (!resolvedConfig) {
560
+ logs.push(`[MCPHealthCheck] ${target.server} reconnect completed but no config was available for a follow-up probe`);
561
+ return { rawInput, exitCode: 0, logs };
562
+ }
563
+
564
+ const reprobe = await probeServer(target.server, resolvedConfig);
565
+ if (!reprobe.ok) {
566
+ logs.push(`[MCPHealthCheck] ${target.server} reconnect command ran, but health probe still failed: ${reprobe.reason}`);
567
+ return { rawInput, exitCode: 0, logs };
568
+ }
569
+
570
+ const refreshed = loadState(statePathValue);
571
+ markHealthy(refreshed, target.server, now, {
572
+ source: resolvedConfig.source,
573
+ restoredBy: 'post-failure-reconnect'
574
+ });
575
+ saveState(statePathValue, refreshed);
576
+ logs.push(`[MCPHealthCheck] ${target.server} connection restored`);
577
+ return { rawInput, exitCode: 0, logs };
578
+ }
579
+
580
+ async function main() {
581
+ const { raw: rawInput, truncated } = await readRawStdin();
582
+ const input = safeParse(rawInput);
583
+ const target = extractMcpTarget(input) || (truncated ? extractMcpTargetFromRaw(rawInput) : null);
584
+
585
+ if (!target) {
586
+ process.stdout.write(rawInput);
587
+ process.exit(0);
588
+ return;
589
+ }
590
+
591
+ if (truncated) {
592
+ const limit = Number(process.env.CCP_HOOK_INPUT_MAX_BYTES) || MAX_STDIN;
593
+ const logs = [
594
+ shouldFailOpen()
595
+ ? `[MCPHealthCheck] Hook input exceeded ${limit} bytes while checking ${target.server}; allowing ${target.tool || 'tool'} because fail-open mode is enabled`
596
+ : `[MCPHealthCheck] Hook input exceeded ${limit} bytes while checking ${target.server}; blocking ${target.tool || 'tool'} to avoid bypassing MCP health checks`
597
+ ];
598
+ emitLogs(logs);
599
+ process.stdout.write(rawInput);
600
+ process.exit(shouldFailOpen() ? 0 : 2);
601
+ return;
602
+ }
603
+
604
+ const eventName = process.env.CLAUDE_HOOK_EVENT_NAME || 'PreToolUse';
605
+ const now = Date.now();
606
+ const statePathValue = stateFilePath();
607
+
608
+ const result = eventName === 'PostToolUseFailure'
609
+ ? await handlePostToolUseFailure(rawInput, input, target, statePathValue, now)
610
+ : await handlePreToolUse(rawInput, input, target, statePathValue, now);
611
+
612
+ emitLogs(result.logs);
613
+ process.stdout.write(result.rawInput);
614
+ process.exit(result.exitCode);
615
+ }
616
+
617
+ main().catch(error => {
618
+ process.stderr.write(`[MCPHealthCheck] Unexpected error: ${error.message}\n`);
619
+ process.exit(0);
620
+ });
@@ -5,7 +5,7 @@ HOOK_ID="${1:-}"
5
5
  REL_SCRIPT_PATH="${2:-}"
6
6
  PROFILES_CSV="${3:-standard,strict}"
7
7
  SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
8
- PLUGIN_ROOT="${CLAUDE_PLUGIN_ROOT:-$(cd "${SCRIPT_DIR}/../.." && pwd)}"
8
+ PLUGIN_ROOT="${CCP_CLAUDE_DIR:-$(cd "${SCRIPT_DIR}/../.." && pwd)}"
9
9
 
10
10
  # Preserve stdin for passthrough or script execution
11
11
  INPUT="$(cat)"