triflux 10.0.0 → 10.0.2

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 (426) hide show
  1. package/CLAUDE.md +171 -0
  2. package/README.md +32 -15
  3. package/bin/triflux.mjs +62 -5
  4. package/hooks/agent-route-guard.mjs +109 -0
  5. package/hooks/cross-review-tracker.mjs +122 -0
  6. package/hooks/error-context.mjs +148 -0
  7. package/hooks/hook-adaptive-collector.mjs +86 -0
  8. package/hooks/hook-manager.mjs +365 -0
  9. package/hooks/hook-orchestrator.mjs +312 -0
  10. package/hooks/hook-registry.json +246 -0
  11. package/hooks/hooks.json +89 -0
  12. package/hooks/keyword-rules.json +574 -0
  13. package/hooks/lib/resolve-root.mjs +59 -0
  14. package/hooks/mcp-config-watcher.mjs +80 -0
  15. package/hooks/pipeline-stop.mjs +76 -0
  16. package/hooks/safety-guard.mjs +169 -0
  17. package/hooks/subagent-verifier.mjs +80 -0
  18. package/hub/account-broker.mjs +251 -0
  19. package/hub/adaptive-diagnostic.mjs +323 -0
  20. package/hub/adaptive-inject.mjs +186 -0
  21. package/hub/adaptive-memory.mjs +163 -0
  22. package/hub/adaptive.mjs +143 -0
  23. package/hub/assign-callbacks.mjs +133 -0
  24. package/hub/bridge.mjs +799 -0
  25. package/hub/cli-adapter-base.mjs +280 -0
  26. package/hub/codex-adapter.mjs +199 -0
  27. package/hub/codex-compat.mjs +11 -0
  28. package/hub/codex-preflight.mjs +166 -0
  29. package/hub/delegator/contracts.mjs +37 -0
  30. package/hub/delegator/index.mjs +14 -0
  31. package/hub/delegator/schema/delegator-tools.schema.json +250 -0
  32. package/hub/delegator/service.mjs +307 -0
  33. package/hub/delegator/tool-definitions.mjs +35 -0
  34. package/hub/fullcycle.mjs +96 -0
  35. package/hub/gemini-adapter.mjs +180 -0
  36. package/hub/hitl.mjs +143 -0
  37. package/hub/intent.mjs +193 -0
  38. package/hub/lib/cache-guard.mjs +114 -0
  39. package/hub/lib/known-errors.json +72 -0
  40. package/hub/lib/memory-store.mjs +748 -0
  41. package/hub/lib/process-utils.mjs +361 -0
  42. package/hub/lib/ssh-command.mjs +211 -0
  43. package/hub/lib/ssh-retry.mjs +59 -0
  44. package/hub/lib/uuidv7.mjs +44 -0
  45. package/hub/memory-doctor.mjs +480 -0
  46. package/hub/middleware/request-logger.mjs +161 -0
  47. package/hub/paths.mjs +30 -0
  48. package/hub/pipe.mjs +664 -0
  49. package/hub/pipeline/gates/confidence.mjs +56 -0
  50. package/hub/pipeline/gates/consensus.mjs +94 -0
  51. package/hub/pipeline/gates/index.mjs +5 -0
  52. package/hub/pipeline/gates/selfcheck.mjs +82 -0
  53. package/hub/pipeline/index.mjs +318 -0
  54. package/hub/pipeline/state.mjs +191 -0
  55. package/hub/pipeline/transitions.mjs +124 -0
  56. package/hub/platform.mjs +225 -0
  57. package/hub/public/dashboard.html +355 -0
  58. package/hub/public/tray-icon.ico +0 -0
  59. package/hub/public/tray-icon.png +0 -0
  60. package/hub/quality/deslop.mjs +253 -0
  61. package/hub/reflexion.mjs +372 -0
  62. package/hub/research.mjs +146 -0
  63. package/hub/router.mjs +791 -0
  64. package/hub/routing/complexity.mjs +166 -0
  65. package/hub/routing/index.mjs +117 -0
  66. package/hub/routing/q-learning.mjs +336 -0
  67. package/hub/schema.sql +148 -0
  68. package/hub/server.mjs +1264 -0
  69. package/hub/session-fingerprint.mjs +352 -0
  70. package/hub/state.mjs +258 -0
  71. package/hub/store-adapter.mjs +118 -0
  72. package/hub/store.mjs +857 -0
  73. package/hub/team/agent-map.json +11 -0
  74. package/hub/team/ansi.mjs +379 -0
  75. package/hub/team/backend.mjs +90 -0
  76. package/hub/team/cli/commands/attach.mjs +37 -0
  77. package/hub/team/cli/commands/control.mjs +43 -0
  78. package/hub/team/cli/commands/debug.mjs +74 -0
  79. package/hub/team/cli/commands/focus.mjs +53 -0
  80. package/hub/team/cli/commands/interrupt.mjs +36 -0
  81. package/hub/team/cli/commands/kill.mjs +37 -0
  82. package/hub/team/cli/commands/list.mjs +24 -0
  83. package/hub/team/cli/commands/send.mjs +37 -0
  84. package/hub/team/cli/commands/start/index.mjs +106 -0
  85. package/hub/team/cli/commands/start/parse-args.mjs +130 -0
  86. package/hub/team/cli/commands/start/start-headless.mjs +109 -0
  87. package/hub/team/cli/commands/start/start-in-process.mjs +40 -0
  88. package/hub/team/cli/commands/start/start-mux.mjs +73 -0
  89. package/hub/team/cli/commands/start/start-wt.mjs +69 -0
  90. package/hub/team/cli/commands/status.mjs +87 -0
  91. package/hub/team/cli/commands/stop.mjs +31 -0
  92. package/hub/team/cli/commands/task.mjs +30 -0
  93. package/hub/team/cli/commands/tasks.mjs +13 -0
  94. package/hub/team/cli/help.mjs +42 -0
  95. package/hub/team/cli/index.mjs +41 -0
  96. package/hub/team/cli/manifest.mjs +29 -0
  97. package/hub/team/cli/render.mjs +30 -0
  98. package/hub/team/cli/services/attach-fallback.mjs +54 -0
  99. package/hub/team/cli/services/hub-client.mjs +227 -0
  100. package/hub/team/cli/services/member-selector.mjs +30 -0
  101. package/hub/team/cli/services/native-control.mjs +117 -0
  102. package/hub/team/cli/services/runtime-mode.mjs +62 -0
  103. package/hub/team/cli/services/state-store.mjs +48 -0
  104. package/hub/team/cli/services/task-model.mjs +30 -0
  105. package/hub/team/conductor-mesh-bridge.mjs +121 -0
  106. package/hub/team/conductor.mjs +671 -0
  107. package/hub/team/dashboard-anchor.mjs +14 -0
  108. package/hub/team/dashboard-layout.mjs +33 -0
  109. package/hub/team/dashboard-open.mjs +153 -0
  110. package/hub/team/dashboard.mjs +274 -0
  111. package/hub/team/event-log.mjs +76 -0
  112. package/hub/team/handoff.mjs +303 -0
  113. package/hub/team/headless.mjs +1156 -0
  114. package/hub/team/health-probe.mjs +272 -0
  115. package/hub/team/launcher-template.mjs +95 -0
  116. package/hub/team/lead-control.mjs +104 -0
  117. package/hub/team/native-supervisor.mjs +392 -0
  118. package/hub/team/native.mjs +649 -0
  119. package/hub/team/nativeProxy.mjs +688 -0
  120. package/hub/team/notify.mjs +293 -0
  121. package/hub/team/orchestrator.mjs +161 -0
  122. package/hub/team/pane.mjs +153 -0
  123. package/hub/team/process-cleanup.mjs +342 -0
  124. package/hub/team/psmux.mjs +1354 -0
  125. package/hub/team/remote-probe.mjs +276 -0
  126. package/hub/team/remote-session.mjs +299 -0
  127. package/hub/team/remote-watcher.mjs +478 -0
  128. package/hub/team/routing.mjs +223 -0
  129. package/hub/team/session-sync.mjs +169 -0
  130. package/hub/team/session.mjs +611 -0
  131. package/hub/team/shared.mjs +13 -0
  132. package/hub/team/staleState.mjs +361 -0
  133. package/hub/team/swarm-hypervisor.mjs +589 -0
  134. package/hub/team/swarm-locks.mjs +204 -0
  135. package/hub/team/swarm-planner.mjs +260 -0
  136. package/hub/team/swarm-reconciler.mjs +137 -0
  137. package/hub/team/tui-lite.mjs +380 -0
  138. package/hub/team/tui-remote-adapter.mjs +393 -0
  139. package/hub/team/tui-viewer.mjs +463 -0
  140. package/hub/team/tui.mjs +1449 -0
  141. package/hub/team/worktree-lifecycle.mjs +193 -0
  142. package/hub/team/wt-manager.mjs +407 -0
  143. package/hub/team/wt-templates.json +43 -0
  144. package/hub/team-bridge.mjs +27 -0
  145. package/hub/token-mode.mjs +224 -0
  146. package/hub/tools.mjs +636 -0
  147. package/hub/tray.mjs +376 -0
  148. package/hub/workers/claude-worker.mjs +475 -0
  149. package/hub/workers/codex-mcp.mjs +507 -0
  150. package/hub/workers/delegator-mcp.mjs +1076 -0
  151. package/hub/workers/factory.mjs +21 -0
  152. package/hub/workers/gemini-worker.mjs +374 -0
  153. package/hub/workers/interface.mjs +52 -0
  154. package/hub/workers/worker-utils.mjs +104 -0
  155. package/hud/colors.mjs +88 -0
  156. package/hud/constants.mjs +88 -0
  157. package/hud/context-monitor.mjs +403 -0
  158. package/hud/hud-qos-status.mjs +210 -0
  159. package/hud/providers/claude.mjs +314 -0
  160. package/hud/providers/codex.mjs +151 -0
  161. package/hud/providers/gemini.mjs +320 -0
  162. package/hud/renderers.mjs +442 -0
  163. package/hud/terminal.mjs +140 -0
  164. package/hud/utils.mjs +313 -0
  165. package/mesh/index.mjs +63 -0
  166. package/mesh/mesh-budget.mjs +128 -0
  167. package/mesh/mesh-heartbeat.mjs +100 -0
  168. package/mesh/mesh-protocol.mjs +96 -0
  169. package/mesh/mesh-queue.mjs +165 -0
  170. package/mesh/mesh-registry.mjs +78 -0
  171. package/mesh/mesh-router.mjs +76 -0
  172. package/package.json +8 -1
  173. package/references/hosts.json +33 -0
  174. package/scripts/__tests__/gen-skill-docs.test.mjs +87 -0
  175. package/scripts/__tests__/keyword-detector.test.mjs +234 -0
  176. package/scripts/__tests__/mcp-guard-engine.test.mjs +118 -0
  177. package/scripts/__tests__/remote-spawn-transfer.test.mjs +117 -0
  178. package/scripts/__tests__/remote-spawn.test.mjs +92 -0
  179. package/scripts/__tests__/skill-template.test.mjs +193 -0
  180. package/scripts/__tests__/smoke.test.mjs +34 -0
  181. package/scripts/cache-buildup.mjs +30 -0
  182. package/scripts/cache-doctor.mjs +149 -0
  183. package/scripts/cache-warmup.mjs +557 -0
  184. package/scripts/claudemd-sync.mjs +148 -0
  185. package/scripts/cli-route.sh +3 -0
  186. package/scripts/completions/tfx.bash +47 -0
  187. package/scripts/completions/tfx.fish +44 -0
  188. package/scripts/completions/tfx.zsh +83 -0
  189. package/scripts/cross-review-gate.mjs +126 -0
  190. package/scripts/cross-review-tracker.mjs +238 -0
  191. package/scripts/gen-skill-docs.mjs +111 -0
  192. package/scripts/headless-guard-fast.sh +21 -0
  193. package/scripts/headless-guard.mjs +360 -0
  194. package/scripts/hub-ensure.mjs +120 -0
  195. package/scripts/keyword-detector.mjs +272 -0
  196. package/scripts/keyword-rules-expander.mjs +521 -0
  197. package/scripts/lib/claudemd-scanner.mjs +218 -0
  198. package/scripts/lib/context.mjs +67 -0
  199. package/scripts/lib/cross-review-utils.mjs +51 -0
  200. package/scripts/lib/env-probe.mjs +241 -0
  201. package/scripts/lib/gemini-profiles.mjs +85 -0
  202. package/scripts/lib/handoff.mjs +171 -0
  203. package/scripts/lib/hook-utils.mjs +14 -0
  204. package/scripts/lib/keyword-rules.mjs +166 -0
  205. package/scripts/lib/logger.mjs +105 -0
  206. package/scripts/lib/mcp-filter.mjs +739 -0
  207. package/scripts/lib/mcp-guard-engine.mjs +954 -0
  208. package/scripts/lib/mcp-manifest.mjs +79 -0
  209. package/scripts/lib/mcp-server-catalog.mjs +118 -0
  210. package/scripts/lib/psmux-info.mjs +119 -0
  211. package/scripts/lib/remote-spawn-transfer.mjs +196 -0
  212. package/scripts/lib/skill-template.mjs +326 -0
  213. package/scripts/mcp-check.mjs +237 -0
  214. package/scripts/mcp-cleanup.ps1 +17 -0
  215. package/scripts/mcp-gateway-config.mjs +207 -0
  216. package/scripts/mcp-gateway-ensure.mjs +85 -0
  217. package/scripts/mcp-gateway-integration-test.mjs +228 -0
  218. package/scripts/mcp-gateway-start.mjs +226 -0
  219. package/scripts/mcp-gateway-start.ps1 +141 -0
  220. package/scripts/mcp-gateway-verify.mjs +77 -0
  221. package/scripts/mcp-safety-guard.mjs +44 -0
  222. package/scripts/notion-read.mjs +556 -0
  223. package/scripts/pack.mjs +295 -0
  224. package/scripts/preflight-cache.mjs +69 -0
  225. package/scripts/preinstall.mjs +96 -0
  226. package/scripts/remote-spawn.mjs +1376 -0
  227. package/scripts/run.cjs +79 -0
  228. package/scripts/session-spawn-helper.mjs +185 -0
  229. package/scripts/setup.mjs +1178 -0
  230. package/scripts/test-lock.mjs +71 -0
  231. package/scripts/test-tfx-route-no-claude-native.mjs +57 -0
  232. package/scripts/tfx-batch-stats.mjs +96 -0
  233. package/scripts/tfx-gate-activate.mjs +89 -0
  234. package/scripts/tfx-route-post.mjs +505 -0
  235. package/scripts/tfx-route-worker.mjs +223 -0
  236. package/scripts/tfx-route.sh +2014 -0
  237. package/scripts/tmp-cleanup.mjs +103 -0
  238. package/scripts/token-snapshot.mjs +575 -0
  239. package/skills/tfx-auto/SKILL.md.tmpl +2 -3
  240. package/skills/tfx-autoresearch/SKILL.md +6 -5
  241. package/skills/tfx-codex/SKILL.md.tmpl +2 -3
  242. package/skills/tfx-codex-swarm-workspace/iteration-1/benchmark.json +33 -0
  243. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/eval_metadata.json +42 -0
  244. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/grading.json +11 -0
  245. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/outputs/analysis.md +87 -0
  246. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/outputs/classification.md +35 -0
  247. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/outputs/commands.sh +275 -0
  248. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/outputs/routing.md +56 -0
  249. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/timing.json +5 -0
  250. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/grading.json +11 -0
  251. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/outputs/analysis.md +92 -0
  252. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/outputs/classification.md +71 -0
  253. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/outputs/commands.sh +264 -0
  254. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/outputs/routing.md +113 -0
  255. package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/timing.json +5 -0
  256. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/eval_metadata.json +32 -0
  257. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/grading.json +9 -0
  258. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/outputs/analysis.md +96 -0
  259. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/outputs/classification.md +38 -0
  260. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/outputs/commands.sh +151 -0
  261. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/outputs/routing.md +51 -0
  262. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/timing.json +5 -0
  263. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/grading.json +9 -0
  264. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/outputs/analysis.md +127 -0
  265. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/outputs/classification.md +57 -0
  266. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/outputs/commands.sh +129 -0
  267. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/outputs/routing.md +84 -0
  268. package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/timing.json +5 -0
  269. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/eval_metadata.json +27 -0
  270. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/grading.json +8 -0
  271. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/outputs/analysis.md +98 -0
  272. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/outputs/classification.md +65 -0
  273. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/outputs/commands.sh +123 -0
  274. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/outputs/routing.md +66 -0
  275. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/timing.json +5 -0
  276. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/grading.json +8 -0
  277. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/outputs/analysis.md +88 -0
  278. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/outputs/classification.md +40 -0
  279. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/outputs/commands.sh +130 -0
  280. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/outputs/routing.md +61 -0
  281. package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/timing.json +5 -0
  282. package/skills/tfx-deep-interview/SKILL.md +1 -2
  283. package/skills/tfx-plan/SKILL.md.tmpl +2 -3
  284. package/skills/tfx-psmux-rules/SKILL.md +11 -2
  285. package/skills/tfx-qa/SKILL.md.tmpl +2 -3
  286. package/skills/tfx-remote-spawn/SKILL.md +8 -11
  287. package/skills/tfx-research/SKILL.md.tmpl +2 -3
  288. package/skills/tfx-review/SKILL.md.tmpl +2 -3
  289. package/skills/tfx-workspace/async-tests/run-tests.sh +203 -0
  290. package/skills/tfx-workspace/evals/evals.json +79 -0
  291. package/skills/tfx-workspace/iteration-1/benchmark.json +162 -0
  292. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/eval_metadata.json +11 -0
  293. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/old_skill/grading.json +9 -0
  294. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/old_skill/outputs/analysis.md +154 -0
  295. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/old_skill/timing.json +5 -0
  296. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/with_skill/grading.json +9 -0
  297. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/with_skill/outputs/analysis.md +126 -0
  298. package/skills/tfx-workspace/iteration-1/codex-gemini-remap/with_skill/timing.json +5 -0
  299. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/eval_metadata.json +11 -0
  300. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/old_skill/grading.json +9 -0
  301. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/old_skill/outputs/analysis.md +119 -0
  302. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/old_skill/timing.json +5 -0
  303. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/with_skill/grading.json +9 -0
  304. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/with_skill/outputs/analysis.md +115 -0
  305. package/skills/tfx-workspace/iteration-1/doctor-diagnosis/with_skill/timing.json +5 -0
  306. package/skills/tfx-workspace/iteration-1/hub-start-sequence/eval_metadata.json +10 -0
  307. package/skills/tfx-workspace/iteration-1/hub-start-sequence/old_skill/grading.json +8 -0
  308. package/skills/tfx-workspace/iteration-1/hub-start-sequence/old_skill/outputs/analysis.md +86 -0
  309. package/skills/tfx-workspace/iteration-1/hub-start-sequence/old_skill/timing.json +5 -0
  310. package/skills/tfx-workspace/iteration-1/hub-start-sequence/with_skill/grading.json +8 -0
  311. package/skills/tfx-workspace/iteration-1/hub-start-sequence/with_skill/outputs/analysis.md +81 -0
  312. package/skills/tfx-workspace/iteration-1/hub-start-sequence/with_skill/timing.json +5 -0
  313. package/skills/tfx-workspace/iteration-1/multi-team-creation/eval_metadata.json +12 -0
  314. package/skills/tfx-workspace/iteration-1/multi-team-creation/old_skill/grading.json +10 -0
  315. package/skills/tfx-workspace/iteration-1/multi-team-creation/old_skill/outputs/analysis.md +316 -0
  316. package/skills/tfx-workspace/iteration-1/multi-team-creation/old_skill/timing.json +5 -0
  317. package/skills/tfx-workspace/iteration-1/multi-team-creation/with_skill/grading.json +10 -0
  318. package/skills/tfx-workspace/iteration-1/multi-team-creation/with_skill/outputs/analysis.md +352 -0
  319. package/skills/tfx-workspace/iteration-1/multi-team-creation/with_skill/timing.json +5 -0
  320. package/skills/tfx-workspace/iteration-1/review.html +1325 -0
  321. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/eval_metadata.json +12 -0
  322. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/old_skill/grading.json +10 -0
  323. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/old_skill/outputs/analysis.md +97 -0
  324. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/old_skill/timing.json +5 -0
  325. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/with_skill/grading.json +10 -0
  326. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/with_skill/outputs/analysis.md +94 -0
  327. package/skills/tfx-workspace/iteration-1/routing-implement-shortcut/with_skill/timing.json +5 -0
  328. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/eval_metadata.json +12 -0
  329. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/old_skill/grading.json +10 -0
  330. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/old_skill/outputs/analysis.md +209 -0
  331. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/old_skill/timing.json +5 -0
  332. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/with_skill/grading.json +10 -0
  333. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/with_skill/outputs/analysis.md +193 -0
  334. package/skills/tfx-workspace/iteration-1/routing-multi-task-triage/with_skill/timing.json +5 -0
  335. package/skills/tfx-workspace/iteration-2/benchmark.json +62 -0
  336. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/eval_metadata.json +13 -0
  337. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/old_skill/grading.json +11 -0
  338. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/old_skill/outputs/analysis.md +382 -0
  339. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/old_skill/timing.json +5 -0
  340. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/with_skill/grading.json +11 -0
  341. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/with_skill/outputs/analysis.md +333 -0
  342. package/skills/tfx-workspace/iteration-2/multi-team-creation-refactored/with_skill/timing.json +5 -0
  343. package/skills/tfx-workspace/iteration-2/review.html +1325 -0
  344. package/skills/tfx-workspace/skill-snapshot/tfx-auto/SKILL.md +217 -0
  345. package/skills/{tfx-auto-codex/SKILL.md.tmpl → tfx-workspace/skill-snapshot/tfx-auto-codex/SKILL.md} +3 -31
  346. package/skills/tfx-workspace/skill-snapshot/tfx-codex/SKILL.md +65 -0
  347. package/skills/tfx-workspace/skill-snapshot/tfx-doctor/SKILL.md +94 -0
  348. package/skills/{tfx-gemini/SKILL.md.tmpl → tfx-workspace/skill-snapshot/tfx-gemini/SKILL.md} +6 -14
  349. package/skills/tfx-workspace/skill-snapshot/tfx-hub/SKILL.md +133 -0
  350. package/skills/tfx-workspace/skill-snapshot/tfx-multi/SKILL.md +426 -0
  351. package/skills/tfx-workspace/skill-snapshot/tfx-setup/SKILL.md +101 -0
  352. package/skills/merge-worktree/SKILL.md.tmpl +0 -144
  353. package/skills/shared/arguments-processing.md +0 -2
  354. package/skills/shared/mandatory-rules.md +0 -6
  355. package/skills/shared/telemetry-segment.md +0 -6
  356. package/skills/star-prompt/SKILL.md.tmpl +0 -122
  357. package/skills/tfx-analysis/SKILL.md.tmpl +0 -106
  358. package/skills/tfx-analysis/skill.json +0 -11
  359. package/skills/tfx-auto/skill.json +0 -26
  360. package/skills/tfx-auto-codex/skill.json +0 -8
  361. package/skills/tfx-autopilot/SKILL.md.tmpl +0 -115
  362. package/skills/tfx-autopilot/skill.json +0 -10
  363. package/skills/tfx-autoresearch/SKILL.md.tmpl +0 -135
  364. package/skills/tfx-autoresearch/skill.json +0 -14
  365. package/skills/tfx-autoroute/SKILL.md.tmpl +0 -188
  366. package/skills/tfx-autoroute/skill.json +0 -12
  367. package/skills/tfx-codex/skill.json +0 -8
  368. package/skills/tfx-codex-swarm/SKILL.md.tmpl +0 -16
  369. package/skills/tfx-codex-swarm/skill.json +0 -5
  370. package/skills/tfx-consensus/SKILL.md.tmpl +0 -145
  371. package/skills/tfx-consensus/skill.json +0 -8
  372. package/skills/tfx-debate/SKILL.md.tmpl +0 -191
  373. package/skills/tfx-debate/skill.json +0 -12
  374. package/skills/tfx-deep-analysis/SKILL.md.tmpl +0 -227
  375. package/skills/tfx-deep-analysis/skill.json +0 -10
  376. package/skills/tfx-deep-interview/SKILL.md.tmpl +0 -203
  377. package/skills/tfx-deep-interview/skill.json +0 -12
  378. package/skills/tfx-deep-plan/SKILL.md.tmpl +0 -281
  379. package/skills/tfx-deep-plan/skill.json +0 -13
  380. package/skills/tfx-deep-qa/SKILL.md.tmpl +0 -164
  381. package/skills/tfx-deep-qa/skill.json +0 -11
  382. package/skills/tfx-deep-research/SKILL.md.tmpl +0 -216
  383. package/skills/tfx-deep-research/skill.json +0 -14
  384. package/skills/tfx-deep-review/SKILL.md.tmpl +0 -178
  385. package/skills/tfx-deep-review/skill.json +0 -12
  386. package/skills/tfx-doctor/SKILL.md.tmpl +0 -172
  387. package/skills/tfx-doctor/skill.json +0 -8
  388. package/skills/tfx-find/skill.json +0 -12
  389. package/skills/tfx-forge/SKILL.md.tmpl +0 -187
  390. package/skills/tfx-forge/skill.json +0 -12
  391. package/skills/tfx-fullcycle/SKILL.md.tmpl +0 -285
  392. package/skills/tfx-fullcycle/skill.json +0 -11
  393. package/skills/tfx-gemini/skill.json +0 -8
  394. package/skills/tfx-hooks/SKILL.md.tmpl +0 -216
  395. package/skills/tfx-hooks/skill.json +0 -8
  396. package/skills/tfx-hub/SKILL.md.tmpl +0 -212
  397. package/skills/tfx-hub/skill.json +0 -8
  398. package/skills/tfx-index/skill.json +0 -11
  399. package/skills/tfx-interview/SKILL.md.tmpl +0 -284
  400. package/skills/tfx-interview/skill.json +0 -12
  401. package/skills/tfx-multi/SKILL.md.tmpl +0 -183
  402. package/skills/tfx-multi/skill.json +0 -8
  403. package/skills/tfx-panel/SKILL.md.tmpl +0 -188
  404. package/skills/tfx-panel/skill.json +0 -12
  405. package/skills/tfx-persist/SKILL.md.tmpl +0 -269
  406. package/skills/tfx-persist/skill.json +0 -12
  407. package/skills/tfx-plan/skill.json +0 -11
  408. package/skills/tfx-profile/SKILL.md.tmpl +0 -239
  409. package/skills/tfx-profile/skill.json +0 -8
  410. package/skills/tfx-prune/SKILL.md.tmpl +0 -199
  411. package/skills/tfx-prune/skill.json +0 -12
  412. package/skills/tfx-psmux-rules/SKILL.md.tmpl +0 -317
  413. package/skills/tfx-psmux-rules/skill.json +0 -8
  414. package/skills/tfx-qa/skill.json +0 -11
  415. package/skills/tfx-ralph/SKILL.md.tmpl +0 -27
  416. package/skills/tfx-ralph/skill.json +0 -8
  417. package/skills/tfx-remote-setup/SKILL.md.tmpl +0 -576
  418. package/skills/tfx-remote-setup/skill.json +0 -8
  419. package/skills/tfx-remote-spawn/SKILL.md.tmpl +0 -263
  420. package/skills/tfx-remote-spawn/skill.json +0 -9
  421. package/skills/tfx-research/skill.json +0 -13
  422. package/skills/tfx-review/skill.json +0 -11
  423. package/skills/tfx-setup/SKILL.md.tmpl +0 -380
  424. package/skills/tfx-setup/skill.json +0 -8
  425. package/skills/tfx-swarm/SKILL.md.tmpl +0 -154
  426. package/skills/tfx-swarm/skill.json +0 -5
@@ -0,0 +1,326 @@
1
+ import { existsSync, readdirSync, readFileSync } from "node:fs";
2
+ import { basename, extname, isAbsolute, join, relative, resolve } from "node:path";
3
+
4
+ const IF_TAG_RE = /{{\s*(#if\s+([A-Za-z0-9_.-]+)|\/if)\s*}}/g;
5
+ const PARTIAL_RE = /{{>\s*([A-Za-z0-9_./-]+)\s*}}/g;
6
+ const FILE_INCLUDE_RE = /{{#include\s+([A-Za-z0-9_./-]+)\s*}}/g;
7
+ const VARIABLE_RE = /{{\s*([A-Za-z0-9_.-]+)\s*}}/g;
8
+ const FRONTMATTER_RE = /^---\r?\n(?:([\s\S]*?)\r?\n)?---\r?\n?/;
9
+
10
+ function isTruthy(value) {
11
+ if (typeof value === "string") return value.trim().length > 0;
12
+ return Boolean(value);
13
+ }
14
+
15
+ function normalizeVariable(context, key) {
16
+ if (Object.prototype.hasOwnProperty.call(context, key)) return context[key];
17
+ const upper = key.toUpperCase();
18
+ if (Object.prototype.hasOwnProperty.call(context, upper)) return context[upper];
19
+ return undefined;
20
+ }
21
+
22
+ function parseBoolean(raw) {
23
+ if (typeof raw !== "string") return undefined;
24
+ if (raw === "true") return true;
25
+ if (raw === "false") return false;
26
+ return undefined;
27
+ }
28
+
29
+ function parseScalar(raw) {
30
+ if (raw == null) return "";
31
+ const value = raw.trim();
32
+ if (!value) return "";
33
+ if (
34
+ (value.startsWith('"') && value.endsWith('"')) ||
35
+ (value.startsWith("'") && value.endsWith("'"))
36
+ ) {
37
+ return value.slice(1, -1);
38
+ }
39
+ const bool = parseBoolean(value);
40
+ return bool ?? value;
41
+ }
42
+
43
+ function parseMultilineValue(lines, startIndex, marker) {
44
+ const fold = marker.startsWith(">");
45
+ const chunks = [];
46
+ let index = startIndex;
47
+
48
+ while (index + 1 < lines.length && /^\s+/.test(lines[index + 1])) {
49
+ index += 1;
50
+ chunks.push(lines[index].replace(/^\s+/, ""));
51
+ }
52
+
53
+ return {
54
+ index,
55
+ value: fold ? chunks.join(" ").trim() : chunks.join("\n").trim(),
56
+ };
57
+ }
58
+
59
+ function parseListValue(lines, startIndex) {
60
+ const items = [];
61
+ let index = startIndex;
62
+
63
+ while (index + 1 < lines.length) {
64
+ const nextLine = lines[index + 1];
65
+ const match = nextLine.match(/^\s*-\s+(.*)$/);
66
+ if (match) {
67
+ index += 1;
68
+ items.push(parseScalar(match[1]));
69
+ continue;
70
+ }
71
+ if (/^\s*$/.test(nextLine)) {
72
+ index += 1;
73
+ continue;
74
+ }
75
+ break;
76
+ }
77
+
78
+ return { index, value: items };
79
+ }
80
+
81
+ function parseFrontmatterBlock(block) {
82
+ const lines = block.split(/\r?\n/);
83
+ const data = {};
84
+
85
+ for (let i = 0; i < lines.length; i += 1) {
86
+ const line = lines[i];
87
+ const match = line.match(/^([A-Za-z0-9_-]+):\s*(.*)$/);
88
+ if (!match) continue;
89
+
90
+ const [, rawKey, rawValue] = match;
91
+ const key = rawKey.trim();
92
+ const value = rawValue.trim();
93
+
94
+ if (value === ">" || value === "|" || value === ">-" || value === "|-") {
95
+ const parsed = parseMultilineValue(lines, i, value);
96
+ data[key] = parsed.value;
97
+ i = parsed.index;
98
+ continue;
99
+ }
100
+
101
+ if (!value) {
102
+ const parsed = parseListValue(lines, i);
103
+ if (parsed.index !== i) {
104
+ data[key] = parsed.value;
105
+ i = parsed.index;
106
+ continue;
107
+ }
108
+ }
109
+
110
+ data[key] = parseScalar(value);
111
+ }
112
+
113
+ return data;
114
+ }
115
+
116
+ function evaluateConditionals(source, context) {
117
+ const matcher = new RegExp(IF_TAG_RE.source, "g");
118
+
119
+ function walk(startIndex, nested) {
120
+ let output = "";
121
+ let cursor = startIndex;
122
+
123
+ while (true) {
124
+ const found = matcher.exec(source);
125
+ if (!found) {
126
+ if (nested) throw new Error("Unclosed {{#if ...}} block in template");
127
+ output += source.slice(cursor);
128
+ return { output, index: source.length };
129
+ }
130
+
131
+ output += source.slice(cursor, found.index);
132
+ cursor = matcher.lastIndex;
133
+
134
+ const directive = found[1];
135
+ const flag = found[2];
136
+ if (directive.startsWith("#if")) {
137
+ const nestedResult = walk(cursor, true);
138
+ cursor = nestedResult.index;
139
+ matcher.lastIndex = cursor;
140
+ if (isTruthy(normalizeVariable(context, flag))) {
141
+ output += nestedResult.output;
142
+ }
143
+ continue;
144
+ }
145
+
146
+ if (!nested) {
147
+ throw new Error("Unexpected {{/if}} without matching {{#if ...}}");
148
+ }
149
+ return { output, index: cursor };
150
+ }
151
+ }
152
+
153
+ return walk(0, false).output;
154
+ }
155
+
156
+ function renderPartials(source, context, partials, options, includeStack) {
157
+ return source.replace(PARTIAL_RE, (_full, partialName) => {
158
+ const partial = partials[partialName];
159
+ if (partial == null) {
160
+ throw new Error(`Missing partial: ${partialName}`);
161
+ }
162
+ if (includeStack.includes(partialName)) {
163
+ const chain = [...includeStack, partialName].join(" -> ");
164
+ throw new Error(`Circular partial include: ${chain}`);
165
+ }
166
+ return renderWithContext(partial, context, partials, options, [
167
+ ...includeStack,
168
+ partialName,
169
+ ]);
170
+ });
171
+ }
172
+
173
+ function normalizeIncludeName(includeName) {
174
+ return includeName.replace(/\\/g, "/");
175
+ }
176
+
177
+ function resolveIncludeContent(includeName, options) {
178
+ const normalizedName = normalizeIncludeName(includeName);
179
+ if (Object.prototype.hasOwnProperty.call(options.includes, normalizedName)) {
180
+ return options.includes[normalizedName];
181
+ }
182
+ if (!options.includeBaseDir) {
183
+ throw new Error(`Missing include: ${normalizedName}`);
184
+ }
185
+
186
+ const baseDir = resolve(options.includeBaseDir);
187
+ const fullPath = resolve(baseDir, normalizedName);
188
+ const relativePath = relative(baseDir, fullPath);
189
+ const isInsideBaseDir =
190
+ relativePath === "" || (!relativePath.startsWith("..") && !isAbsolute(relativePath));
191
+
192
+ if (!isInsideBaseDir || !existsSync(fullPath)) {
193
+ throw new Error(`Missing include: ${normalizedName}`);
194
+ }
195
+
196
+ const content = readFileSync(fullPath, "utf8");
197
+ options.includes[normalizedName] = content;
198
+ return content;
199
+ }
200
+
201
+ function renderFileIncludes(source, context, partials, options, includeStack) {
202
+ return source.replace(FILE_INCLUDE_RE, (_full, includeName) => {
203
+ const normalizedName = normalizeIncludeName(includeName);
204
+ const content = resolveIncludeContent(normalizedName, options);
205
+ if (includeStack.includes(normalizedName)) {
206
+ const chain = [...includeStack, normalizedName].join(" -> ");
207
+ throw new Error(`Circular template include: ${chain}`);
208
+ }
209
+ return renderWithContext(content, context, partials, options, [
210
+ ...includeStack,
211
+ normalizedName,
212
+ ]);
213
+ });
214
+ }
215
+
216
+ function renderVariables(source, context) {
217
+ return source.replace(VARIABLE_RE, (full, key) => {
218
+ const value = normalizeVariable(context, key);
219
+ if (value == null) {
220
+ throw new Error(`Missing template variable: ${key}`);
221
+ }
222
+ return String(value);
223
+ });
224
+ }
225
+
226
+ function renderWithContext(source, context, partials, options, includeStack = []) {
227
+ const afterIf = evaluateConditionals(source, context);
228
+ const afterPartials = renderPartials(afterIf, context, partials, options, includeStack);
229
+ const afterFileIncludes = renderFileIncludes(
230
+ afterPartials,
231
+ context,
232
+ partials,
233
+ options,
234
+ includeStack,
235
+ );
236
+ return renderVariables(afterFileIncludes, context);
237
+ }
238
+
239
+ function readAllTemplateFiles(rootDir, currentDir = rootDir) {
240
+ if (!rootDir || !existsSync(rootDir)) return [];
241
+
242
+ const entries = readdirSync(currentDir, { withFileTypes: true }).sort((left, right) =>
243
+ left.name.localeCompare(right.name),
244
+ );
245
+ const files = [];
246
+
247
+ for (const entry of entries) {
248
+ const fullPath = join(currentDir, entry.name);
249
+ if (entry.isDirectory()) {
250
+ files.push(...readAllTemplateFiles(rootDir, fullPath));
251
+ continue;
252
+ }
253
+
254
+ if (!entry.isFile()) continue;
255
+ if (!entry.name.endsWith(".md") && !entry.name.endsWith(".tmpl")) continue;
256
+
257
+ const relativePath = relative(rootDir, fullPath).replace(/\\/g, "/");
258
+ const content = readFileSync(fullPath, "utf8");
259
+ files.push({ fullPath, relativePath, content });
260
+ }
261
+
262
+ return files;
263
+ }
264
+
265
+ function setPartial(partials, key, value) {
266
+ if (!key) return;
267
+ if (!Object.prototype.hasOwnProperty.call(partials, key)) {
268
+ partials[key] = value;
269
+ }
270
+ }
271
+
272
+ export function parseFrontmatter(source) {
273
+ const match = source.match(FRONTMATTER_RE);
274
+ if (!match) return { data: {}, body: source };
275
+
276
+ const data = parseFrontmatterBlock(match[1] ?? "");
277
+ const body = source.slice(match[0].length);
278
+ return { data, body };
279
+ }
280
+
281
+ export function buildSkillTemplateContext({ frontmatter = {}, skillDirName = "" } = {}) {
282
+ const context = { ...frontmatter };
283
+ const skillName = String(frontmatter.name || skillDirName || "").trim();
284
+ const skillDescription = String(frontmatter.description || "").trim();
285
+
286
+ let deep = frontmatter.DEEP ?? frontmatter.deep;
287
+ if (typeof deep === "string") {
288
+ const parsed = parseBoolean(deep.trim().toLowerCase());
289
+ deep = parsed ?? deep;
290
+ }
291
+ if (typeof deep !== "boolean") {
292
+ deep = /(^|[-_])deep($|[-_])/i.test(skillName);
293
+ }
294
+
295
+ context.SKILL_NAME = skillName;
296
+ context.SKILL_DESCRIPTION = skillDescription;
297
+ context.DEEP = deep;
298
+ return context;
299
+ }
300
+
301
+ export function loadTemplatePartials(partialsDir) {
302
+ const files = readAllTemplateFiles(partialsDir);
303
+ const partials = {};
304
+
305
+ for (const file of files) {
306
+ const extension = extname(file.relativePath);
307
+ const withoutExt = extension
308
+ ? file.relativePath.slice(0, -extension.length)
309
+ : file.relativePath;
310
+ const normalized = withoutExt.replace(/\\/g, "/");
311
+ const base = basename(normalized);
312
+
313
+ setPartial(partials, normalized, file.content);
314
+ setPartial(partials, base, file.content);
315
+ }
316
+
317
+ return partials;
318
+ }
319
+
320
+ export function renderSkillTemplate(template, context = {}, options = {}) {
321
+ const { partials = {}, includes = {}, includeBaseDir = "" } = options;
322
+ return renderWithContext(template, context, partials, {
323
+ includeBaseDir,
324
+ includes: { ...includes },
325
+ });
326
+ }
@@ -0,0 +1,237 @@
1
+ #!/usr/bin/env node
2
+ // MCP inventory cache for dynamic MCP filtering.
3
+
4
+ import { execSync } from 'node:child_process';
5
+ import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from 'node:fs';
6
+ import { homedir } from 'node:os';
7
+ import { basename, dirname, join, resolve } from 'node:path';
8
+ import { fileURLToPath } from 'node:url';
9
+
10
+ import { MCP_SERVER_TOOL_CATALOG, normalizeServerMetadata } from './lib/mcp-server-catalog.mjs';
11
+
12
+ const CACHE_DIR = join(homedir(), '.claude', 'cache');
13
+ const CACHE_FILE = join(CACHE_DIR, 'mcp-inventory.json');
14
+
15
+ function countConfiguredTools(config = {}, fallbackToolCount = 0) {
16
+ const directKeys = ['tools', 'toolNames', 'allowedTools', 'includeTools'];
17
+ for (const key of directKeys) {
18
+ if (Array.isArray(config[key])) return config[key].length;
19
+ }
20
+
21
+ if (Array.isArray(config.excludeTools)) {
22
+ return Math.max(0, fallbackToolCount - config.excludeTools.length);
23
+ }
24
+
25
+ return fallbackToolCount;
26
+ }
27
+
28
+ export function createServerRecord(name, status, config = {}) {
29
+ const normalizedName = typeof name === 'string' ? name.trim() : '';
30
+ const fallback = normalizeServerMetadata(normalizedName, {});
31
+ const toolCount = countConfiguredTools(config, fallback.tool_count);
32
+ const domainTags = Array.isArray(config.domain_tags)
33
+ ? config.domain_tags
34
+ : Array.isArray(config.domainTags)
35
+ ? config.domainTags
36
+ : [];
37
+
38
+ const metadata = normalizeServerMetadata(normalizedName, {
39
+ ...config,
40
+ tool_count: toolCount,
41
+ domain_tags: domainTags,
42
+ });
43
+
44
+ return {
45
+ name: normalizedName,
46
+ status,
47
+ tool_count: metadata.tool_count,
48
+ domain_tags: metadata.domain_tags,
49
+ };
50
+ }
51
+
52
+ export function getCodexMcp() {
53
+ try {
54
+ const output = execSync('codex mcp list', {
55
+ encoding: 'utf8',
56
+ timeout: 15000,
57
+ stdio: ['pipe', 'pipe', 'ignore'],
58
+ windowsHide: true,
59
+ });
60
+ const lines = output.trim().split(/\r?\n/).filter((line) => line.trim());
61
+ if (lines.length < 2) return [];
62
+
63
+ const servers = [];
64
+ for (let i = 1; i < lines.length; i += 1) {
65
+ const cols = lines[i].split(/\s{2,}/);
66
+ if (cols.length < 2) continue;
67
+
68
+ const name = cols[0].trim();
69
+ const statusMatch = lines[i].match(/\b(enabled|disabled)\b/i);
70
+ const status = statusMatch ? statusMatch[1].toLowerCase() : 'unknown';
71
+ if (!name || name.startsWith('-')) continue;
72
+ servers.push(createServerRecord(name, status));
73
+ }
74
+ return servers;
75
+ } catch {
76
+ return null;
77
+ }
78
+ }
79
+
80
+ export function getGeminiMcp() {
81
+ try {
82
+ const settingsPath = join(homedir(), '.gemini', 'settings.json');
83
+ if (!existsSync(settingsPath)) return null;
84
+
85
+ const settings = JSON.parse(readFileSync(settingsPath, 'utf8'));
86
+ const mcpServers = settings.mcpServers || {};
87
+ return Object.entries(mcpServers).map(([name, config]) => createServerRecord(name, 'configured', config || {}));
88
+ } catch {
89
+ return null;
90
+ }
91
+ }
92
+
93
+ // ── Claude MCP 서버 발견 ──
94
+
95
+ const CLAUDE_DIR = join(homedir(), '.claude');
96
+
97
+ // Anthropic 클라우드 호스팅 MCP 서버 — Claude Code 런타임에서 항상 가용.
98
+ // mcp-server-catalog.mjs의 MCP_SERVER_TOOL_CATALOG에 정의된 서버 중
99
+ // 로컬 설치가 불필요한 클라우드 제공 서버.
100
+ const WELL_KNOWN_CLOUD_SERVERS = Object.freeze([
101
+ 'brave-search',
102
+ 'exa',
103
+ 'context7',
104
+ ]);
105
+
106
+ function readJsonSafe(filePath) {
107
+ try {
108
+ return JSON.parse(readFileSync(filePath, 'utf8'));
109
+ } catch {
110
+ return null;
111
+ }
112
+ }
113
+
114
+ function extractMcpServerNames(mcpJsonPath) {
115
+ const data = readJsonSafe(mcpJsonPath);
116
+ if (!data?.mcpServers || typeof data.mcpServers !== 'object') return [];
117
+ return Object.entries(data.mcpServers)
118
+ .filter(([name]) => typeof name === 'string' && name.trim())
119
+ .map(([name, config]) => ({ name: name.trim(), config: config || {} }));
120
+ }
121
+
122
+ function walkUpForMcpJson(startDir, maxDepth = 5) {
123
+ const found = [];
124
+ let dir = resolve(startDir);
125
+ for (let i = 0; i < maxDepth; i += 1) {
126
+ const candidate = join(dir, '.mcp.json');
127
+ if (existsSync(candidate)) found.push(candidate);
128
+ const parent = dirname(dir);
129
+ if (parent === dir) break;
130
+ dir = parent;
131
+ }
132
+ return found;
133
+ }
134
+
135
+ function scanPluginsMcpJson() {
136
+ const pluginsDir = join(CLAUDE_DIR, 'plugins');
137
+ if (!existsSync(pluginsDir)) return [];
138
+ const found = [];
139
+ const walk = (dir, depth = 0) => {
140
+ if (depth > 4) return;
141
+ try {
142
+ for (const entry of readdirSync(dir, { withFileTypes: true })) {
143
+ if (entry.name === 'node_modules' || entry.name === '.git') continue;
144
+ const full = join(dir, entry.name);
145
+ if (entry.isFile() && entry.name === '.mcp.json') {
146
+ found.push(full);
147
+ } else if (entry.isDirectory()) {
148
+ walk(full, depth + 1);
149
+ }
150
+ }
151
+ } catch { /* permission errors */ }
152
+ };
153
+ walk(pluginsDir);
154
+ return found;
155
+ }
156
+
157
+ export function getClaudeMcp(cwd = process.cwd()) {
158
+ const serverMap = new Map();
159
+
160
+ // 1) .mcp.json 파일 트리 스캔 (CWD → root)
161
+ for (const mcpJson of walkUpForMcpJson(cwd)) {
162
+ for (const { name, config } of extractMcpServerNames(mcpJson)) {
163
+ if (!serverMap.has(name)) {
164
+ serverMap.set(name, createServerRecord(name, 'configured', config));
165
+ }
166
+ }
167
+ }
168
+
169
+ // 2) ~/.claude/plugins/ 스캔
170
+ for (const mcpJson of scanPluginsMcpJson()) {
171
+ for (const { name, config } of extractMcpServerNames(mcpJson)) {
172
+ if (!serverMap.has(name)) {
173
+ serverMap.set(name, createServerRecord(name, 'configured', config));
174
+ }
175
+ }
176
+ }
177
+
178
+ // 3) ~/.claude/settings.json + settings.local.json의 mcpServers
179
+ for (const settingsFile of ['settings.json', 'settings.local.json']) {
180
+ const data = readJsonSafe(join(CLAUDE_DIR, settingsFile));
181
+ if (!data?.mcpServers) continue;
182
+ for (const [name, config] of Object.entries(data.mcpServers)) {
183
+ if (typeof name !== 'string' || !name.trim()) continue;
184
+ if (!serverMap.has(name.trim())) {
185
+ serverMap.set(name.trim(), createServerRecord(name.trim(), 'configured', config || {}));
186
+ }
187
+ }
188
+ }
189
+
190
+ // 4) Well-known Anthropic 클라우드 MCP 서버 (카탈로그에 정의된 것만)
191
+ for (const name of WELL_KNOWN_CLOUD_SERVERS) {
192
+ if (serverMap.has(name)) continue;
193
+ if (!MCP_SERVER_TOOL_CATALOG[name]) continue;
194
+ serverMap.set(name, createServerRecord(name, 'available'));
195
+ }
196
+
197
+ return [...serverMap.values()];
198
+ }
199
+
200
+ export function buildInventory(cwd = process.cwd()) {
201
+ const inventory = {
202
+ timestamp: new Date().toISOString(),
203
+ codex: { available: false, servers: [] },
204
+ gemini: { available: false, servers: [] },
205
+ claude: { available: true, servers: [] },
206
+ };
207
+
208
+ const codexServers = getCodexMcp();
209
+ if (codexServers !== null) {
210
+ inventory.codex.available = true;
211
+ inventory.codex.servers = codexServers;
212
+ }
213
+
214
+ const geminiServers = getGeminiMcp();
215
+ if (geminiServers !== null) {
216
+ inventory.gemini.available = true;
217
+ inventory.gemini.servers = geminiServers;
218
+ }
219
+
220
+ inventory.claude.servers = getClaudeMcp(cwd);
221
+
222
+ return inventory;
223
+ }
224
+
225
+ export function writeInventoryCache(inventory = buildInventory()) {
226
+ if (!existsSync(CACHE_DIR)) mkdirSync(CACHE_DIR, { recursive: true });
227
+ writeFileSync(CACHE_FILE, JSON.stringify(inventory, null, 2));
228
+ return inventory;
229
+ }
230
+
231
+ export function main() {
232
+ writeInventoryCache();
233
+ }
234
+
235
+ if (process.argv[1] && fileURLToPath(import.meta.url) === process.argv[1]) {
236
+ main();
237
+ }
@@ -0,0 +1,17 @@
1
+ # mcp-cleanup.ps1 — Claude Code Stop hook: MCP 고아 프로세스 정리
2
+ # Windows에서 Claude Code 세션 종료 시 남는 MCP 서버 고아 프로세스를 정리한다.
3
+ # 원인: Claude Code가 stdio MCP 자식 프로세스 트리를 Windows에서 제대로 kill하지 못함
4
+ # (GitHub Issues #1935, #15211, #28126)
5
+ $ErrorActionPreference = 'SilentlyContinue'
6
+
7
+ # npx MCP servers (brave, notion, context7, exa, tavily, jira, playwright, etc.)
8
+ # + oh-my-codex MCP servers (team/code-intel/memory/trace/state)
9
+ # + omc bridge
10
+ Get-CimInstance Win32_Process -Filter "Name='node.exe' OR Name='cmd.exe'" |
11
+ Where-Object { $_.CommandLine -match 'npx-cli|oh-my-codex[\\/]dist[\\/]mcp|omc.*bridge.*mcp-server' } |
12
+ ForEach-Object { taskkill /F /PID $_.ProcessId 2>$null }
13
+
14
+ # serena (uvx) + python MCP orphans
15
+ Get-CimInstance Win32_Process -Filter "Name='python.exe' OR Name='uvx.exe'" |
16
+ Where-Object { $_.CommandLine -match 'serena|uv[\\/](cache|python)' } |
17
+ ForEach-Object { taskkill /F /PID $_.ProcessId 2>$null }