calvyn-code 0.14.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 (1718) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +217 -0
  3. package/README.zh-CN.md +180 -0
  4. package/acp_adapter/__init__.py +1 -0
  5. package/acp_adapter/__main__.py +5 -0
  6. package/acp_adapter/auth.py +68 -0
  7. package/acp_adapter/bootstrap/__init__.py +0 -0
  8. package/acp_adapter/bootstrap/bootstrap_browser_tools.ps1 +288 -0
  9. package/acp_adapter/bootstrap/bootstrap_browser_tools.sh +399 -0
  10. package/acp_adapter/entry.py +292 -0
  11. package/acp_adapter/events.py +265 -0
  12. package/acp_adapter/permissions.py +148 -0
  13. package/acp_adapter/server.py +1713 -0
  14. package/acp_adapter/session.py +629 -0
  15. package/acp_adapter/tools.py +1180 -0
  16. package/agent/__init__.py +6 -0
  17. package/agent/__pycache__/__init__.cpython-312.pyc +0 -0
  18. package/agent/__pycache__/account_usage.cpython-312.pyc +0 -0
  19. package/agent/__pycache__/anthropic_adapter.cpython-312.pyc +0 -0
  20. package/agent/__pycache__/async_utils.cpython-312.pyc +0 -0
  21. package/agent/__pycache__/auxiliary_client.cpython-312.pyc +0 -0
  22. package/agent/__pycache__/codex_responses_adapter.cpython-312.pyc +0 -0
  23. package/agent/__pycache__/context_compressor.cpython-312.pyc +0 -0
  24. package/agent/__pycache__/context_engine.cpython-312.pyc +0 -0
  25. package/agent/__pycache__/context_references.cpython-312.pyc +0 -0
  26. package/agent/__pycache__/credential_pool.cpython-312.pyc +0 -0
  27. package/agent/__pycache__/curator.cpython-312.pyc +0 -0
  28. package/agent/__pycache__/display.cpython-312.pyc +0 -0
  29. package/agent/__pycache__/error_classifier.cpython-312.pyc +0 -0
  30. package/agent/__pycache__/file_safety.cpython-312.pyc +0 -0
  31. package/agent/__pycache__/google_code_assist.cpython-312.pyc +0 -0
  32. package/agent/__pycache__/google_oauth.cpython-312.pyc +0 -0
  33. package/agent/__pycache__/i18n.cpython-312.pyc +0 -0
  34. package/agent/__pycache__/image_gen_provider.cpython-312.pyc +0 -0
  35. package/agent/__pycache__/image_gen_registry.cpython-312.pyc +0 -0
  36. package/agent/__pycache__/insights.cpython-312.pyc +0 -0
  37. package/agent/__pycache__/lmstudio_reasoning.cpython-312.pyc +0 -0
  38. package/agent/__pycache__/manual_compression_feedback.cpython-312.pyc +0 -0
  39. package/agent/__pycache__/markdown_tables.cpython-312.pyc +0 -0
  40. package/agent/__pycache__/memory_manager.cpython-312.pyc +0 -0
  41. package/agent/__pycache__/memory_provider.cpython-312.pyc +0 -0
  42. package/agent/__pycache__/model_metadata.cpython-312.pyc +0 -0
  43. package/agent/__pycache__/models_dev.cpython-312.pyc +0 -0
  44. package/agent/__pycache__/moonshot_schema.cpython-312.pyc +0 -0
  45. package/agent/__pycache__/onboarding.cpython-312.pyc +0 -0
  46. package/agent/__pycache__/portal_tags.cpython-312.pyc +0 -0
  47. package/agent/__pycache__/prompt_builder.cpython-312.pyc +0 -0
  48. package/agent/__pycache__/prompt_caching.cpython-312.pyc +0 -0
  49. package/agent/__pycache__/redact.cpython-312.pyc +0 -0
  50. package/agent/__pycache__/retry_utils.cpython-312.pyc +0 -0
  51. package/agent/__pycache__/shell_hooks.cpython-312.pyc +0 -0
  52. package/agent/__pycache__/skill_commands.cpython-312.pyc +0 -0
  53. package/agent/__pycache__/skill_preprocessing.cpython-312.pyc +0 -0
  54. package/agent/__pycache__/skill_utils.cpython-312.pyc +0 -0
  55. package/agent/__pycache__/subdirectory_hints.cpython-312.pyc +0 -0
  56. package/agent/__pycache__/think_scrubber.cpython-312.pyc +0 -0
  57. package/agent/__pycache__/title_generator.cpython-312.pyc +0 -0
  58. package/agent/__pycache__/tool_guardrails.cpython-312.pyc +0 -0
  59. package/agent/__pycache__/tool_result_classification.cpython-312.pyc +0 -0
  60. package/agent/__pycache__/trajectory.cpython-312.pyc +0 -0
  61. package/agent/__pycache__/usage_pricing.cpython-312.pyc +0 -0
  62. package/agent/__pycache__/video_gen_provider.cpython-312.pyc +0 -0
  63. package/agent/__pycache__/video_gen_registry.cpython-312.pyc +0 -0
  64. package/agent/__pycache__/web_search_provider.cpython-312.pyc +0 -0
  65. package/agent/__pycache__/web_search_registry.cpython-312.pyc +0 -0
  66. package/agent/account_usage.py +326 -0
  67. package/agent/anthropic_adapter.py +2087 -0
  68. package/agent/async_utils.py +68 -0
  69. package/agent/auxiliary_client.py +4893 -0
  70. package/agent/bedrock_adapter.py +1276 -0
  71. package/agent/codex_responses_adapter.py +1084 -0
  72. package/agent/context_compressor.py +1583 -0
  73. package/agent/context_engine.py +211 -0
  74. package/agent/context_references.py +519 -0
  75. package/agent/copilot_acp_client.py +684 -0
  76. package/agent/credential_pool.py +1780 -0
  77. package/agent/credential_sources.py +449 -0
  78. package/agent/curator.py +1782 -0
  79. package/agent/curator_backup.py +694 -0
  80. package/agent/display.py +987 -0
  81. package/agent/error_classifier.py +1058 -0
  82. package/agent/file_safety.py +112 -0
  83. package/agent/gemini_cloudcode_adapter.py +909 -0
  84. package/agent/gemini_native_adapter.py +971 -0
  85. package/agent/gemini_schema.py +99 -0
  86. package/agent/google_code_assist.py +452 -0
  87. package/agent/google_oauth.py +1062 -0
  88. package/agent/i18n.py +258 -0
  89. package/agent/image_gen_provider.py +243 -0
  90. package/agent/image_gen_registry.py +145 -0
  91. package/agent/image_routing.py +301 -0
  92. package/agent/insights.py +931 -0
  93. package/agent/lmstudio_reasoning.py +48 -0
  94. package/agent/lsp/__init__.py +106 -0
  95. package/agent/lsp/__pycache__/__init__.cpython-312.pyc +0 -0
  96. package/agent/lsp/__pycache__/cli.cpython-312.pyc +0 -0
  97. package/agent/lsp/__pycache__/client.cpython-312.pyc +0 -0
  98. package/agent/lsp/__pycache__/eventlog.cpython-312.pyc +0 -0
  99. package/agent/lsp/__pycache__/manager.cpython-312.pyc +0 -0
  100. package/agent/lsp/__pycache__/protocol.cpython-312.pyc +0 -0
  101. package/agent/lsp/__pycache__/servers.cpython-312.pyc +0 -0
  102. package/agent/lsp/__pycache__/workspace.cpython-312.pyc +0 -0
  103. package/agent/lsp/cli.py +308 -0
  104. package/agent/lsp/client.py +930 -0
  105. package/agent/lsp/eventlog.py +213 -0
  106. package/agent/lsp/install.py +376 -0
  107. package/agent/lsp/manager.py +644 -0
  108. package/agent/lsp/protocol.py +196 -0
  109. package/agent/lsp/range_shift.py +149 -0
  110. package/agent/lsp/reporter.py +78 -0
  111. package/agent/lsp/servers.py +1040 -0
  112. package/agent/lsp/workspace.py +223 -0
  113. package/agent/manual_compression_feedback.py +49 -0
  114. package/agent/markdown_tables.py +309 -0
  115. package/agent/memory_manager.py +556 -0
  116. package/agent/memory_provider.py +279 -0
  117. package/agent/model_metadata.py +1827 -0
  118. package/agent/models_dev.py +724 -0
  119. package/agent/moonshot_schema.py +231 -0
  120. package/agent/nous_rate_guard.py +326 -0
  121. package/agent/onboarding.py +193 -0
  122. package/agent/plugin_llm.py +1046 -0
  123. package/agent/portal_tags.py +64 -0
  124. package/agent/prompt_builder.py +1457 -0
  125. package/agent/prompt_caching.py +79 -0
  126. package/agent/rate_limit_tracker.py +246 -0
  127. package/agent/redact.py +403 -0
  128. package/agent/retry_utils.py +57 -0
  129. package/agent/shell_hooks.py +837 -0
  130. package/agent/skill_commands.py +502 -0
  131. package/agent/skill_preprocessing.py +131 -0
  132. package/agent/skill_utils.py +512 -0
  133. package/agent/subdirectory_hints.py +224 -0
  134. package/agent/think_scrubber.py +386 -0
  135. package/agent/title_generator.py +171 -0
  136. package/agent/tool_guardrails.py +458 -0
  137. package/agent/tool_result_classification.py +26 -0
  138. package/agent/trajectory.py +56 -0
  139. package/agent/transports/__init__.py +68 -0
  140. package/agent/transports/__pycache__/__init__.cpython-312.pyc +0 -0
  141. package/agent/transports/__pycache__/anthropic.cpython-312.pyc +0 -0
  142. package/agent/transports/__pycache__/base.cpython-312.pyc +0 -0
  143. package/agent/transports/__pycache__/bedrock.cpython-312.pyc +0 -0
  144. package/agent/transports/__pycache__/chat_completions.cpython-312.pyc +0 -0
  145. package/agent/transports/__pycache__/codex.cpython-312.pyc +0 -0
  146. package/agent/transports/__pycache__/types.cpython-312.pyc +0 -0
  147. package/agent/transports/anthropic.py +179 -0
  148. package/agent/transports/base.py +89 -0
  149. package/agent/transports/bedrock.py +154 -0
  150. package/agent/transports/chat_completions.py +614 -0
  151. package/agent/transports/codex.py +283 -0
  152. package/agent/transports/codex_app_server.py +368 -0
  153. package/agent/transports/codex_app_server_session.py +810 -0
  154. package/agent/transports/codex_event_projector.py +312 -0
  155. package/agent/transports/hermes_tools_mcp_server.py +233 -0
  156. package/agent/transports/types.py +162 -0
  157. package/agent/usage_pricing.py +877 -0
  158. package/agent/video_gen_provider.py +300 -0
  159. package/agent/video_gen_registry.py +117 -0
  160. package/agent/web_search_provider.py +221 -0
  161. package/agent/web_search_registry.py +262 -0
  162. package/assets/banner.png +0 -0
  163. package/batch_runner.py +1303 -0
  164. package/bin/calvyn.js +67 -0
  165. package/calvyn_bootstrap.py +130 -0
  166. package/calvyn_constants.py +346 -0
  167. package/calvyn_logging.py +390 -0
  168. package/calvyn_state.py +2967 -0
  169. package/calvyn_time.py +105 -0
  170. package/cli.py +14160 -0
  171. package/cron/__init__.py +42 -0
  172. package/cron/__pycache__/__init__.cpython-312.pyc +0 -0
  173. package/cron/__pycache__/jobs.cpython-312.pyc +0 -0
  174. package/cron/__pycache__/scheduler.cpython-312.pyc +0 -0
  175. package/cron/jobs.py +1160 -0
  176. package/cron/scheduler.py +1832 -0
  177. package/gateway/__init__.py +35 -0
  178. package/gateway/__pycache__/__init__.cpython-312.pyc +0 -0
  179. package/gateway/__pycache__/channel_directory.cpython-312.pyc +0 -0
  180. package/gateway/__pycache__/config.cpython-312.pyc +0 -0
  181. package/gateway/__pycache__/delivery.cpython-312.pyc +0 -0
  182. package/gateway/__pycache__/display_config.cpython-312.pyc +0 -0
  183. package/gateway/__pycache__/hooks.cpython-312.pyc +0 -0
  184. package/gateway/__pycache__/pairing.cpython-312.pyc +0 -0
  185. package/gateway/__pycache__/platform_registry.cpython-312.pyc +0 -0
  186. package/gateway/__pycache__/restart.cpython-312.pyc +0 -0
  187. package/gateway/__pycache__/run.cpython-312.pyc +0 -0
  188. package/gateway/__pycache__/runtime_footer.cpython-312.pyc +0 -0
  189. package/gateway/__pycache__/session.cpython-312.pyc +0 -0
  190. package/gateway/__pycache__/session_context.cpython-312.pyc +0 -0
  191. package/gateway/__pycache__/shutdown_forensics.cpython-312.pyc +0 -0
  192. package/gateway/__pycache__/slash_access.cpython-312.pyc +0 -0
  193. package/gateway/__pycache__/status.cpython-312.pyc +0 -0
  194. package/gateway/__pycache__/stream_consumer.cpython-312.pyc +0 -0
  195. package/gateway/__pycache__/whatsapp_identity.cpython-312.pyc +0 -0
  196. package/gateway/assets/telegram-botfather-threads-settings.jpg +0 -0
  197. package/gateway/builtin_hooks/__init__.py +1 -0
  198. package/gateway/channel_directory.py +357 -0
  199. package/gateway/config.py +1873 -0
  200. package/gateway/delivery.py +258 -0
  201. package/gateway/display_config.py +206 -0
  202. package/gateway/hooks.py +210 -0
  203. package/gateway/mirror.py +179 -0
  204. package/gateway/pairing.py +322 -0
  205. package/gateway/platform_registry.py +260 -0
  206. package/gateway/platforms/ADDING_A_PLATFORM.md +374 -0
  207. package/gateway/platforms/__init__.py +45 -0
  208. package/gateway/platforms/__pycache__/__init__.cpython-312.pyc +0 -0
  209. package/gateway/platforms/__pycache__/base.cpython-312.pyc +0 -0
  210. package/gateway/platforms/__pycache__/helpers.cpython-312.pyc +0 -0
  211. package/gateway/platforms/__pycache__/telegram.cpython-312.pyc +0 -0
  212. package/gateway/platforms/__pycache__/telegram_network.cpython-312.pyc +0 -0
  213. package/gateway/platforms/__pycache__/yuanbao.cpython-312.pyc +0 -0
  214. package/gateway/platforms/__pycache__/yuanbao_media.cpython-312.pyc +0 -0
  215. package/gateway/platforms/__pycache__/yuanbao_proto.cpython-312.pyc +0 -0
  216. package/gateway/platforms/_http_client_limits.py +84 -0
  217. package/gateway/platforms/api_server.py +3488 -0
  218. package/gateway/platforms/base.py +3747 -0
  219. package/gateway/platforms/bluebubbles.py +937 -0
  220. package/gateway/platforms/dingtalk.py +1473 -0
  221. package/gateway/platforms/discord.py +5584 -0
  222. package/gateway/platforms/email.py +773 -0
  223. package/gateway/platforms/feishu.py +5059 -0
  224. package/gateway/platforms/feishu_comment.py +1382 -0
  225. package/gateway/platforms/feishu_comment_rules.py +430 -0
  226. package/gateway/platforms/helpers.py +279 -0
  227. package/gateway/platforms/homeassistant.py +449 -0
  228. package/gateway/platforms/matrix.py +2777 -0
  229. package/gateway/platforms/mattermost.py +852 -0
  230. package/gateway/platforms/msgraph_webhook.py +397 -0
  231. package/gateway/platforms/qqbot/__init__.py +91 -0
  232. package/gateway/platforms/qqbot/adapter.py +3072 -0
  233. package/gateway/platforms/qqbot/chunked_upload.py +602 -0
  234. package/gateway/platforms/qqbot/constants.py +74 -0
  235. package/gateway/platforms/qqbot/crypto.py +45 -0
  236. package/gateway/platforms/qqbot/keyboards.py +473 -0
  237. package/gateway/platforms/qqbot/onboard.py +220 -0
  238. package/gateway/platforms/qqbot/utils.py +71 -0
  239. package/gateway/platforms/signal.py +1518 -0
  240. package/gateway/platforms/signal_rate_limit.py +369 -0
  241. package/gateway/platforms/slack.py +3028 -0
  242. package/gateway/platforms/sms.py +377 -0
  243. package/gateway/platforms/telegram.py +4836 -0
  244. package/gateway/platforms/telegram_network.py +249 -0
  245. package/gateway/platforms/webhook.py +806 -0
  246. package/gateway/platforms/wecom.py +1610 -0
  247. package/gateway/platforms/wecom_callback.py +403 -0
  248. package/gateway/platforms/wecom_crypto.py +142 -0
  249. package/gateway/platforms/weixin.py +2170 -0
  250. package/gateway/platforms/whatsapp.py +1283 -0
  251. package/gateway/platforms/yuanbao.py +4873 -0
  252. package/gateway/platforms/yuanbao_media.py +645 -0
  253. package/gateway/platforms/yuanbao_proto.py +1209 -0
  254. package/gateway/platforms/yuanbao_sticker.py +558 -0
  255. package/gateway/restart.py +20 -0
  256. package/gateway/run.py +17074 -0
  257. package/gateway/runtime_footer.py +150 -0
  258. package/gateway/session.py +1399 -0
  259. package/gateway/session_context.py +156 -0
  260. package/gateway/shutdown_forensics.py +462 -0
  261. package/gateway/slash_access.py +229 -0
  262. package/gateway/status.py +972 -0
  263. package/gateway/sticker_cache.py +111 -0
  264. package/gateway/stream_consumer.py +1286 -0
  265. package/gateway/whatsapp_identity.py +156 -0
  266. package/hermes_cli/__init__.py +47 -0
  267. package/hermes_cli/__pycache__/__init__.cpython-312.pyc +0 -0
  268. package/hermes_cli/__pycache__/_parser.cpython-312.pyc +0 -0
  269. package/hermes_cli/__pycache__/auth.cpython-312.pyc +0 -0
  270. package/hermes_cli/__pycache__/banner.cpython-312.pyc +0 -0
  271. package/hermes_cli/__pycache__/browser_connect.cpython-312.pyc +0 -0
  272. package/hermes_cli/__pycache__/callbacks.cpython-312.pyc +0 -0
  273. package/hermes_cli/__pycache__/checkpoints.cpython-312.pyc +0 -0
  274. package/hermes_cli/__pycache__/cli_output.cpython-312.pyc +0 -0
  275. package/hermes_cli/__pycache__/codex_models.cpython-312.pyc +0 -0
  276. package/hermes_cli/__pycache__/codex_runtime_switch.cpython-312.pyc +0 -0
  277. package/hermes_cli/__pycache__/colors.cpython-312.pyc +0 -0
  278. package/hermes_cli/__pycache__/commands.cpython-312.pyc +0 -0
  279. package/hermes_cli/__pycache__/config.cpython-312.pyc +0 -0
  280. package/hermes_cli/__pycache__/copilot_auth.cpython-312.pyc +0 -0
  281. package/hermes_cli/__pycache__/curator.cpython-312.pyc +0 -0
  282. package/hermes_cli/__pycache__/curses_ui.cpython-312.pyc +0 -0
  283. package/hermes_cli/__pycache__/debug.cpython-312.pyc +0 -0
  284. package/hermes_cli/__pycache__/default_soul.cpython-312.pyc +0 -0
  285. package/hermes_cli/__pycache__/env_loader.cpython-312.pyc +0 -0
  286. package/hermes_cli/__pycache__/fallback_cmd.cpython-312.pyc +0 -0
  287. package/hermes_cli/__pycache__/gateway.cpython-312.pyc +0 -0
  288. package/hermes_cli/__pycache__/gateway_windows.cpython-312.pyc +0 -0
  289. package/hermes_cli/__pycache__/goals.cpython-312.pyc +0 -0
  290. package/hermes_cli/__pycache__/inventory.cpython-312.pyc +0 -0
  291. package/hermes_cli/__pycache__/kanban.cpython-312.pyc +0 -0
  292. package/hermes_cli/__pycache__/kanban_db.cpython-312.pyc +0 -0
  293. package/hermes_cli/__pycache__/main.cpython-312.pyc +0 -0
  294. package/hermes_cli/__pycache__/model_catalog.cpython-312.pyc +0 -0
  295. package/hermes_cli/__pycache__/model_normalize.cpython-312.pyc +0 -0
  296. package/hermes_cli/__pycache__/model_switch.cpython-312.pyc +0 -0
  297. package/hermes_cli/__pycache__/models.cpython-312.pyc +0 -0
  298. package/hermes_cli/__pycache__/nous_subscription.cpython-312.pyc +0 -0
  299. package/hermes_cli/__pycache__/pairing.cpython-312.pyc +0 -0
  300. package/hermes_cli/__pycache__/platforms.cpython-312.pyc +0 -0
  301. package/hermes_cli/__pycache__/plugins.cpython-312.pyc +0 -0
  302. package/hermes_cli/__pycache__/profiles.cpython-312.pyc +0 -0
  303. package/hermes_cli/__pycache__/providers.cpython-312.pyc +0 -0
  304. package/hermes_cli/__pycache__/pt_input_extras.cpython-312.pyc +0 -0
  305. package/hermes_cli/__pycache__/runtime_provider.cpython-312.pyc +0 -0
  306. package/hermes_cli/__pycache__/security_advisories.cpython-312.pyc +0 -0
  307. package/hermes_cli/__pycache__/setup.cpython-312.pyc +0 -0
  308. package/hermes_cli/__pycache__/skills_hub.cpython-312.pyc +0 -0
  309. package/hermes_cli/__pycache__/skin_engine.cpython-312.pyc +0 -0
  310. package/hermes_cli/__pycache__/stdio.cpython-312.pyc +0 -0
  311. package/hermes_cli/__pycache__/timeouts.cpython-312.pyc +0 -0
  312. package/hermes_cli/__pycache__/tips.cpython-312.pyc +0 -0
  313. package/hermes_cli/__pycache__/tools_config.cpython-312.pyc +0 -0
  314. package/hermes_cli/__pycache__/voice.cpython-312.pyc +0 -0
  315. package/hermes_cli/_parser.py +365 -0
  316. package/hermes_cli/_subprocess_compat.py +175 -0
  317. package/hermes_cli/auth.py +6299 -0
  318. package/hermes_cli/auth_commands.py +749 -0
  319. package/hermes_cli/azure_detect.py +300 -0
  320. package/hermes_cli/backup.py +938 -0
  321. package/hermes_cli/banner.py +703 -0
  322. package/hermes_cli/browser_connect.py +139 -0
  323. package/hermes_cli/callbacks.py +243 -0
  324. package/hermes_cli/checkpoints.py +244 -0
  325. package/hermes_cli/claw.py +810 -0
  326. package/hermes_cli/cli_output.py +78 -0
  327. package/hermes_cli/clipboard.py +495 -0
  328. package/hermes_cli/codex_models.py +198 -0
  329. package/hermes_cli/codex_runtime_plugin_migration.py +757 -0
  330. package/hermes_cli/codex_runtime_switch.py +266 -0
  331. package/hermes_cli/colors.py +38 -0
  332. package/hermes_cli/commands.py +1728 -0
  333. package/hermes_cli/completion.py +315 -0
  334. package/hermes_cli/config.py +5382 -0
  335. package/hermes_cli/copilot_auth.py +392 -0
  336. package/hermes_cli/cron.py +313 -0
  337. package/hermes_cli/curator.py +598 -0
  338. package/hermes_cli/curses_ui.py +472 -0
  339. package/hermes_cli/debug.py +747 -0
  340. package/hermes_cli/default_soul.py +11 -0
  341. package/hermes_cli/dep_ensure.py +107 -0
  342. package/hermes_cli/dingtalk_auth.py +293 -0
  343. package/hermes_cli/doctor.py +1863 -0
  344. package/hermes_cli/dump.py +326 -0
  345. package/hermes_cli/env_loader.py +175 -0
  346. package/hermes_cli/fallback_cmd.py +361 -0
  347. package/hermes_cli/gateway.py +5422 -0
  348. package/hermes_cli/gateway_windows.py +692 -0
  349. package/hermes_cli/goals.py +757 -0
  350. package/hermes_cli/hooks.py +385 -0
  351. package/hermes_cli/inventory.py +240 -0
  352. package/hermes_cli/kanban.py +2252 -0
  353. package/hermes_cli/kanban_db.py +4840 -0
  354. package/hermes_cli/kanban_diagnostics.py +776 -0
  355. package/hermes_cli/kanban_specify.py +266 -0
  356. package/hermes_cli/logs.py +391 -0
  357. package/hermes_cli/main.py +12396 -0
  358. package/hermes_cli/mcp_config.py +781 -0
  359. package/hermes_cli/memory_setup.py +465 -0
  360. package/hermes_cli/model_catalog.py +330 -0
  361. package/hermes_cli/model_normalize.py +473 -0
  362. package/hermes_cli/model_switch.py +1777 -0
  363. package/hermes_cli/models.py +3789 -0
  364. package/hermes_cli/nous_subscription.py +799 -0
  365. package/hermes_cli/oneshot.py +351 -0
  366. package/hermes_cli/pairing.py +115 -0
  367. package/hermes_cli/platforms.py +83 -0
  368. package/hermes_cli/plugins.py +1562 -0
  369. package/hermes_cli/plugins_cmd.py +1587 -0
  370. package/hermes_cli/profile_distribution.py +703 -0
  371. package/hermes_cli/profiles.py +1319 -0
  372. package/hermes_cli/providers.py +720 -0
  373. package/hermes_cli/proxy/__init__.py +20 -0
  374. package/hermes_cli/proxy/adapters/__init__.py +35 -0
  375. package/hermes_cli/proxy/adapters/base.py +94 -0
  376. package/hermes_cli/proxy/adapters/nous_portal.py +137 -0
  377. package/hermes_cli/proxy/cli.py +141 -0
  378. package/hermes_cli/proxy/server.py +265 -0
  379. package/hermes_cli/pt_input_extras.py +83 -0
  380. package/hermes_cli/pty_bridge.py +237 -0
  381. package/hermes_cli/relaunch.py +205 -0
  382. package/hermes_cli/runtime_provider.py +1428 -0
  383. package/hermes_cli/security_advisories.py +452 -0
  384. package/hermes_cli/setup.py +3559 -0
  385. package/hermes_cli/skills_config.py +177 -0
  386. package/hermes_cli/skills_hub.py +1595 -0
  387. package/hermes_cli/skin_engine.py +929 -0
  388. package/hermes_cli/slack_cli.py +160 -0
  389. package/hermes_cli/status.py +550 -0
  390. package/hermes_cli/stdio.py +252 -0
  391. package/hermes_cli/timeouts.py +82 -0
  392. package/hermes_cli/tips.py +487 -0
  393. package/hermes_cli/tools_config.py +3151 -0
  394. package/hermes_cli/uninstall.py +681 -0
  395. package/hermes_cli/vercel_auth.py +70 -0
  396. package/hermes_cli/voice.py +846 -0
  397. package/hermes_cli/web_server.py +4438 -0
  398. package/hermes_cli/webhook.py +275 -0
  399. package/locales/af.yaml +350 -0
  400. package/locales/de.yaml +350 -0
  401. package/locales/en.yaml +365 -0
  402. package/locales/es.yaml +350 -0
  403. package/locales/fr.yaml +350 -0
  404. package/locales/ga.yaml +354 -0
  405. package/locales/hu.yaml +350 -0
  406. package/locales/it.yaml +350 -0
  407. package/locales/ja.yaml +350 -0
  408. package/locales/ko.yaml +350 -0
  409. package/locales/pt.yaml +350 -0
  410. package/locales/ru.yaml +350 -0
  411. package/locales/tr.yaml +350 -0
  412. package/locales/uk.yaml +350 -0
  413. package/locales/zh-hant.yaml +350 -0
  414. package/locales/zh.yaml +350 -0
  415. package/mcp_serve.py +898 -0
  416. package/model_tools.py +899 -0
  417. package/optional-skills/DESCRIPTION.md +24 -0
  418. package/optional-skills/autonomous-ai-agents/DESCRIPTION.md +2 -0
  419. package/optional-skills/autonomous-ai-agents/blackbox/SKILL.md +144 -0
  420. package/optional-skills/autonomous-ai-agents/honcho/SKILL.md +431 -0
  421. package/optional-skills/blockchain/evm/SKILL.md +211 -0
  422. package/optional-skills/blockchain/evm/scripts/evm_client.py +1508 -0
  423. package/optional-skills/blockchain/hyperliquid/SKILL.md +211 -0
  424. package/optional-skills/blockchain/hyperliquid/scripts/hyperliquid_client.py +1660 -0
  425. package/optional-skills/blockchain/solana/SKILL.md +208 -0
  426. package/optional-skills/blockchain/solana/scripts/solana_client.py +698 -0
  427. package/optional-skills/communication/DESCRIPTION.md +1 -0
  428. package/optional-skills/communication/one-three-one-rule/SKILL.md +104 -0
  429. package/optional-skills/creative/blender-mcp/SKILL.md +117 -0
  430. package/optional-skills/creative/concept-diagrams/SKILL.md +362 -0
  431. package/optional-skills/creative/concept-diagrams/examples/apartment-floor-plan-conversion.md +244 -0
  432. package/optional-skills/creative/concept-diagrams/examples/automated-password-reset-flow.md +276 -0
  433. package/optional-skills/creative/concept-diagrams/examples/autonomous-llm-research-agent-flow.md +240 -0
  434. package/optional-skills/creative/concept-diagrams/examples/banana-journey-tree-to-smoothie.md +161 -0
  435. package/optional-skills/creative/concept-diagrams/examples/commercial-aircraft-structure.md +209 -0
  436. package/optional-skills/creative/concept-diagrams/examples/cpu-ooo-microarchitecture.md +236 -0
  437. package/optional-skills/creative/concept-diagrams/examples/electricity-grid-flow.md +182 -0
  438. package/optional-skills/creative/concept-diagrams/examples/feature-film-production-pipeline.md +172 -0
  439. package/optional-skills/creative/concept-diagrams/examples/hospital-emergency-department-flow.md +165 -0
  440. package/optional-skills/creative/concept-diagrams/examples/ml-benchmark-grouped-bar-chart.md +114 -0
  441. package/optional-skills/creative/concept-diagrams/examples/place-order-uml-sequence.md +325 -0
  442. package/optional-skills/creative/concept-diagrams/examples/smart-city-infrastructure.md +173 -0
  443. package/optional-skills/creative/concept-diagrams/examples/smartphone-layer-anatomy.md +154 -0
  444. package/optional-skills/creative/concept-diagrams/examples/sn2-reaction-mechanism.md +247 -0
  445. package/optional-skills/creative/concept-diagrams/examples/wind-turbine-structure.md +338 -0
  446. package/optional-skills/creative/concept-diagrams/references/dashboard-patterns.md +43 -0
  447. package/optional-skills/creative/concept-diagrams/references/infrastructure-patterns.md +144 -0
  448. package/optional-skills/creative/concept-diagrams/references/physical-shape-cookbook.md +42 -0
  449. package/optional-skills/creative/concept-diagrams/templates/template.html +174 -0
  450. package/optional-skills/creative/hyperframes/SKILL.md +191 -0
  451. package/optional-skills/creative/hyperframes/references/cli.md +185 -0
  452. package/optional-skills/creative/hyperframes/references/composition.md +129 -0
  453. package/optional-skills/creative/hyperframes/references/features.md +289 -0
  454. package/optional-skills/creative/hyperframes/references/gsap.md +136 -0
  455. package/optional-skills/creative/hyperframes/references/troubleshooting.md +137 -0
  456. package/optional-skills/creative/hyperframes/references/website-to-video.md +145 -0
  457. package/optional-skills/creative/hyperframes/scripts/setup.sh +135 -0
  458. package/optional-skills/creative/kanban-video-orchestrator/SKILL.md +207 -0
  459. package/optional-skills/creative/kanban-video-orchestrator/assets/brief.md.tmpl +79 -0
  460. package/optional-skills/creative/kanban-video-orchestrator/assets/setup.sh.tmpl +185 -0
  461. package/optional-skills/creative/kanban-video-orchestrator/assets/soul.md.tmpl +38 -0
  462. package/optional-skills/creative/kanban-video-orchestrator/references/examples.md +227 -0
  463. package/optional-skills/creative/kanban-video-orchestrator/references/intake.md +166 -0
  464. package/optional-skills/creative/kanban-video-orchestrator/references/kanban-setup.md +276 -0
  465. package/optional-skills/creative/kanban-video-orchestrator/references/monitoring.md +180 -0
  466. package/optional-skills/creative/kanban-video-orchestrator/references/role-archetypes.md +298 -0
  467. package/optional-skills/creative/kanban-video-orchestrator/references/tool-matrix.md +317 -0
  468. package/optional-skills/creative/kanban-video-orchestrator/scripts/bootstrap_pipeline.py +501 -0
  469. package/optional-skills/creative/kanban-video-orchestrator/scripts/monitor.py +195 -0
  470. package/optional-skills/creative/meme-generation/EXAMPLES.md +46 -0
  471. package/optional-skills/creative/meme-generation/SKILL.md +130 -0
  472. package/optional-skills/creative/meme-generation/scripts/generate_meme.py +471 -0
  473. package/optional-skills/creative/meme-generation/scripts/templates.json +97 -0
  474. package/optional-skills/devops/cli/SKILL.md +156 -0
  475. package/optional-skills/devops/cli/references/app-discovery.md +112 -0
  476. package/optional-skills/devops/cli/references/authentication.md +59 -0
  477. package/optional-skills/devops/cli/references/cli-reference.md +104 -0
  478. package/optional-skills/devops/cli/references/running-apps.md +171 -0
  479. package/optional-skills/devops/docker-management/SKILL.md +281 -0
  480. package/optional-skills/devops/pinggy-tunnel/SKILL.md +309 -0
  481. package/optional-skills/devops/watchers/SKILL.md +112 -0
  482. package/optional-skills/devops/watchers/scripts/_watermark.py +148 -0
  483. package/optional-skills/devops/watchers/scripts/watch_github.py +168 -0
  484. package/optional-skills/devops/watchers/scripts/watch_http_json.py +131 -0
  485. package/optional-skills/devops/watchers/scripts/watch_rss.py +121 -0
  486. package/optional-skills/dogfood/DESCRIPTION.md +3 -0
  487. package/optional-skills/dogfood/adversarial-ux-test/SKILL.md +191 -0
  488. package/optional-skills/email/agentmail/SKILL.md +126 -0
  489. package/optional-skills/finance/3-statement-model/SKILL.md +433 -0
  490. package/optional-skills/finance/3-statement-model/references/formatting.md +118 -0
  491. package/optional-skills/finance/3-statement-model/references/formulas.md +292 -0
  492. package/optional-skills/finance/3-statement-model/references/sec-filings.md +125 -0
  493. package/optional-skills/finance/comps-analysis/SKILL.md +662 -0
  494. package/optional-skills/finance/dcf-model/SKILL.md +1270 -0
  495. package/optional-skills/finance/dcf-model/TROUBLESHOOTING.md +40 -0
  496. package/optional-skills/finance/dcf-model/requirements.txt +7 -0
  497. package/optional-skills/finance/dcf-model/scripts/validate_dcf.py +292 -0
  498. package/optional-skills/finance/excel-author/SKILL.md +244 -0
  499. package/optional-skills/finance/excel-author/scripts/recalc.py +88 -0
  500. package/optional-skills/finance/lbo-model/SKILL.md +291 -0
  501. package/optional-skills/finance/merger-model/SKILL.md +144 -0
  502. package/optional-skills/finance/pptx-author/SKILL.md +173 -0
  503. package/optional-skills/finance/stocks/SKILL.md +95 -0
  504. package/optional-skills/finance/stocks/scripts/stocks_client.py +755 -0
  505. package/optional-skills/health/DESCRIPTION.md +1 -0
  506. package/optional-skills/health/fitness-nutrition/SKILL.md +256 -0
  507. package/optional-skills/health/fitness-nutrition/references/FORMULAS.md +100 -0
  508. package/optional-skills/health/fitness-nutrition/scripts/body_calc.py +210 -0
  509. package/optional-skills/health/fitness-nutrition/scripts/nutrition_search.py +86 -0
  510. package/optional-skills/health/neuroskill-bci/SKILL.md +459 -0
  511. package/optional-skills/health/neuroskill-bci/references/api.md +286 -0
  512. package/optional-skills/health/neuroskill-bci/references/metrics.md +220 -0
  513. package/optional-skills/health/neuroskill-bci/references/protocols.md +452 -0
  514. package/optional-skills/mcp/DESCRIPTION.md +3 -0
  515. package/optional-skills/mcp/fastmcp/SKILL.md +300 -0
  516. package/optional-skills/mcp/fastmcp/references/fastmcp-cli.md +110 -0
  517. package/optional-skills/mcp/fastmcp/scripts/scaffold_fastmcp.py +56 -0
  518. package/optional-skills/mcp/fastmcp/templates/api_wrapper.py +54 -0
  519. package/optional-skills/mcp/fastmcp/templates/database_server.py +77 -0
  520. package/optional-skills/mcp/fastmcp/templates/file_processor.py +55 -0
  521. package/optional-skills/mcp/mcporter/SKILL.md +123 -0
  522. package/optional-skills/migration/DESCRIPTION.md +2 -0
  523. package/optional-skills/migration/openclaw-migration/SKILL.md +298 -0
  524. package/optional-skills/migration/openclaw-migration/scripts/openclaw_to_hermes.py +3136 -0
  525. package/optional-skills/mlops/accelerate/SKILL.md +336 -0
  526. package/optional-skills/mlops/accelerate/references/custom-plugins.md +453 -0
  527. package/optional-skills/mlops/accelerate/references/megatron-integration.md +489 -0
  528. package/optional-skills/mlops/accelerate/references/performance.md +525 -0
  529. package/optional-skills/mlops/chroma/SKILL.md +410 -0
  530. package/optional-skills/mlops/chroma/references/integration.md +38 -0
  531. package/optional-skills/mlops/clip/SKILL.md +257 -0
  532. package/optional-skills/mlops/clip/references/applications.md +207 -0
  533. package/optional-skills/mlops/faiss/SKILL.md +225 -0
  534. package/optional-skills/mlops/faiss/references/index_types.md +280 -0
  535. package/optional-skills/mlops/flash-attention/SKILL.md +367 -0
  536. package/optional-skills/mlops/flash-attention/references/benchmarks.md +215 -0
  537. package/optional-skills/mlops/flash-attention/references/transformers-integration.md +293 -0
  538. package/optional-skills/mlops/guidance/SKILL.md +576 -0
  539. package/optional-skills/mlops/guidance/references/backends.md +554 -0
  540. package/optional-skills/mlops/guidance/references/constraints.md +674 -0
  541. package/optional-skills/mlops/guidance/references/examples.md +767 -0
  542. package/optional-skills/mlops/huggingface-tokenizers/SKILL.md +520 -0
  543. package/optional-skills/mlops/huggingface-tokenizers/references/algorithms.md +653 -0
  544. package/optional-skills/mlops/huggingface-tokenizers/references/integration.md +637 -0
  545. package/optional-skills/mlops/huggingface-tokenizers/references/pipeline.md +723 -0
  546. package/optional-skills/mlops/huggingface-tokenizers/references/training.md +565 -0
  547. package/optional-skills/mlops/inference/outlines/SKILL.md +656 -0
  548. package/optional-skills/mlops/inference/outlines/references/backends.md +615 -0
  549. package/optional-skills/mlops/inference/outlines/references/examples.md +773 -0
  550. package/optional-skills/mlops/inference/outlines/references/json_generation.md +652 -0
  551. package/optional-skills/mlops/instructor/SKILL.md +744 -0
  552. package/optional-skills/mlops/instructor/references/examples.md +107 -0
  553. package/optional-skills/mlops/instructor/references/providers.md +70 -0
  554. package/optional-skills/mlops/instructor/references/validation.md +606 -0
  555. package/optional-skills/mlops/lambda-labs/SKILL.md +549 -0
  556. package/optional-skills/mlops/lambda-labs/references/advanced-usage.md +611 -0
  557. package/optional-skills/mlops/lambda-labs/references/troubleshooting.md +530 -0
  558. package/optional-skills/mlops/llava/SKILL.md +308 -0
  559. package/optional-skills/mlops/llava/references/training.md +197 -0
  560. package/optional-skills/mlops/modal/SKILL.md +345 -0
  561. package/optional-skills/mlops/modal/references/advanced-usage.md +503 -0
  562. package/optional-skills/mlops/modal/references/troubleshooting.md +494 -0
  563. package/optional-skills/mlops/nemo-curator/SKILL.md +387 -0
  564. package/optional-skills/mlops/nemo-curator/references/deduplication.md +87 -0
  565. package/optional-skills/mlops/nemo-curator/references/filtering.md +102 -0
  566. package/optional-skills/mlops/peft/SKILL.md +435 -0
  567. package/optional-skills/mlops/peft/references/advanced-usage.md +514 -0
  568. package/optional-skills/mlops/peft/references/troubleshooting.md +480 -0
  569. package/optional-skills/mlops/pinecone/SKILL.md +362 -0
  570. package/optional-skills/mlops/pinecone/references/deployment.md +181 -0
  571. package/optional-skills/mlops/pytorch-fsdp/SKILL.md +130 -0
  572. package/optional-skills/mlops/pytorch-fsdp/references/index.md +7 -0
  573. package/optional-skills/mlops/pytorch-fsdp/references/other.md +4261 -0
  574. package/optional-skills/mlops/pytorch-lightning/SKILL.md +350 -0
  575. package/optional-skills/mlops/pytorch-lightning/references/callbacks.md +436 -0
  576. package/optional-skills/mlops/pytorch-lightning/references/distributed.md +490 -0
  577. package/optional-skills/mlops/pytorch-lightning/references/hyperparameter-tuning.md +556 -0
  578. package/optional-skills/mlops/qdrant/SKILL.md +497 -0
  579. package/optional-skills/mlops/qdrant/references/advanced-usage.md +648 -0
  580. package/optional-skills/mlops/qdrant/references/troubleshooting.md +631 -0
  581. package/optional-skills/mlops/saelens/SKILL.md +390 -0
  582. package/optional-skills/mlops/saelens/references/README.md +69 -0
  583. package/optional-skills/mlops/saelens/references/api.md +333 -0
  584. package/optional-skills/mlops/saelens/references/tutorials.md +318 -0
  585. package/optional-skills/mlops/simpo/SKILL.md +223 -0
  586. package/optional-skills/mlops/simpo/references/datasets.md +478 -0
  587. package/optional-skills/mlops/simpo/references/hyperparameters.md +452 -0
  588. package/optional-skills/mlops/simpo/references/loss-functions.md +350 -0
  589. package/optional-skills/mlops/slime/SKILL.md +468 -0
  590. package/optional-skills/mlops/slime/references/api-reference.md +392 -0
  591. package/optional-skills/mlops/slime/references/troubleshooting.md +386 -0
  592. package/optional-skills/mlops/stable-diffusion/SKILL.md +523 -0
  593. package/optional-skills/mlops/stable-diffusion/references/advanced-usage.md +716 -0
  594. package/optional-skills/mlops/stable-diffusion/references/troubleshooting.md +555 -0
  595. package/optional-skills/mlops/tensorrt-llm/SKILL.md +191 -0
  596. package/optional-skills/mlops/tensorrt-llm/references/multi-gpu.md +298 -0
  597. package/optional-skills/mlops/tensorrt-llm/references/optimization.md +242 -0
  598. package/optional-skills/mlops/tensorrt-llm/references/serving.md +470 -0
  599. package/optional-skills/mlops/torchtitan/SKILL.md +362 -0
  600. package/optional-skills/mlops/torchtitan/references/checkpoint.md +181 -0
  601. package/optional-skills/mlops/torchtitan/references/custom-models.md +258 -0
  602. package/optional-skills/mlops/torchtitan/references/float8.md +133 -0
  603. package/optional-skills/mlops/torchtitan/references/fsdp.md +126 -0
  604. package/optional-skills/mlops/training/axolotl/SKILL.md +166 -0
  605. package/optional-skills/mlops/training/axolotl/references/api.md +5548 -0
  606. package/optional-skills/mlops/training/axolotl/references/dataset-formats.md +1029 -0
  607. package/optional-skills/mlops/training/axolotl/references/index.md +15 -0
  608. package/optional-skills/mlops/training/axolotl/references/other.md +3563 -0
  609. package/optional-skills/mlops/training/trl-fine-tuning/SKILL.md +463 -0
  610. package/optional-skills/mlops/training/trl-fine-tuning/references/dpo-variants.md +227 -0
  611. package/optional-skills/mlops/training/trl-fine-tuning/references/grpo-training.md +504 -0
  612. package/optional-skills/mlops/training/trl-fine-tuning/references/online-rl.md +82 -0
  613. package/optional-skills/mlops/training/trl-fine-tuning/references/reward-modeling.md +122 -0
  614. package/optional-skills/mlops/training/trl-fine-tuning/references/sft-training.md +168 -0
  615. package/optional-skills/mlops/training/trl-fine-tuning/templates/basic_grpo_training.py +228 -0
  616. package/optional-skills/mlops/training/unsloth/SKILL.md +84 -0
  617. package/optional-skills/mlops/training/unsloth/references/index.md +7 -0
  618. package/optional-skills/mlops/training/unsloth/references/llms-full.md +16799 -0
  619. package/optional-skills/mlops/training/unsloth/references/llms-txt.md +12044 -0
  620. package/optional-skills/mlops/training/unsloth/references/llms.md +82 -0
  621. package/optional-skills/mlops/whisper/SKILL.md +321 -0
  622. package/optional-skills/mlops/whisper/references/languages.md +189 -0
  623. package/optional-skills/productivity/canvas/SKILL.md +98 -0
  624. package/optional-skills/productivity/canvas/scripts/canvas_api.py +157 -0
  625. package/optional-skills/productivity/here-now/SKILL.md +217 -0
  626. package/optional-skills/productivity/here-now/scripts/drive.sh +406 -0
  627. package/optional-skills/productivity/here-now/scripts/publish.sh +445 -0
  628. package/optional-skills/productivity/memento-flashcards/SKILL.md +324 -0
  629. package/optional-skills/productivity/memento-flashcards/scripts/memento_cards.py +353 -0
  630. package/optional-skills/productivity/memento-flashcards/scripts/youtube_quiz.py +88 -0
  631. package/optional-skills/productivity/shop-app/SKILL.md +340 -0
  632. package/optional-skills/productivity/shopify/SKILL.md +373 -0
  633. package/optional-skills/productivity/siyuan/SKILL.md +298 -0
  634. package/optional-skills/productivity/telephony/SKILL.md +418 -0
  635. package/optional-skills/productivity/telephony/scripts/telephony.py +1343 -0
  636. package/optional-skills/research/bioinformatics/SKILL.md +235 -0
  637. package/optional-skills/research/darwinian-evolver/SKILL.md +199 -0
  638. package/optional-skills/research/darwinian-evolver/scripts/parrot_openrouter.py +218 -0
  639. package/optional-skills/research/darwinian-evolver/scripts/show_snapshot.py +69 -0
  640. package/optional-skills/research/darwinian-evolver/templates/custom_problem_template.py +240 -0
  641. package/optional-skills/research/domain-intel/SKILL.md +97 -0
  642. package/optional-skills/research/domain-intel/scripts/domain_intel.py +397 -0
  643. package/optional-skills/research/drug-discovery/SKILL.md +227 -0
  644. package/optional-skills/research/drug-discovery/references/ADMET_REFERENCE.md +66 -0
  645. package/optional-skills/research/drug-discovery/scripts/chembl_target.py +53 -0
  646. package/optional-skills/research/drug-discovery/scripts/ro5_screen.py +44 -0
  647. package/optional-skills/research/duckduckgo-search/SKILL.md +238 -0
  648. package/optional-skills/research/duckduckgo-search/scripts/duckduckgo.sh +28 -0
  649. package/optional-skills/research/gitnexus-explorer/SKILL.md +214 -0
  650. package/optional-skills/research/gitnexus-explorer/scripts/proxy.mjs +92 -0
  651. package/optional-skills/research/osint-investigation/SKILL.md +277 -0
  652. package/optional-skills/research/osint-investigation/references/sources/courtlistener.md +98 -0
  653. package/optional-skills/research/osint-investigation/references/sources/gdelt.md +104 -0
  654. package/optional-skills/research/osint-investigation/references/sources/icij-offshore.md +104 -0
  655. package/optional-skills/research/osint-investigation/references/sources/nyc-acris.md +90 -0
  656. package/optional-skills/research/osint-investigation/references/sources/ofac-sdn.md +92 -0
  657. package/optional-skills/research/osint-investigation/references/sources/opencorporates.md +103 -0
  658. package/optional-skills/research/osint-investigation/references/sources/sec-edgar.md +83 -0
  659. package/optional-skills/research/osint-investigation/references/sources/senate-ld.md +89 -0
  660. package/optional-skills/research/osint-investigation/references/sources/usaspending.md +97 -0
  661. package/optional-skills/research/osint-investigation/references/sources/wayback.md +93 -0
  662. package/optional-skills/research/osint-investigation/references/sources/wikipedia.md +107 -0
  663. package/optional-skills/research/osint-investigation/scripts/_http.py +82 -0
  664. package/optional-skills/research/osint-investigation/scripts/_normalize.py +67 -0
  665. package/optional-skills/research/osint-investigation/scripts/build_findings.py +221 -0
  666. package/optional-skills/research/osint-investigation/scripts/entity_resolution.py +228 -0
  667. package/optional-skills/research/osint-investigation/scripts/fetch_courtlistener.py +149 -0
  668. package/optional-skills/research/osint-investigation/scripts/fetch_gdelt.py +162 -0
  669. package/optional-skills/research/osint-investigation/scripts/fetch_icij_offshore.py +234 -0
  670. package/optional-skills/research/osint-investigation/scripts/fetch_nyc_acris.py +203 -0
  671. package/optional-skills/research/osint-investigation/scripts/fetch_ofac_sdn.py +175 -0
  672. package/optional-skills/research/osint-investigation/scripts/fetch_opencorporates.py +192 -0
  673. package/optional-skills/research/osint-investigation/scripts/fetch_sec_edgar.py +184 -0
  674. package/optional-skills/research/osint-investigation/scripts/fetch_senate_ld.py +146 -0
  675. package/optional-skills/research/osint-investigation/scripts/fetch_usaspending.py +170 -0
  676. package/optional-skills/research/osint-investigation/scripts/fetch_wayback.py +142 -0
  677. package/optional-skills/research/osint-investigation/scripts/fetch_wikipedia.py +267 -0
  678. package/optional-skills/research/osint-investigation/scripts/timing_analysis.py +253 -0
  679. package/optional-skills/research/osint-investigation/templates/source-template.md +59 -0
  680. package/optional-skills/research/parallel-cli/SKILL.md +391 -0
  681. package/optional-skills/research/qmd/SKILL.md +441 -0
  682. package/optional-skills/research/scrapling/SKILL.md +336 -0
  683. package/optional-skills/research/searxng-search/SKILL.md +212 -0
  684. package/optional-skills/research/searxng-search/scripts/searxng.sh +22 -0
  685. package/optional-skills/security/1password/SKILL.md +163 -0
  686. package/optional-skills/security/1password/references/cli-examples.md +31 -0
  687. package/optional-skills/security/1password/references/get-started.md +21 -0
  688. package/optional-skills/security/DESCRIPTION.md +3 -0
  689. package/optional-skills/security/oss-forensics/SKILL.md +423 -0
  690. package/optional-skills/security/oss-forensics/references/evidence-types.md +89 -0
  691. package/optional-skills/security/oss-forensics/references/github-archive-guide.md +184 -0
  692. package/optional-skills/security/oss-forensics/references/investigation-templates.md +131 -0
  693. package/optional-skills/security/oss-forensics/references/recovery-techniques.md +164 -0
  694. package/optional-skills/security/oss-forensics/scripts/evidence-store.py +313 -0
  695. package/optional-skills/security/oss-forensics/templates/forensic-report.md +151 -0
  696. package/optional-skills/security/oss-forensics/templates/malicious-package-report.md +43 -0
  697. package/optional-skills/security/sherlock/SKILL.md +193 -0
  698. package/optional-skills/software-development/rest-graphql-debug/SKILL.md +514 -0
  699. package/optional-skills/web-development/DESCRIPTION.md +5 -0
  700. package/optional-skills/web-development/page-agent/SKILL.md +190 -0
  701. package/package.json +78 -0
  702. package/plugins/__init__.py +1 -0
  703. package/plugins/__pycache__/__init__.cpython-312.pyc +0 -0
  704. package/plugins/context_engine/__init__.py +219 -0
  705. package/plugins/disk-cleanup/README.md +51 -0
  706. package/plugins/disk-cleanup/__init__.py +316 -0
  707. package/plugins/disk-cleanup/disk_cleanup.py +497 -0
  708. package/plugins/disk-cleanup/plugin.yaml +7 -0
  709. package/plugins/example-dashboard/dashboard/manifest.json +14 -0
  710. package/plugins/example-dashboard/dashboard/plugin_api.py +17 -0
  711. package/plugins/google_meet/README.md +131 -0
  712. package/plugins/google_meet/SKILL.md +148 -0
  713. package/plugins/google_meet/__init__.py +103 -0
  714. package/plugins/google_meet/audio_bridge.py +244 -0
  715. package/plugins/google_meet/cli.py +479 -0
  716. package/plugins/google_meet/meet_bot.py +852 -0
  717. package/plugins/google_meet/node/__init__.py +54 -0
  718. package/plugins/google_meet/node/cli.py +125 -0
  719. package/plugins/google_meet/node/client.py +107 -0
  720. package/plugins/google_meet/node/protocol.py +124 -0
  721. package/plugins/google_meet/node/registry.py +113 -0
  722. package/plugins/google_meet/node/server.py +201 -0
  723. package/plugins/google_meet/plugin.yaml +16 -0
  724. package/plugins/google_meet/process_manager.py +324 -0
  725. package/plugins/google_meet/realtime/__init__.py +10 -0
  726. package/plugins/google_meet/realtime/openai_client.py +332 -0
  727. package/plugins/google_meet/tools.py +348 -0
  728. package/plugins/hermes-achievements/LICENSE +21 -0
  729. package/plugins/hermes-achievements/README.md +150 -0
  730. package/plugins/hermes-achievements/dashboard/dist/index.js +732 -0
  731. package/plugins/hermes-achievements/dashboard/dist/style.css +146 -0
  732. package/plugins/hermes-achievements/dashboard/manifest.json +11 -0
  733. package/plugins/hermes-achievements/dashboard/plugin_api.py +1062 -0
  734. package/plugins/hermes-achievements/docs/achievements-performance-implementation-plan.md +157 -0
  735. package/plugins/hermes-achievements/docs/achievements-performance-implementation-spec.md +219 -0
  736. package/plugins/hermes-achievements/docs/achievements-performance-spec.md +174 -0
  737. package/plugins/hermes-achievements/docs/assets/achievements-dashboard-hd.png +0 -0
  738. package/plugins/hermes-achievements/docs/assets/achievements-tier-showcase-hd.png +0 -0
  739. package/plugins/hermes-achievements/tests/test_achievement_engine.py +156 -0
  740. package/plugins/image_gen/openai/__init__.py +303 -0
  741. package/plugins/image_gen/openai/__pycache__/__init__.cpython-312.pyc +0 -0
  742. package/plugins/image_gen/openai/plugin.yaml +7 -0
  743. package/plugins/image_gen/openai-codex/__init__.py +378 -0
  744. package/plugins/image_gen/openai-codex/__pycache__/__init__.cpython-312.pyc +0 -0
  745. package/plugins/image_gen/openai-codex/plugin.yaml +5 -0
  746. package/plugins/image_gen/xai/__init__.py +316 -0
  747. package/plugins/image_gen/xai/__pycache__/__init__.cpython-312.pyc +0 -0
  748. package/plugins/image_gen/xai/plugin.yaml +7 -0
  749. package/plugins/kanban/dashboard/dist/index.js +3143 -0
  750. package/plugins/kanban/dashboard/dist/style.css +1500 -0
  751. package/plugins/kanban/dashboard/manifest.json +14 -0
  752. package/plugins/kanban/dashboard/plugin_api.py +1612 -0
  753. package/plugins/kanban/systemd/hermes-kanban-dispatcher.service +32 -0
  754. package/plugins/memory/__init__.py +408 -0
  755. package/plugins/memory/byterover/README.md +41 -0
  756. package/plugins/memory/byterover/__init__.py +384 -0
  757. package/plugins/memory/byterover/plugin.yaml +9 -0
  758. package/plugins/memory/hindsight/README.md +138 -0
  759. package/plugins/memory/hindsight/__init__.py +1758 -0
  760. package/plugins/memory/hindsight/plugin.yaml +8 -0
  761. package/plugins/memory/holographic/README.md +36 -0
  762. package/plugins/memory/holographic/__init__.py +409 -0
  763. package/plugins/memory/holographic/holographic.py +203 -0
  764. package/plugins/memory/holographic/plugin.yaml +5 -0
  765. package/plugins/memory/holographic/retrieval.py +593 -0
  766. package/plugins/memory/holographic/store.py +579 -0
  767. package/plugins/memory/honcho/README.md +328 -0
  768. package/plugins/memory/honcho/__init__.py +1329 -0
  769. package/plugins/memory/honcho/cli.py +1452 -0
  770. package/plugins/memory/honcho/client.py +784 -0
  771. package/plugins/memory/honcho/plugin.yaml +7 -0
  772. package/plugins/memory/honcho/session.py +1255 -0
  773. package/plugins/memory/mem0/README.md +38 -0
  774. package/plugins/memory/mem0/__init__.py +374 -0
  775. package/plugins/memory/mem0/plugin.yaml +5 -0
  776. package/plugins/memory/openviking/README.md +40 -0
  777. package/plugins/memory/openviking/__init__.py +945 -0
  778. package/plugins/memory/openviking/plugin.yaml +9 -0
  779. package/plugins/memory/retaindb/README.md +40 -0
  780. package/plugins/memory/retaindb/__init__.py +767 -0
  781. package/plugins/memory/retaindb/plugin.yaml +7 -0
  782. package/plugins/memory/supermemory/README.md +99 -0
  783. package/plugins/memory/supermemory/__init__.py +792 -0
  784. package/plugins/memory/supermemory/plugin.yaml +5 -0
  785. package/plugins/model-providers/README.md +70 -0
  786. package/plugins/model-providers/ai-gateway/__init__.py +43 -0
  787. package/plugins/model-providers/ai-gateway/__pycache__/__init__.cpython-312.pyc +0 -0
  788. package/plugins/model-providers/ai-gateway/plugin.yaml +5 -0
  789. package/plugins/model-providers/alibaba/__init__.py +13 -0
  790. package/plugins/model-providers/alibaba/__pycache__/__init__.cpython-312.pyc +0 -0
  791. package/plugins/model-providers/alibaba/plugin.yaml +5 -0
  792. package/plugins/model-providers/alibaba-coding-plan/__init__.py +21 -0
  793. package/plugins/model-providers/alibaba-coding-plan/__pycache__/__init__.cpython-312.pyc +0 -0
  794. package/plugins/model-providers/alibaba-coding-plan/plugin.yaml +5 -0
  795. package/plugins/model-providers/anthropic/__init__.py +52 -0
  796. package/plugins/model-providers/anthropic/__pycache__/__init__.cpython-312.pyc +0 -0
  797. package/plugins/model-providers/anthropic/plugin.yaml +5 -0
  798. package/plugins/model-providers/arcee/__init__.py +13 -0
  799. package/plugins/model-providers/arcee/__pycache__/__init__.cpython-312.pyc +0 -0
  800. package/plugins/model-providers/arcee/plugin.yaml +5 -0
  801. package/plugins/model-providers/azure-foundry/__init__.py +21 -0
  802. package/plugins/model-providers/azure-foundry/__pycache__/__init__.cpython-312.pyc +0 -0
  803. package/plugins/model-providers/azure-foundry/plugin.yaml +5 -0
  804. package/plugins/model-providers/bedrock/__init__.py +29 -0
  805. package/plugins/model-providers/bedrock/__pycache__/__init__.cpython-312.pyc +0 -0
  806. package/plugins/model-providers/bedrock/plugin.yaml +5 -0
  807. package/plugins/model-providers/copilot/__init__.py +58 -0
  808. package/plugins/model-providers/copilot/__pycache__/__init__.cpython-312.pyc +0 -0
  809. package/plugins/model-providers/copilot/plugin.yaml +5 -0
  810. package/plugins/model-providers/copilot-acp/__init__.py +34 -0
  811. package/plugins/model-providers/copilot-acp/__pycache__/__init__.cpython-312.pyc +0 -0
  812. package/plugins/model-providers/copilot-acp/plugin.yaml +5 -0
  813. package/plugins/model-providers/custom/__init__.py +68 -0
  814. package/plugins/model-providers/custom/__pycache__/__init__.cpython-312.pyc +0 -0
  815. package/plugins/model-providers/custom/plugin.yaml +5 -0
  816. package/plugins/model-providers/deepseek/__init__.py +99 -0
  817. package/plugins/model-providers/deepseek/__pycache__/__init__.cpython-312.pyc +0 -0
  818. package/plugins/model-providers/deepseek/plugin.yaml +5 -0
  819. package/plugins/model-providers/gemini/__init__.py +72 -0
  820. package/plugins/model-providers/gemini/__pycache__/__init__.cpython-312.pyc +0 -0
  821. package/plugins/model-providers/gemini/plugin.yaml +5 -0
  822. package/plugins/model-providers/gmi/__init__.py +31 -0
  823. package/plugins/model-providers/gmi/__pycache__/__init__.cpython-312.pyc +0 -0
  824. package/plugins/model-providers/gmi/plugin.yaml +5 -0
  825. package/plugins/model-providers/huggingface/__init__.py +20 -0
  826. package/plugins/model-providers/huggingface/__pycache__/__init__.cpython-312.pyc +0 -0
  827. package/plugins/model-providers/huggingface/plugin.yaml +5 -0
  828. package/plugins/model-providers/kilocode/__init__.py +14 -0
  829. package/plugins/model-providers/kilocode/__pycache__/__init__.cpython-312.pyc +0 -0
  830. package/plugins/model-providers/kilocode/plugin.yaml +5 -0
  831. package/plugins/model-providers/kimi-coding/__init__.py +71 -0
  832. package/plugins/model-providers/kimi-coding/__pycache__/__init__.cpython-312.pyc +0 -0
  833. package/plugins/model-providers/kimi-coding/plugin.yaml +5 -0
  834. package/plugins/model-providers/minimax/__init__.py +45 -0
  835. package/plugins/model-providers/minimax/__pycache__/__init__.cpython-312.pyc +0 -0
  836. package/plugins/model-providers/minimax/plugin.yaml +5 -0
  837. package/plugins/model-providers/nous/__init__.py +54 -0
  838. package/plugins/model-providers/nous/__pycache__/__init__.cpython-312.pyc +0 -0
  839. package/plugins/model-providers/nous/plugin.yaml +5 -0
  840. package/plugins/model-providers/novita/__init__.py +27 -0
  841. package/plugins/model-providers/novita/__pycache__/__init__.cpython-312.pyc +0 -0
  842. package/plugins/model-providers/novita/plugin.yaml +5 -0
  843. package/plugins/model-providers/nvidia/__init__.py +21 -0
  844. package/plugins/model-providers/nvidia/__pycache__/__init__.cpython-312.pyc +0 -0
  845. package/plugins/model-providers/nvidia/plugin.yaml +5 -0
  846. package/plugins/model-providers/ollama-cloud/__init__.py +14 -0
  847. package/plugins/model-providers/ollama-cloud/__pycache__/__init__.cpython-312.pyc +0 -0
  848. package/plugins/model-providers/ollama-cloud/plugin.yaml +5 -0
  849. package/plugins/model-providers/openai-codex/__init__.py +15 -0
  850. package/plugins/model-providers/openai-codex/__pycache__/__init__.cpython-312.pyc +0 -0
  851. package/plugins/model-providers/openai-codex/plugin.yaml +5 -0
  852. package/plugins/model-providers/opencode-zen/__init__.py +30 -0
  853. package/plugins/model-providers/opencode-zen/__pycache__/__init__.cpython-312.pyc +0 -0
  854. package/plugins/model-providers/opencode-zen/plugin.yaml +5 -0
  855. package/plugins/model-providers/openrouter/__init__.py +115 -0
  856. package/plugins/model-providers/openrouter/__pycache__/__init__.cpython-312.pyc +0 -0
  857. package/plugins/model-providers/openrouter/plugin.yaml +5 -0
  858. package/plugins/model-providers/qwen-oauth/__init__.py +82 -0
  859. package/plugins/model-providers/qwen-oauth/__pycache__/__init__.cpython-312.pyc +0 -0
  860. package/plugins/model-providers/qwen-oauth/plugin.yaml +5 -0
  861. package/plugins/model-providers/stepfun/__init__.py +14 -0
  862. package/plugins/model-providers/stepfun/__pycache__/__init__.cpython-312.pyc +0 -0
  863. package/plugins/model-providers/stepfun/plugin.yaml +5 -0
  864. package/plugins/model-providers/xai/__init__.py +15 -0
  865. package/plugins/model-providers/xai/__pycache__/__init__.cpython-312.pyc +0 -0
  866. package/plugins/model-providers/xai/plugin.yaml +5 -0
  867. package/plugins/model-providers/xiaomi/__init__.py +14 -0
  868. package/plugins/model-providers/xiaomi/__pycache__/__init__.cpython-312.pyc +0 -0
  869. package/plugins/model-providers/xiaomi/plugin.yaml +5 -0
  870. package/plugins/model-providers/zai/__init__.py +21 -0
  871. package/plugins/model-providers/zai/__pycache__/__init__.cpython-312.pyc +0 -0
  872. package/plugins/model-providers/zai/plugin.yaml +5 -0
  873. package/plugins/observability/langfuse/README.md +53 -0
  874. package/plugins/observability/langfuse/__init__.py +1004 -0
  875. package/plugins/observability/langfuse/plugin.yaml +14 -0
  876. package/plugins/platforms/google_chat/__init__.py +3 -0
  877. package/plugins/platforms/google_chat/__pycache__/__init__.cpython-312.pyc +0 -0
  878. package/plugins/platforms/google_chat/__pycache__/adapter.cpython-312.pyc +0 -0
  879. package/plugins/platforms/google_chat/adapter.py +3343 -0
  880. package/plugins/platforms/google_chat/oauth.py +639 -0
  881. package/plugins/platforms/google_chat/plugin.yaml +39 -0
  882. package/plugins/platforms/irc/__init__.py +3 -0
  883. package/plugins/platforms/irc/__pycache__/__init__.cpython-312.pyc +0 -0
  884. package/plugins/platforms/irc/__pycache__/adapter.cpython-312.pyc +0 -0
  885. package/plugins/platforms/irc/adapter.py +969 -0
  886. package/plugins/platforms/irc/plugin.yaml +54 -0
  887. package/plugins/platforms/line/__init__.py +3 -0
  888. package/plugins/platforms/line/__pycache__/__init__.cpython-312.pyc +0 -0
  889. package/plugins/platforms/line/__pycache__/adapter.cpython-312.pyc +0 -0
  890. package/plugins/platforms/line/adapter.py +1639 -0
  891. package/plugins/platforms/line/plugin.yaml +65 -0
  892. package/plugins/platforms/simplex/__init__.py +3 -0
  893. package/plugins/platforms/simplex/__pycache__/__init__.cpython-312.pyc +0 -0
  894. package/plugins/platforms/simplex/__pycache__/adapter.cpython-312.pyc +0 -0
  895. package/plugins/platforms/simplex/adapter.py +746 -0
  896. package/plugins/platforms/simplex/plugin.yaml +37 -0
  897. package/plugins/platforms/teams/__init__.py +3 -0
  898. package/plugins/platforms/teams/__pycache__/__init__.cpython-312.pyc +0 -0
  899. package/plugins/platforms/teams/__pycache__/adapter.cpython-312.pyc +0 -0
  900. package/plugins/platforms/teams/adapter.py +1188 -0
  901. package/plugins/platforms/teams/plugin.yaml +48 -0
  902. package/plugins/spotify/__init__.py +66 -0
  903. package/plugins/spotify/__pycache__/__init__.cpython-312.pyc +0 -0
  904. package/plugins/spotify/__pycache__/client.cpython-312.pyc +0 -0
  905. package/plugins/spotify/__pycache__/tools.cpython-312.pyc +0 -0
  906. package/plugins/spotify/client.py +435 -0
  907. package/plugins/spotify/plugin.yaml +13 -0
  908. package/plugins/spotify/tools.py +454 -0
  909. package/plugins/teams_pipeline/__init__.py +23 -0
  910. package/plugins/teams_pipeline/cli.py +463 -0
  911. package/plugins/teams_pipeline/meetings.py +333 -0
  912. package/plugins/teams_pipeline/models.py +350 -0
  913. package/plugins/teams_pipeline/pipeline.py +692 -0
  914. package/plugins/teams_pipeline/plugin.yaml +9 -0
  915. package/plugins/teams_pipeline/runtime.py +135 -0
  916. package/plugins/teams_pipeline/store.py +194 -0
  917. package/plugins/teams_pipeline/subscriptions.py +249 -0
  918. package/plugins/video_gen/fal/__init__.py +523 -0
  919. package/plugins/video_gen/fal/__pycache__/__init__.cpython-312.pyc +0 -0
  920. package/plugins/video_gen/fal/plugin.yaml +7 -0
  921. package/plugins/video_gen/xai/__init__.py +441 -0
  922. package/plugins/video_gen/xai/__pycache__/__init__.cpython-312.pyc +0 -0
  923. package/plugins/video_gen/xai/plugin.yaml +7 -0
  924. package/plugins/web/__init__.py +7 -0
  925. package/plugins/web/__pycache__/__init__.cpython-312.pyc +0 -0
  926. package/plugins/web/brave_free/__init__.py +14 -0
  927. package/plugins/web/brave_free/__pycache__/__init__.cpython-312.pyc +0 -0
  928. package/plugins/web/brave_free/__pycache__/provider.cpython-312.pyc +0 -0
  929. package/plugins/web/brave_free/plugin.yaml +7 -0
  930. package/plugins/web/brave_free/provider.py +137 -0
  931. package/plugins/web/ddgs/__init__.py +15 -0
  932. package/plugins/web/ddgs/__pycache__/__init__.cpython-312.pyc +0 -0
  933. package/plugins/web/ddgs/__pycache__/provider.cpython-312.pyc +0 -0
  934. package/plugins/web/ddgs/plugin.yaml +7 -0
  935. package/plugins/web/ddgs/provider.py +104 -0
  936. package/plugins/web/exa/__init__.py +15 -0
  937. package/plugins/web/exa/__pycache__/__init__.cpython-312.pyc +0 -0
  938. package/plugins/web/exa/__pycache__/provider.cpython-312.pyc +0 -0
  939. package/plugins/web/exa/plugin.yaml +7 -0
  940. package/plugins/web/exa/provider.py +212 -0
  941. package/plugins/web/firecrawl/__init__.py +28 -0
  942. package/plugins/web/firecrawl/__pycache__/__init__.cpython-312.pyc +0 -0
  943. package/plugins/web/firecrawl/__pycache__/provider.cpython-312.pyc +0 -0
  944. package/plugins/web/firecrawl/plugin.yaml +7 -0
  945. package/plugins/web/firecrawl/provider.py +773 -0
  946. package/plugins/web/parallel/__init__.py +16 -0
  947. package/plugins/web/parallel/__pycache__/__init__.cpython-312.pyc +0 -0
  948. package/plugins/web/parallel/__pycache__/provider.cpython-312.pyc +0 -0
  949. package/plugins/web/parallel/plugin.yaml +7 -0
  950. package/plugins/web/parallel/provider.py +291 -0
  951. package/plugins/web/searxng/__init__.py +15 -0
  952. package/plugins/web/searxng/__pycache__/__init__.cpython-312.pyc +0 -0
  953. package/plugins/web/searxng/__pycache__/provider.cpython-312.pyc +0 -0
  954. package/plugins/web/searxng/plugin.yaml +7 -0
  955. package/plugins/web/searxng/provider.py +140 -0
  956. package/plugins/web/tavily/__init__.py +15 -0
  957. package/plugins/web/tavily/__pycache__/__init__.cpython-312.pyc +0 -0
  958. package/plugins/web/tavily/__pycache__/provider.cpython-312.pyc +0 -0
  959. package/plugins/web/tavily/plugin.yaml +7 -0
  960. package/plugins/web/tavily/provider.py +285 -0
  961. package/providers/README.md +78 -0
  962. package/providers/__init__.py +192 -0
  963. package/providers/__pycache__/__init__.cpython-312.pyc +0 -0
  964. package/providers/__pycache__/base.cpython-312.pyc +0 -0
  965. package/providers/base.py +184 -0
  966. package/pyproject.toml +255 -0
  967. package/run_agent.py +16409 -0
  968. package/scripts/benchmark_browser_eval.py +138 -0
  969. package/scripts/build_model_catalog.py +95 -0
  970. package/scripts/build_skills_index.py +325 -0
  971. package/scripts/check-windows-footguns.py +624 -0
  972. package/scripts/contributor_audit.py +473 -0
  973. package/scripts/discord-voice-doctor.py +396 -0
  974. package/scripts/hermes-gateway +416 -0
  975. package/scripts/install.cmd +28 -0
  976. package/scripts/install.ps1 +1611 -0
  977. package/scripts/install.sh +2007 -0
  978. package/scripts/install_psutil_android.py +117 -0
  979. package/scripts/keystroke_diagnostic.py +81 -0
  980. package/scripts/kill_modal.sh +34 -0
  981. package/scripts/lib/node-bootstrap.sh +238 -0
  982. package/scripts/lint_diff.py +207 -0
  983. package/scripts/postinstall.js +150 -0
  984. package/scripts/profile-tui.py +626 -0
  985. package/scripts/release.py +1680 -0
  986. package/scripts/run_tests.sh +129 -0
  987. package/scripts/sample_and_compress.py +409 -0
  988. package/scripts/setup_open_webui.sh +349 -0
  989. package/scripts/whatsapp-bridge/allowlist.js +88 -0
  990. package/scripts/whatsapp-bridge/allowlist.test.mjs +80 -0
  991. package/scripts/whatsapp-bridge/bridge.js +729 -0
  992. package/scripts/whatsapp-bridge/package-lock.json +2141 -0
  993. package/scripts/whatsapp-bridge/package.json +19 -0
  994. package/skills/apple/DESCRIPTION.md +2 -0
  995. package/skills/apple/apple-notes/SKILL.md +90 -0
  996. package/skills/apple/apple-reminders/SKILL.md +98 -0
  997. package/skills/apple/findmy/SKILL.md +131 -0
  998. package/skills/apple/imessage/SKILL.md +102 -0
  999. package/skills/apple/macos-computer-use/SKILL.md +201 -0
  1000. package/skills/autonomous-ai-agents/DESCRIPTION.md +3 -0
  1001. package/skills/autonomous-ai-agents/claude-code/SKILL.md +745 -0
  1002. package/skills/autonomous-ai-agents/codex/SKILL.md +130 -0
  1003. package/skills/autonomous-ai-agents/hermes-agent/SKILL.md +1014 -0
  1004. package/skills/autonomous-ai-agents/opencode/SKILL.md +219 -0
  1005. package/skills/creative/DESCRIPTION.md +3 -0
  1006. package/skills/creative/architecture-diagram/SKILL.md +148 -0
  1007. package/skills/creative/architecture-diagram/templates/template.html +319 -0
  1008. package/skills/creative/ascii-art/SKILL.md +322 -0
  1009. package/skills/creative/ascii-video/README.md +290 -0
  1010. package/skills/creative/ascii-video/SKILL.md +241 -0
  1011. package/skills/creative/ascii-video/references/architecture.md +802 -0
  1012. package/skills/creative/ascii-video/references/composition.md +892 -0
  1013. package/skills/creative/ascii-video/references/effects.md +1865 -0
  1014. package/skills/creative/ascii-video/references/inputs.md +685 -0
  1015. package/skills/creative/ascii-video/references/optimization.md +688 -0
  1016. package/skills/creative/ascii-video/references/scenes.md +1011 -0
  1017. package/skills/creative/ascii-video/references/shaders.md +1385 -0
  1018. package/skills/creative/ascii-video/references/troubleshooting.md +367 -0
  1019. package/skills/creative/baoyu-comic/PORT_NOTES.md +77 -0
  1020. package/skills/creative/baoyu-comic/SKILL.md +247 -0
  1021. package/skills/creative/baoyu-comic/references/analysis-framework.md +176 -0
  1022. package/skills/creative/baoyu-comic/references/art-styles/chalk.md +101 -0
  1023. package/skills/creative/baoyu-comic/references/art-styles/ink-brush.md +97 -0
  1024. package/skills/creative/baoyu-comic/references/art-styles/ligne-claire.md +75 -0
  1025. package/skills/creative/baoyu-comic/references/art-styles/manga.md +93 -0
  1026. package/skills/creative/baoyu-comic/references/art-styles/minimalist.md +84 -0
  1027. package/skills/creative/baoyu-comic/references/art-styles/realistic.md +89 -0
  1028. package/skills/creative/baoyu-comic/references/auto-selection.md +71 -0
  1029. package/skills/creative/baoyu-comic/references/base-prompt.md +98 -0
  1030. package/skills/creative/baoyu-comic/references/character-template.md +180 -0
  1031. package/skills/creative/baoyu-comic/references/layouts/cinematic.md +23 -0
  1032. package/skills/creative/baoyu-comic/references/layouts/dense.md +23 -0
  1033. package/skills/creative/baoyu-comic/references/layouts/four-panel.md +40 -0
  1034. package/skills/creative/baoyu-comic/references/layouts/mixed.md +23 -0
  1035. package/skills/creative/baoyu-comic/references/layouts/splash.md +23 -0
  1036. package/skills/creative/baoyu-comic/references/layouts/standard.md +23 -0
  1037. package/skills/creative/baoyu-comic/references/layouts/webtoon.md +30 -0
  1038. package/skills/creative/baoyu-comic/references/ohmsha-guide.md +85 -0
  1039. package/skills/creative/baoyu-comic/references/partial-workflows.md +106 -0
  1040. package/skills/creative/baoyu-comic/references/presets/concept-story.md +121 -0
  1041. package/skills/creative/baoyu-comic/references/presets/four-panel.md +107 -0
  1042. package/skills/creative/baoyu-comic/references/presets/ohmsha.md +114 -0
  1043. package/skills/creative/baoyu-comic/references/presets/shoujo.md +116 -0
  1044. package/skills/creative/baoyu-comic/references/presets/wuxia.md +110 -0
  1045. package/skills/creative/baoyu-comic/references/storyboard-template.md +143 -0
  1046. package/skills/creative/baoyu-comic/references/tones/action.md +110 -0
  1047. package/skills/creative/baoyu-comic/references/tones/dramatic.md +95 -0
  1048. package/skills/creative/baoyu-comic/references/tones/energetic.md +105 -0
  1049. package/skills/creative/baoyu-comic/references/tones/neutral.md +63 -0
  1050. package/skills/creative/baoyu-comic/references/tones/romantic.md +100 -0
  1051. package/skills/creative/baoyu-comic/references/tones/vintage.md +104 -0
  1052. package/skills/creative/baoyu-comic/references/tones/warm.md +94 -0
  1053. package/skills/creative/baoyu-comic/references/workflow.md +401 -0
  1054. package/skills/creative/baoyu-infographic/PORT_NOTES.md +43 -0
  1055. package/skills/creative/baoyu-infographic/SKILL.md +237 -0
  1056. package/skills/creative/baoyu-infographic/references/analysis-framework.md +182 -0
  1057. package/skills/creative/baoyu-infographic/references/base-prompt.md +43 -0
  1058. package/skills/creative/baoyu-infographic/references/layouts/bento-grid.md +41 -0
  1059. package/skills/creative/baoyu-infographic/references/layouts/binary-comparison.md +48 -0
  1060. package/skills/creative/baoyu-infographic/references/layouts/bridge.md +41 -0
  1061. package/skills/creative/baoyu-infographic/references/layouts/circular-flow.md +41 -0
  1062. package/skills/creative/baoyu-infographic/references/layouts/comic-strip.md +41 -0
  1063. package/skills/creative/baoyu-infographic/references/layouts/comparison-matrix.md +41 -0
  1064. package/skills/creative/baoyu-infographic/references/layouts/dashboard.md +41 -0
  1065. package/skills/creative/baoyu-infographic/references/layouts/dense-modules.md +72 -0
  1066. package/skills/creative/baoyu-infographic/references/layouts/funnel.md +41 -0
  1067. package/skills/creative/baoyu-infographic/references/layouts/hierarchical-layers.md +48 -0
  1068. package/skills/creative/baoyu-infographic/references/layouts/hub-spoke.md +41 -0
  1069. package/skills/creative/baoyu-infographic/references/layouts/iceberg.md +41 -0
  1070. package/skills/creative/baoyu-infographic/references/layouts/isometric-map.md +41 -0
  1071. package/skills/creative/baoyu-infographic/references/layouts/jigsaw.md +41 -0
  1072. package/skills/creative/baoyu-infographic/references/layouts/linear-progression.md +48 -0
  1073. package/skills/creative/baoyu-infographic/references/layouts/periodic-table.md +41 -0
  1074. package/skills/creative/baoyu-infographic/references/layouts/story-mountain.md +41 -0
  1075. package/skills/creative/baoyu-infographic/references/layouts/structural-breakdown.md +48 -0
  1076. package/skills/creative/baoyu-infographic/references/layouts/tree-branching.md +41 -0
  1077. package/skills/creative/baoyu-infographic/references/layouts/venn-diagram.md +41 -0
  1078. package/skills/creative/baoyu-infographic/references/layouts/winding-roadmap.md +41 -0
  1079. package/skills/creative/baoyu-infographic/references/structured-content-template.md +244 -0
  1080. package/skills/creative/baoyu-infographic/references/styles/aged-academia.md +36 -0
  1081. package/skills/creative/baoyu-infographic/references/styles/bold-graphic.md +36 -0
  1082. package/skills/creative/baoyu-infographic/references/styles/chalkboard.md +61 -0
  1083. package/skills/creative/baoyu-infographic/references/styles/claymation.md +29 -0
  1084. package/skills/creative/baoyu-infographic/references/styles/corporate-memphis.md +29 -0
  1085. package/skills/creative/baoyu-infographic/references/styles/craft-handmade.md +44 -0
  1086. package/skills/creative/baoyu-infographic/references/styles/cyberpunk-neon.md +29 -0
  1087. package/skills/creative/baoyu-infographic/references/styles/hand-drawn-edu.md +63 -0
  1088. package/skills/creative/baoyu-infographic/references/styles/ikea-manual.md +29 -0
  1089. package/skills/creative/baoyu-infographic/references/styles/kawaii.md +29 -0
  1090. package/skills/creative/baoyu-infographic/references/styles/knolling.md +29 -0
  1091. package/skills/creative/baoyu-infographic/references/styles/lego-brick.md +29 -0
  1092. package/skills/creative/baoyu-infographic/references/styles/morandi-journal.md +60 -0
  1093. package/skills/creative/baoyu-infographic/references/styles/origami.md +29 -0
  1094. package/skills/creative/baoyu-infographic/references/styles/pixel-art.md +29 -0
  1095. package/skills/creative/baoyu-infographic/references/styles/pop-laboratory.md +48 -0
  1096. package/skills/creative/baoyu-infographic/references/styles/retro-pop-grid.md +47 -0
  1097. package/skills/creative/baoyu-infographic/references/styles/storybook-watercolor.md +29 -0
  1098. package/skills/creative/baoyu-infographic/references/styles/subway-map.md +29 -0
  1099. package/skills/creative/baoyu-infographic/references/styles/technical-schematic.md +36 -0
  1100. package/skills/creative/baoyu-infographic/references/styles/ui-wireframe.md +29 -0
  1101. package/skills/creative/claude-design/SKILL.md +591 -0
  1102. package/skills/creative/comfyui/SKILL.md +612 -0
  1103. package/skills/creative/comfyui/references/official-cli.md +255 -0
  1104. package/skills/creative/comfyui/references/rest-api.md +312 -0
  1105. package/skills/creative/comfyui/references/template-integrity.md +243 -0
  1106. package/skills/creative/comfyui/references/workflow-format.md +226 -0
  1107. package/skills/creative/comfyui/scripts/_common.py +835 -0
  1108. package/skills/creative/comfyui/scripts/auto_fix_deps.py +225 -0
  1109. package/skills/creative/comfyui/scripts/check_deps.py +437 -0
  1110. package/skills/creative/comfyui/scripts/comfyui_setup.sh +286 -0
  1111. package/skills/creative/comfyui/scripts/extract_schema.py +315 -0
  1112. package/skills/creative/comfyui/scripts/fetch_logs.py +158 -0
  1113. package/skills/creative/comfyui/scripts/hardware_check.py +497 -0
  1114. package/skills/creative/comfyui/scripts/health_check.py +223 -0
  1115. package/skills/creative/comfyui/scripts/run_batch.py +243 -0
  1116. package/skills/creative/comfyui/scripts/run_workflow.py +796 -0
  1117. package/skills/creative/comfyui/scripts/ws_monitor.py +267 -0
  1118. package/skills/creative/comfyui/tests/README.md +50 -0
  1119. package/skills/creative/comfyui/tests/conftest.py +64 -0
  1120. package/skills/creative/comfyui/tests/pytest.ini +5 -0
  1121. package/skills/creative/comfyui/tests/test_check_deps.py +68 -0
  1122. package/skills/creative/comfyui/tests/test_cloud_integration.py +95 -0
  1123. package/skills/creative/comfyui/tests/test_common.py +447 -0
  1124. package/skills/creative/comfyui/tests/test_extract_schema.py +185 -0
  1125. package/skills/creative/comfyui/tests/test_run_workflow.py +213 -0
  1126. package/skills/creative/comfyui/workflows/README.md +86 -0
  1127. package/skills/creative/comfyui/workflows/animatediff_video.json +64 -0
  1128. package/skills/creative/comfyui/workflows/flux_dev_txt2img.json +78 -0
  1129. package/skills/creative/comfyui/workflows/sd15_txt2img.json +49 -0
  1130. package/skills/creative/comfyui/workflows/sdxl_img2img.json +54 -0
  1131. package/skills/creative/comfyui/workflows/sdxl_inpaint.json +59 -0
  1132. package/skills/creative/comfyui/workflows/sdxl_txt2img.json +49 -0
  1133. package/skills/creative/comfyui/workflows/upscale_4x.json +27 -0
  1134. package/skills/creative/comfyui/workflows/wan_video_t2v.json +69 -0
  1135. package/skills/creative/creative-ideation/SKILL.md +152 -0
  1136. package/skills/creative/creative-ideation/references/full-prompt-library.md +110 -0
  1137. package/skills/creative/design-md/SKILL.md +199 -0
  1138. package/skills/creative/design-md/templates/starter.md +99 -0
  1139. package/skills/creative/excalidraw/SKILL.md +199 -0
  1140. package/skills/creative/excalidraw/references/colors.md +44 -0
  1141. package/skills/creative/excalidraw/references/dark-mode.md +68 -0
  1142. package/skills/creative/excalidraw/references/examples.md +141 -0
  1143. package/skills/creative/excalidraw/scripts/upload.py +133 -0
  1144. package/skills/creative/humanizer/LICENSE +21 -0
  1145. package/skills/creative/humanizer/SKILL.md +578 -0
  1146. package/skills/creative/manim-video/README.md +23 -0
  1147. package/skills/creative/manim-video/SKILL.md +269 -0
  1148. package/skills/creative/manim-video/references/animation-design-thinking.md +161 -0
  1149. package/skills/creative/manim-video/references/animations.md +282 -0
  1150. package/skills/creative/manim-video/references/camera-and-3d.md +135 -0
  1151. package/skills/creative/manim-video/references/decorations.md +202 -0
  1152. package/skills/creative/manim-video/references/equations.md +216 -0
  1153. package/skills/creative/manim-video/references/graphs-and-data.md +163 -0
  1154. package/skills/creative/manim-video/references/mobjects.md +333 -0
  1155. package/skills/creative/manim-video/references/paper-explainer.md +255 -0
  1156. package/skills/creative/manim-video/references/production-quality.md +190 -0
  1157. package/skills/creative/manim-video/references/rendering.md +185 -0
  1158. package/skills/creative/manim-video/references/scene-planning.md +118 -0
  1159. package/skills/creative/manim-video/references/troubleshooting.md +135 -0
  1160. package/skills/creative/manim-video/references/updaters-and-trackers.md +260 -0
  1161. package/skills/creative/manim-video/references/visual-design.md +124 -0
  1162. package/skills/creative/manim-video/scripts/setup.sh +14 -0
  1163. package/skills/creative/p5js/README.md +64 -0
  1164. package/skills/creative/p5js/SKILL.md +556 -0
  1165. package/skills/creative/p5js/references/animation.md +439 -0
  1166. package/skills/creative/p5js/references/color-systems.md +352 -0
  1167. package/skills/creative/p5js/references/core-api.md +410 -0
  1168. package/skills/creative/p5js/references/export-pipeline.md +566 -0
  1169. package/skills/creative/p5js/references/interaction.md +398 -0
  1170. package/skills/creative/p5js/references/shapes-and-geometry.md +300 -0
  1171. package/skills/creative/p5js/references/troubleshooting.md +532 -0
  1172. package/skills/creative/p5js/references/typography.md +302 -0
  1173. package/skills/creative/p5js/references/visual-effects.md +895 -0
  1174. package/skills/creative/p5js/references/webgl-and-3d.md +423 -0
  1175. package/skills/creative/p5js/scripts/export-frames.js +179 -0
  1176. package/skills/creative/p5js/scripts/render.sh +108 -0
  1177. package/skills/creative/p5js/scripts/serve.sh +28 -0
  1178. package/skills/creative/p5js/scripts/setup.sh +87 -0
  1179. package/skills/creative/p5js/templates/viewer.html +395 -0
  1180. package/skills/creative/pixel-art/ATTRIBUTION.md +54 -0
  1181. package/skills/creative/pixel-art/SKILL.md +218 -0
  1182. package/skills/creative/pixel-art/references/palettes.md +49 -0
  1183. package/skills/creative/pixel-art/scripts/__init__.py +0 -0
  1184. package/skills/creative/pixel-art/scripts/palettes.py +167 -0
  1185. package/skills/creative/pixel-art/scripts/pixel_art.py +162 -0
  1186. package/skills/creative/pixel-art/scripts/pixel_art_video.py +345 -0
  1187. package/skills/creative/popular-web-designs/SKILL.md +214 -0
  1188. package/skills/creative/popular-web-designs/templates/airbnb.md +259 -0
  1189. package/skills/creative/popular-web-designs/templates/airtable.md +102 -0
  1190. package/skills/creative/popular-web-designs/templates/apple.md +326 -0
  1191. package/skills/creative/popular-web-designs/templates/bmw.md +193 -0
  1192. package/skills/creative/popular-web-designs/templates/cal.md +272 -0
  1193. package/skills/creative/popular-web-designs/templates/claude.md +325 -0
  1194. package/skills/creative/popular-web-designs/templates/clay.md +317 -0
  1195. package/skills/creative/popular-web-designs/templates/clickhouse.md +294 -0
  1196. package/skills/creative/popular-web-designs/templates/cohere.md +279 -0
  1197. package/skills/creative/popular-web-designs/templates/coinbase.md +142 -0
  1198. package/skills/creative/popular-web-designs/templates/composio.md +320 -0
  1199. package/skills/creative/popular-web-designs/templates/cursor.md +322 -0
  1200. package/skills/creative/popular-web-designs/templates/elevenlabs.md +278 -0
  1201. package/skills/creative/popular-web-designs/templates/expo.md +294 -0
  1202. package/skills/creative/popular-web-designs/templates/figma.md +233 -0
  1203. package/skills/creative/popular-web-designs/templates/framer.md +259 -0
  1204. package/skills/creative/popular-web-designs/templates/hashicorp.md +291 -0
  1205. package/skills/creative/popular-web-designs/templates/ibm.md +345 -0
  1206. package/skills/creative/popular-web-designs/templates/intercom.md +159 -0
  1207. package/skills/creative/popular-web-designs/templates/kraken.md +138 -0
  1208. package/skills/creative/popular-web-designs/templates/linear.app.md +380 -0
  1209. package/skills/creative/popular-web-designs/templates/lovable.md +311 -0
  1210. package/skills/creative/popular-web-designs/templates/minimax.md +270 -0
  1211. package/skills/creative/popular-web-designs/templates/mintlify.md +339 -0
  1212. package/skills/creative/popular-web-designs/templates/miro.md +121 -0
  1213. package/skills/creative/popular-web-designs/templates/mistral.ai.md +274 -0
  1214. package/skills/creative/popular-web-designs/templates/mongodb.md +279 -0
  1215. package/skills/creative/popular-web-designs/templates/notion.md +322 -0
  1216. package/skills/creative/popular-web-designs/templates/nvidia.md +306 -0
  1217. package/skills/creative/popular-web-designs/templates/ollama.md +280 -0
  1218. package/skills/creative/popular-web-designs/templates/opencode.ai.md +294 -0
  1219. package/skills/creative/popular-web-designs/templates/pinterest.md +243 -0
  1220. package/skills/creative/popular-web-designs/templates/posthog.md +269 -0
  1221. package/skills/creative/popular-web-designs/templates/raycast.md +281 -0
  1222. package/skills/creative/popular-web-designs/templates/replicate.md +274 -0
  1223. package/skills/creative/popular-web-designs/templates/resend.md +316 -0
  1224. package/skills/creative/popular-web-designs/templates/revolut.md +198 -0
  1225. package/skills/creative/popular-web-designs/templates/runwayml.md +257 -0
  1226. package/skills/creative/popular-web-designs/templates/sanity.md +370 -0
  1227. package/skills/creative/popular-web-designs/templates/sentry.md +275 -0
  1228. package/skills/creative/popular-web-designs/templates/spacex.md +207 -0
  1229. package/skills/creative/popular-web-designs/templates/spotify.md +259 -0
  1230. package/skills/creative/popular-web-designs/templates/stripe.md +335 -0
  1231. package/skills/creative/popular-web-designs/templates/supabase.md +268 -0
  1232. package/skills/creative/popular-web-designs/templates/superhuman.md +265 -0
  1233. package/skills/creative/popular-web-designs/templates/together.ai.md +276 -0
  1234. package/skills/creative/popular-web-designs/templates/uber.md +308 -0
  1235. package/skills/creative/popular-web-designs/templates/vercel.md +323 -0
  1236. package/skills/creative/popular-web-designs/templates/voltagent.md +336 -0
  1237. package/skills/creative/popular-web-designs/templates/warp.md +266 -0
  1238. package/skills/creative/popular-web-designs/templates/webflow.md +105 -0
  1239. package/skills/creative/popular-web-designs/templates/wise.md +186 -0
  1240. package/skills/creative/popular-web-designs/templates/x.ai.md +270 -0
  1241. package/skills/creative/popular-web-designs/templates/zapier.md +341 -0
  1242. package/skills/creative/pretext/SKILL.md +220 -0
  1243. package/skills/creative/pretext/references/patterns.md +258 -0
  1244. package/skills/creative/pretext/templates/donut-orbit.html +1468 -0
  1245. package/skills/creative/pretext/templates/hello-orb-flow.html +95 -0
  1246. package/skills/creative/sketch/SKILL.md +218 -0
  1247. package/skills/creative/songwriting-and-ai-music/SKILL.md +287 -0
  1248. package/skills/creative/touchdesigner-mcp/SKILL.md +356 -0
  1249. package/skills/creative/touchdesigner-mcp/references/3d-scene.md +275 -0
  1250. package/skills/creative/touchdesigner-mcp/references/animation.md +221 -0
  1251. package/skills/creative/touchdesigner-mcp/references/audio-reactive.md +175 -0
  1252. package/skills/creative/touchdesigner-mcp/references/dat-scripting.md +352 -0
  1253. package/skills/creative/touchdesigner-mcp/references/external-data.md +322 -0
  1254. package/skills/creative/touchdesigner-mcp/references/geometry-comp.md +121 -0
  1255. package/skills/creative/touchdesigner-mcp/references/glsl.md +151 -0
  1256. package/skills/creative/touchdesigner-mcp/references/layout-compositor.md +131 -0
  1257. package/skills/creative/touchdesigner-mcp/references/mcp-tools.md +382 -0
  1258. package/skills/creative/touchdesigner-mcp/references/midi-osc.md +211 -0
  1259. package/skills/creative/touchdesigner-mcp/references/network-patterns.md +966 -0
  1260. package/skills/creative/touchdesigner-mcp/references/operator-tips.md +106 -0
  1261. package/skills/creative/touchdesigner-mcp/references/operators.md +239 -0
  1262. package/skills/creative/touchdesigner-mcp/references/panel-ui.md +281 -0
  1263. package/skills/creative/touchdesigner-mcp/references/particles.md +245 -0
  1264. package/skills/creative/touchdesigner-mcp/references/pitfalls.md +704 -0
  1265. package/skills/creative/touchdesigner-mcp/references/postfx.md +183 -0
  1266. package/skills/creative/touchdesigner-mcp/references/projection-mapping.md +211 -0
  1267. package/skills/creative/touchdesigner-mcp/references/python-api.md +463 -0
  1268. package/skills/creative/touchdesigner-mcp/references/replicator.md +198 -0
  1269. package/skills/creative/touchdesigner-mcp/references/troubleshooting.md +244 -0
  1270. package/skills/creative/touchdesigner-mcp/scripts/setup.sh +115 -0
  1271. package/skills/data-science/DESCRIPTION.md +3 -0
  1272. package/skills/data-science/jupyter-live-kernel/SKILL.md +167 -0
  1273. package/skills/devops/kanban-orchestrator/SKILL.md +189 -0
  1274. package/skills/devops/kanban-worker/SKILL.md +184 -0
  1275. package/skills/devops/webhook-subscriptions/SKILL.md +204 -0
  1276. package/skills/diagramming/DESCRIPTION.md +3 -0
  1277. package/skills/dogfood/SKILL.md +162 -0
  1278. package/skills/dogfood/references/issue-taxonomy.md +109 -0
  1279. package/skills/dogfood/templates/dogfood-report-template.md +86 -0
  1280. package/skills/domain/DESCRIPTION.md +24 -0
  1281. package/skills/email/DESCRIPTION.md +3 -0
  1282. package/skills/email/himalaya/SKILL.md +299 -0
  1283. package/skills/email/himalaya/references/configuration.md +227 -0
  1284. package/skills/email/himalaya/references/message-composition.md +199 -0
  1285. package/skills/gaming/DESCRIPTION.md +3 -0
  1286. package/skills/gaming/minecraft-modpack-server/SKILL.md +187 -0
  1287. package/skills/gaming/pokemon-player/SKILL.md +216 -0
  1288. package/skills/gifs/DESCRIPTION.md +3 -0
  1289. package/skills/github/DESCRIPTION.md +3 -0
  1290. package/skills/github/codebase-inspection/SKILL.md +116 -0
  1291. package/skills/github/github-auth/SKILL.md +247 -0
  1292. package/skills/github/github-auth/scripts/gh-env.sh +66 -0
  1293. package/skills/github/github-code-review/SKILL.md +481 -0
  1294. package/skills/github/github-code-review/references/review-output-template.md +74 -0
  1295. package/skills/github/github-issues/SKILL.md +370 -0
  1296. package/skills/github/github-issues/templates/bug-report.md +35 -0
  1297. package/skills/github/github-issues/templates/feature-request.md +31 -0
  1298. package/skills/github/github-pr-workflow/SKILL.md +367 -0
  1299. package/skills/github/github-pr-workflow/references/ci-troubleshooting.md +183 -0
  1300. package/skills/github/github-pr-workflow/references/conventional-commits.md +71 -0
  1301. package/skills/github/github-pr-workflow/templates/pr-body-bugfix.md +35 -0
  1302. package/skills/github/github-pr-workflow/templates/pr-body-feature.md +33 -0
  1303. package/skills/github/github-repo-management/SKILL.md +516 -0
  1304. package/skills/github/github-repo-management/references/github-api-cheatsheet.md +161 -0
  1305. package/skills/index-cache/anthropics_skills_skills_.json +1 -0
  1306. package/skills/index-cache/claude_marketplace_anthropics_skills.json +1 -0
  1307. package/skills/index-cache/lobehub_index.json +1 -0
  1308. package/skills/index-cache/openai_skills_skills_.json +1 -0
  1309. package/skills/inference-sh/DESCRIPTION.md +19 -0
  1310. package/skills/mcp/DESCRIPTION.md +3 -0
  1311. package/skills/mcp/native-mcp/SKILL.md +357 -0
  1312. package/skills/media/DESCRIPTION.md +3 -0
  1313. package/skills/media/gif-search/SKILL.md +91 -0
  1314. package/skills/media/heartmula/SKILL.md +171 -0
  1315. package/skills/media/songsee/SKILL.md +83 -0
  1316. package/skills/media/spotify/SKILL.md +135 -0
  1317. package/skills/media/youtube-content/SKILL.md +73 -0
  1318. package/skills/media/youtube-content/references/output-formats.md +56 -0
  1319. package/skills/media/youtube-content/scripts/fetch_transcript.py +124 -0
  1320. package/skills/mlops/DESCRIPTION.md +3 -0
  1321. package/skills/mlops/evaluation/DESCRIPTION.md +3 -0
  1322. package/skills/mlops/evaluation/lm-evaluation-harness/SKILL.md +498 -0
  1323. package/skills/mlops/evaluation/lm-evaluation-harness/references/api-evaluation.md +490 -0
  1324. package/skills/mlops/evaluation/lm-evaluation-harness/references/benchmark-guide.md +488 -0
  1325. package/skills/mlops/evaluation/lm-evaluation-harness/references/custom-tasks.md +602 -0
  1326. package/skills/mlops/evaluation/lm-evaluation-harness/references/distributed-eval.md +519 -0
  1327. package/skills/mlops/evaluation/weights-and-biases/SKILL.md +594 -0
  1328. package/skills/mlops/evaluation/weights-and-biases/references/artifacts.md +584 -0
  1329. package/skills/mlops/evaluation/weights-and-biases/references/integrations.md +700 -0
  1330. package/skills/mlops/evaluation/weights-and-biases/references/sweeps.md +847 -0
  1331. package/skills/mlops/huggingface-hub/SKILL.md +81 -0
  1332. package/skills/mlops/inference/DESCRIPTION.md +3 -0
  1333. package/skills/mlops/inference/llama-cpp/SKILL.md +249 -0
  1334. package/skills/mlops/inference/llama-cpp/references/advanced-usage.md +504 -0
  1335. package/skills/mlops/inference/llama-cpp/references/hub-discovery.md +168 -0
  1336. package/skills/mlops/inference/llama-cpp/references/optimization.md +89 -0
  1337. package/skills/mlops/inference/llama-cpp/references/quantization.md +243 -0
  1338. package/skills/mlops/inference/llama-cpp/references/server.md +150 -0
  1339. package/skills/mlops/inference/llama-cpp/references/troubleshooting.md +442 -0
  1340. package/skills/mlops/inference/obliteratus/SKILL.md +342 -0
  1341. package/skills/mlops/inference/obliteratus/references/analysis-modules.md +166 -0
  1342. package/skills/mlops/inference/obliteratus/references/methods-guide.md +141 -0
  1343. package/skills/mlops/inference/obliteratus/templates/abliteration-config.yaml +33 -0
  1344. package/skills/mlops/inference/obliteratus/templates/analysis-study.yaml +40 -0
  1345. package/skills/mlops/inference/obliteratus/templates/batch-abliteration.yaml +41 -0
  1346. package/skills/mlops/inference/vllm/SKILL.md +372 -0
  1347. package/skills/mlops/inference/vllm/references/optimization.md +226 -0
  1348. package/skills/mlops/inference/vllm/references/quantization.md +284 -0
  1349. package/skills/mlops/inference/vllm/references/server-deployment.md +255 -0
  1350. package/skills/mlops/inference/vllm/references/troubleshooting.md +447 -0
  1351. package/skills/mlops/models/DESCRIPTION.md +3 -0
  1352. package/skills/mlops/models/audiocraft/SKILL.md +568 -0
  1353. package/skills/mlops/models/audiocraft/references/advanced-usage.md +666 -0
  1354. package/skills/mlops/models/audiocraft/references/troubleshooting.md +504 -0
  1355. package/skills/mlops/models/segment-anything/SKILL.md +506 -0
  1356. package/skills/mlops/models/segment-anything/references/advanced-usage.md +589 -0
  1357. package/skills/mlops/models/segment-anything/references/troubleshooting.md +484 -0
  1358. package/skills/mlops/research/DESCRIPTION.md +3 -0
  1359. package/skills/mlops/research/dspy/SKILL.md +594 -0
  1360. package/skills/mlops/research/dspy/references/examples.md +663 -0
  1361. package/skills/mlops/research/dspy/references/modules.md +475 -0
  1362. package/skills/mlops/research/dspy/references/optimizers.md +566 -0
  1363. package/skills/mlops/training/DESCRIPTION.md +3 -0
  1364. package/skills/mlops/vector-databases/DESCRIPTION.md +3 -0
  1365. package/skills/note-taking/DESCRIPTION.md +3 -0
  1366. package/skills/note-taking/obsidian/SKILL.md +61 -0
  1367. package/skills/productivity/DESCRIPTION.md +3 -0
  1368. package/skills/productivity/airtable/SKILL.md +229 -0
  1369. package/skills/productivity/google-workspace/SKILL.md +335 -0
  1370. package/skills/productivity/google-workspace/references/gmail-search-syntax.md +63 -0
  1371. package/skills/productivity/google-workspace/scripts/_hermes_home.py +43 -0
  1372. package/skills/productivity/google-workspace/scripts/google_api.py +1221 -0
  1373. package/skills/productivity/google-workspace/scripts/gws_bridge.py +108 -0
  1374. package/skills/productivity/google-workspace/scripts/setup.py +454 -0
  1375. package/skills/productivity/linear/SKILL.md +380 -0
  1376. package/skills/productivity/linear/scripts/linear_api.py +445 -0
  1377. package/skills/productivity/maps/SKILL.md +195 -0
  1378. package/skills/productivity/maps/scripts/maps_client.py +1298 -0
  1379. package/skills/productivity/nano-pdf/SKILL.md +52 -0
  1380. package/skills/productivity/notion/SKILL.md +448 -0
  1381. package/skills/productivity/notion/references/block-types.md +112 -0
  1382. package/skills/productivity/ocr-and-documents/DESCRIPTION.md +3 -0
  1383. package/skills/productivity/ocr-and-documents/SKILL.md +172 -0
  1384. package/skills/productivity/ocr-and-documents/scripts/extract_marker.py +87 -0
  1385. package/skills/productivity/ocr-and-documents/scripts/extract_pymupdf.py +98 -0
  1386. package/skills/productivity/powerpoint/LICENSE.txt +30 -0
  1387. package/skills/productivity/powerpoint/SKILL.md +237 -0
  1388. package/skills/productivity/powerpoint/editing.md +205 -0
  1389. package/skills/productivity/powerpoint/pptxgenjs.md +420 -0
  1390. package/skills/productivity/powerpoint/scripts/__init__.py +0 -0
  1391. package/skills/productivity/powerpoint/scripts/add_slide.py +195 -0
  1392. package/skills/productivity/powerpoint/scripts/clean.py +286 -0
  1393. package/skills/productivity/powerpoint/scripts/office/helpers/__init__.py +0 -0
  1394. package/skills/productivity/powerpoint/scripts/office/helpers/merge_runs.py +199 -0
  1395. package/skills/productivity/powerpoint/scripts/office/helpers/simplify_redlines.py +197 -0
  1396. package/skills/productivity/powerpoint/scripts/office/pack.py +159 -0
  1397. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  1398. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  1399. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  1400. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  1401. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  1402. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  1403. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  1404. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  1405. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  1406. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  1407. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  1408. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  1409. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  1410. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  1411. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  1412. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  1413. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  1414. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  1415. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  1416. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  1417. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  1418. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  1419. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  1420. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  1421. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  1422. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  1423. package/skills/productivity/powerpoint/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  1424. package/skills/productivity/powerpoint/scripts/office/schemas/ecma/fourth-edition/opc-contentTypes.xsd +42 -0
  1425. package/skills/productivity/powerpoint/scripts/office/schemas/ecma/fourth-edition/opc-coreProperties.xsd +50 -0
  1426. package/skills/productivity/powerpoint/scripts/office/schemas/ecma/fourth-edition/opc-digSig.xsd +49 -0
  1427. package/skills/productivity/powerpoint/scripts/office/schemas/ecma/fourth-edition/opc-relationships.xsd +33 -0
  1428. package/skills/productivity/powerpoint/scripts/office/schemas/mce/mc.xsd +75 -0
  1429. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
  1430. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
  1431. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
  1432. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
  1433. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
  1434. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  1435. package/skills/productivity/powerpoint/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
  1436. package/skills/productivity/teams-meeting-pipeline/SKILL.md +116 -0
  1437. package/skills/red-teaming/godmode/SKILL.md +404 -0
  1438. package/skills/red-teaming/godmode/references/jailbreak-templates.md +128 -0
  1439. package/skills/red-teaming/godmode/references/refusal-detection.md +142 -0
  1440. package/skills/red-teaming/godmode/scripts/auto_jailbreak.py +769 -0
  1441. package/skills/red-teaming/godmode/scripts/godmode_race.py +530 -0
  1442. package/skills/red-teaming/godmode/scripts/load_godmode.py +45 -0
  1443. package/skills/red-teaming/godmode/scripts/parseltongue.py +550 -0
  1444. package/skills/red-teaming/godmode/templates/prefill-subtle.json +10 -0
  1445. package/skills/red-teaming/godmode/templates/prefill.json +18 -0
  1446. package/skills/research/DESCRIPTION.md +3 -0
  1447. package/skills/research/arxiv/SKILL.md +282 -0
  1448. package/skills/research/arxiv/scripts/search_arxiv.py +114 -0
  1449. package/skills/research/blogwatcher/SKILL.md +137 -0
  1450. package/skills/research/llm-wiki/SKILL.md +507 -0
  1451. package/skills/research/polymarket/SKILL.md +77 -0
  1452. package/skills/research/polymarket/references/api-endpoints.md +220 -0
  1453. package/skills/research/polymarket/scripts/polymarket.py +284 -0
  1454. package/skills/research/research-paper-writing/SKILL.md +2377 -0
  1455. package/skills/research/research-paper-writing/references/autoreason-methodology.md +394 -0
  1456. package/skills/research/research-paper-writing/references/checklists.md +434 -0
  1457. package/skills/research/research-paper-writing/references/citation-workflow.md +564 -0
  1458. package/skills/research/research-paper-writing/references/experiment-patterns.md +728 -0
  1459. package/skills/research/research-paper-writing/references/human-evaluation.md +476 -0
  1460. package/skills/research/research-paper-writing/references/paper-types.md +481 -0
  1461. package/skills/research/research-paper-writing/references/reviewer-guidelines.md +433 -0
  1462. package/skills/research/research-paper-writing/references/sources.md +191 -0
  1463. package/skills/research/research-paper-writing/references/writing-guide.md +474 -0
  1464. package/skills/research/research-paper-writing/templates/README.md +251 -0
  1465. package/skills/research/research-paper-writing/templates/aaai2026/README.md +534 -0
  1466. package/skills/research/research-paper-writing/templates/aaai2026/aaai2026-unified-supp.tex +144 -0
  1467. package/skills/research/research-paper-writing/templates/aaai2026/aaai2026-unified-template.tex +952 -0
  1468. package/skills/research/research-paper-writing/templates/aaai2026/aaai2026.bib +111 -0
  1469. package/skills/research/research-paper-writing/templates/aaai2026/aaai2026.bst +1493 -0
  1470. package/skills/research/research-paper-writing/templates/aaai2026/aaai2026.sty +315 -0
  1471. package/skills/research/research-paper-writing/templates/acl/README.md +50 -0
  1472. package/skills/research/research-paper-writing/templates/acl/acl.sty +312 -0
  1473. package/skills/research/research-paper-writing/templates/acl/acl_latex.tex +377 -0
  1474. package/skills/research/research-paper-writing/templates/acl/acl_lualatex.tex +101 -0
  1475. package/skills/research/research-paper-writing/templates/acl/acl_natbib.bst +1940 -0
  1476. package/skills/research/research-paper-writing/templates/acl/anthology.bib.txt +26 -0
  1477. package/skills/research/research-paper-writing/templates/acl/custom.bib +70 -0
  1478. package/skills/research/research-paper-writing/templates/acl/formatting.md +326 -0
  1479. package/skills/research/research-paper-writing/templates/colm2025/README.md +3 -0
  1480. package/skills/research/research-paper-writing/templates/colm2025/colm2025_conference.bib +11 -0
  1481. package/skills/research/research-paper-writing/templates/colm2025/colm2025_conference.bst +1440 -0
  1482. package/skills/research/research-paper-writing/templates/colm2025/colm2025_conference.pdf +0 -0
  1483. package/skills/research/research-paper-writing/templates/colm2025/colm2025_conference.sty +218 -0
  1484. package/skills/research/research-paper-writing/templates/colm2025/colm2025_conference.tex +305 -0
  1485. package/skills/research/research-paper-writing/templates/colm2025/fancyhdr.sty +485 -0
  1486. package/skills/research/research-paper-writing/templates/colm2025/math_commands.tex +508 -0
  1487. package/skills/research/research-paper-writing/templates/colm2025/natbib.sty +1246 -0
  1488. package/skills/research/research-paper-writing/templates/iclr2026/fancyhdr.sty +485 -0
  1489. package/skills/research/research-paper-writing/templates/iclr2026/iclr2026_conference.bib +24 -0
  1490. package/skills/research/research-paper-writing/templates/iclr2026/iclr2026_conference.bst +1440 -0
  1491. package/skills/research/research-paper-writing/templates/iclr2026/iclr2026_conference.pdf +0 -0
  1492. package/skills/research/research-paper-writing/templates/iclr2026/iclr2026_conference.sty +246 -0
  1493. package/skills/research/research-paper-writing/templates/iclr2026/iclr2026_conference.tex +414 -0
  1494. package/skills/research/research-paper-writing/templates/iclr2026/math_commands.tex +508 -0
  1495. package/skills/research/research-paper-writing/templates/iclr2026/natbib.sty +1246 -0
  1496. package/skills/research/research-paper-writing/templates/icml2026/algorithm.sty +79 -0
  1497. package/skills/research/research-paper-writing/templates/icml2026/algorithmic.sty +201 -0
  1498. package/skills/research/research-paper-writing/templates/icml2026/example_paper.bib +75 -0
  1499. package/skills/research/research-paper-writing/templates/icml2026/example_paper.pdf +0 -0
  1500. package/skills/research/research-paper-writing/templates/icml2026/example_paper.tex +662 -0
  1501. package/skills/research/research-paper-writing/templates/icml2026/fancyhdr.sty +864 -0
  1502. package/skills/research/research-paper-writing/templates/icml2026/icml2026.bst +1443 -0
  1503. package/skills/research/research-paper-writing/templates/icml2026/icml2026.sty +767 -0
  1504. package/skills/research/research-paper-writing/templates/icml2026/icml_numpapers.pdf +0 -0
  1505. package/skills/research/research-paper-writing/templates/neurips2025/Makefile +36 -0
  1506. package/skills/research/research-paper-writing/templates/neurips2025/extra_pkgs.tex +53 -0
  1507. package/skills/research/research-paper-writing/templates/neurips2025/main.tex +38 -0
  1508. package/skills/research/research-paper-writing/templates/neurips2025/neurips.sty +382 -0
  1509. package/skills/smart-home/DESCRIPTION.md +3 -0
  1510. package/skills/smart-home/openhue/SKILL.md +109 -0
  1511. package/skills/social-media/DESCRIPTION.md +3 -0
  1512. package/skills/social-media/xurl/SKILL.md +414 -0
  1513. package/skills/software-development/debugging-hermes-tui-commands/SKILL.md +152 -0
  1514. package/skills/software-development/hermes-agent-skill-authoring/SKILL.md +165 -0
  1515. package/skills/software-development/node-inspect-debugger/SKILL.md +319 -0
  1516. package/skills/software-development/plan/SKILL.md +58 -0
  1517. package/skills/software-development/python-debugpy/SKILL.md +375 -0
  1518. package/skills/software-development/requesting-code-review/SKILL.md +280 -0
  1519. package/skills/software-development/spike/SKILL.md +197 -0
  1520. package/skills/software-development/subagent-driven-development/SKILL.md +352 -0
  1521. package/skills/software-development/subagent-driven-development/references/context-budget-discipline.md +53 -0
  1522. package/skills/software-development/subagent-driven-development/references/gates-taxonomy.md +93 -0
  1523. package/skills/software-development/systematic-debugging/SKILL.md +367 -0
  1524. package/skills/software-development/test-driven-development/SKILL.md +343 -0
  1525. package/skills/software-development/writing-plans/SKILL.md +297 -0
  1526. package/skills/yuanbao/SKILL.md +108 -0
  1527. package/tools/__init__.py +25 -0
  1528. package/tools/__pycache__/__init__.cpython-312.pyc +0 -0
  1529. package/tools/__pycache__/approval.cpython-312.pyc +0 -0
  1530. package/tools/__pycache__/binary_extensions.cpython-312.pyc +0 -0
  1531. package/tools/__pycache__/browser_camofox.cpython-312.pyc +0 -0
  1532. package/tools/__pycache__/browser_camofox_state.cpython-312.pyc +0 -0
  1533. package/tools/__pycache__/browser_cdp_tool.cpython-312.pyc +0 -0
  1534. package/tools/__pycache__/browser_dialog_tool.cpython-312.pyc +0 -0
  1535. package/tools/__pycache__/browser_supervisor.cpython-312.pyc +0 -0
  1536. package/tools/__pycache__/browser_tool.cpython-312.pyc +0 -0
  1537. package/tools/__pycache__/budget_config.cpython-312.pyc +0 -0
  1538. package/tools/__pycache__/checkpoint_manager.cpython-312.pyc +0 -0
  1539. package/tools/__pycache__/clarify_gateway.cpython-312.pyc +0 -0
  1540. package/tools/__pycache__/clarify_tool.cpython-312.pyc +0 -0
  1541. package/tools/__pycache__/code_execution_tool.cpython-312.pyc +0 -0
  1542. package/tools/__pycache__/computer_use_tool.cpython-312.pyc +0 -0
  1543. package/tools/__pycache__/cronjob_tools.cpython-312.pyc +0 -0
  1544. package/tools/__pycache__/debug_helpers.cpython-312.pyc +0 -0
  1545. package/tools/__pycache__/delegate_tool.cpython-312.pyc +0 -0
  1546. package/tools/__pycache__/discord_tool.cpython-312.pyc +0 -0
  1547. package/tools/__pycache__/feishu_doc_tool.cpython-312.pyc +0 -0
  1548. package/tools/__pycache__/feishu_drive_tool.cpython-312.pyc +0 -0
  1549. package/tools/__pycache__/file_operations.cpython-312.pyc +0 -0
  1550. package/tools/__pycache__/file_state.cpython-312.pyc +0 -0
  1551. package/tools/__pycache__/file_tools.cpython-312.pyc +0 -0
  1552. package/tools/__pycache__/homeassistant_tool.cpython-312.pyc +0 -0
  1553. package/tools/__pycache__/image_generation_tool.cpython-312.pyc +0 -0
  1554. package/tools/__pycache__/interrupt.cpython-312.pyc +0 -0
  1555. package/tools/__pycache__/kanban_tools.cpython-312.pyc +0 -0
  1556. package/tools/__pycache__/lazy_deps.cpython-312.pyc +0 -0
  1557. package/tools/__pycache__/managed_tool_gateway.cpython-312.pyc +0 -0
  1558. package/tools/__pycache__/mcp_tool.cpython-312.pyc +0 -0
  1559. package/tools/__pycache__/memory_tool.cpython-312.pyc +0 -0
  1560. package/tools/__pycache__/mixture_of_agents_tool.cpython-312.pyc +0 -0
  1561. package/tools/__pycache__/openrouter_client.cpython-312.pyc +0 -0
  1562. package/tools/__pycache__/process_registry.cpython-312.pyc +0 -0
  1563. package/tools/__pycache__/registry.cpython-312.pyc +0 -0
  1564. package/tools/__pycache__/schema_sanitizer.cpython-312.pyc +0 -0
  1565. package/tools/__pycache__/send_message_tool.cpython-312.pyc +0 -0
  1566. package/tools/__pycache__/session_search_tool.cpython-312.pyc +0 -0
  1567. package/tools/__pycache__/skill_manager_tool.cpython-312.pyc +0 -0
  1568. package/tools/__pycache__/skill_provenance.cpython-312.pyc +0 -0
  1569. package/tools/__pycache__/skill_usage.cpython-312.pyc +0 -0
  1570. package/tools/__pycache__/skills_guard.cpython-312.pyc +0 -0
  1571. package/tools/__pycache__/skills_sync.cpython-312.pyc +0 -0
  1572. package/tools/__pycache__/skills_tool.cpython-312.pyc +0 -0
  1573. package/tools/__pycache__/slash_confirm.cpython-312.pyc +0 -0
  1574. package/tools/__pycache__/terminal_tool.cpython-312.pyc +0 -0
  1575. package/tools/__pycache__/tirith_security.cpython-312.pyc +0 -0
  1576. package/tools/__pycache__/todo_tool.cpython-312.pyc +0 -0
  1577. package/tools/__pycache__/tool_backend_helpers.cpython-312.pyc +0 -0
  1578. package/tools/__pycache__/tool_result_storage.cpython-312.pyc +0 -0
  1579. package/tools/__pycache__/tts_tool.cpython-312.pyc +0 -0
  1580. package/tools/__pycache__/url_safety.cpython-312.pyc +0 -0
  1581. package/tools/__pycache__/video_generation_tool.cpython-312.pyc +0 -0
  1582. package/tools/__pycache__/vision_tools.cpython-312.pyc +0 -0
  1583. package/tools/__pycache__/voice_mode.cpython-312.pyc +0 -0
  1584. package/tools/__pycache__/web_tools.cpython-312.pyc +0 -0
  1585. package/tools/__pycache__/website_policy.cpython-312.pyc +0 -0
  1586. package/tools/__pycache__/x_search_tool.cpython-312.pyc +0 -0
  1587. package/tools/__pycache__/xai_http.cpython-312.pyc +0 -0
  1588. package/tools/__pycache__/yuanbao_tools.cpython-312.pyc +0 -0
  1589. package/tools/ansi_strip.py +44 -0
  1590. package/tools/approval.py +1392 -0
  1591. package/tools/binary_extensions.py +42 -0
  1592. package/tools/browser_camofox.py +700 -0
  1593. package/tools/browser_camofox_state.py +48 -0
  1594. package/tools/browser_cdp_tool.py +569 -0
  1595. package/tools/browser_dialog_tool.py +148 -0
  1596. package/tools/browser_providers/__init__.py +10 -0
  1597. package/tools/browser_providers/__pycache__/__init__.cpython-312.pyc +0 -0
  1598. package/tools/browser_providers/__pycache__/base.cpython-312.pyc +0 -0
  1599. package/tools/browser_providers/__pycache__/browser_use.cpython-312.pyc +0 -0
  1600. package/tools/browser_providers/__pycache__/browserbase.cpython-312.pyc +0 -0
  1601. package/tools/browser_providers/__pycache__/firecrawl.cpython-312.pyc +0 -0
  1602. package/tools/browser_providers/base.py +59 -0
  1603. package/tools/browser_providers/browser_use.py +225 -0
  1604. package/tools/browser_providers/browserbase.py +222 -0
  1605. package/tools/browser_providers/firecrawl.py +112 -0
  1606. package/tools/browser_supervisor.py +1457 -0
  1607. package/tools/browser_tool.py +3676 -0
  1608. package/tools/budget_config.py +51 -0
  1609. package/tools/checkpoint_manager.py +1639 -0
  1610. package/tools/clarify_gateway.py +278 -0
  1611. package/tools/clarify_tool.py +141 -0
  1612. package/tools/code_execution_tool.py +1782 -0
  1613. package/tools/computer_use/__init__.py +43 -0
  1614. package/tools/computer_use/__pycache__/__init__.cpython-312.pyc +0 -0
  1615. package/tools/computer_use/__pycache__/backend.cpython-312.pyc +0 -0
  1616. package/tools/computer_use/__pycache__/schema.cpython-312.pyc +0 -0
  1617. package/tools/computer_use/__pycache__/tool.cpython-312.pyc +0 -0
  1618. package/tools/computer_use/backend.py +150 -0
  1619. package/tools/computer_use/cua_backend.py +682 -0
  1620. package/tools/computer_use/schema.py +191 -0
  1621. package/tools/computer_use/tool.py +521 -0
  1622. package/tools/computer_use_tool.py +39 -0
  1623. package/tools/credential_files.py +437 -0
  1624. package/tools/cronjob_tools.py +719 -0
  1625. package/tools/debug_helpers.py +106 -0
  1626. package/tools/delegate_tool.py +2797 -0
  1627. package/tools/discord_tool.py +959 -0
  1628. package/tools/env_passthrough.py +145 -0
  1629. package/tools/environments/__init__.py +14 -0
  1630. package/tools/environments/__pycache__/__init__.cpython-312.pyc +0 -0
  1631. package/tools/environments/__pycache__/base.cpython-312.pyc +0 -0
  1632. package/tools/environments/__pycache__/docker.cpython-312.pyc +0 -0
  1633. package/tools/environments/__pycache__/file_sync.cpython-312.pyc +0 -0
  1634. package/tools/environments/__pycache__/local.cpython-312.pyc +0 -0
  1635. package/tools/environments/__pycache__/managed_modal.cpython-312.pyc +0 -0
  1636. package/tools/environments/__pycache__/modal.cpython-312.pyc +0 -0
  1637. package/tools/environments/__pycache__/modal_utils.cpython-312.pyc +0 -0
  1638. package/tools/environments/__pycache__/singularity.cpython-312.pyc +0 -0
  1639. package/tools/environments/__pycache__/ssh.cpython-312.pyc +0 -0
  1640. package/tools/environments/base.py +844 -0
  1641. package/tools/environments/daytona.py +270 -0
  1642. package/tools/environments/docker.py +656 -0
  1643. package/tools/environments/file_sync.py +400 -0
  1644. package/tools/environments/local.py +658 -0
  1645. package/tools/environments/managed_modal.py +282 -0
  1646. package/tools/environments/modal.py +479 -0
  1647. package/tools/environments/modal_utils.py +199 -0
  1648. package/tools/environments/singularity.py +263 -0
  1649. package/tools/environments/ssh.py +295 -0
  1650. package/tools/environments/vercel_sandbox.py +655 -0
  1651. package/tools/feishu_doc_tool.py +138 -0
  1652. package/tools/feishu_drive_tool.py +431 -0
  1653. package/tools/file_operations.py +1825 -0
  1654. package/tools/file_state.py +332 -0
  1655. package/tools/file_tools.py +1172 -0
  1656. package/tools/fuzzy_match.py +703 -0
  1657. package/tools/homeassistant_tool.py +513 -0
  1658. package/tools/image_generation_tool.py +1098 -0
  1659. package/tools/interrupt.py +98 -0
  1660. package/tools/kanban_tools.py +1139 -0
  1661. package/tools/lazy_deps.py +608 -0
  1662. package/tools/managed_tool_gateway.py +168 -0
  1663. package/tools/mcp_oauth.py +633 -0
  1664. package/tools/mcp_oauth_manager.py +607 -0
  1665. package/tools/mcp_tool.py +3483 -0
  1666. package/tools/memory_tool.py +584 -0
  1667. package/tools/microsoft_graph_auth.py +245 -0
  1668. package/tools/microsoft_graph_client.py +408 -0
  1669. package/tools/mixture_of_agents_tool.py +542 -0
  1670. package/tools/neutts_samples/jo.txt +1 -0
  1671. package/tools/neutts_samples/jo.wav +0 -0
  1672. package/tools/neutts_synth.py +104 -0
  1673. package/tools/openrouter_client.py +33 -0
  1674. package/tools/osv_check.py +155 -0
  1675. package/tools/patch_parser.py +592 -0
  1676. package/tools/path_security.py +43 -0
  1677. package/tools/process_registry.py +1534 -0
  1678. package/tools/registry.py +589 -0
  1679. package/tools/schema_sanitizer.py +370 -0
  1680. package/tools/send_message_tool.py +1900 -0
  1681. package/tools/session_search_tool.py +613 -0
  1682. package/tools/skill_manager_tool.py +932 -0
  1683. package/tools/skill_provenance.py +78 -0
  1684. package/tools/skill_usage.py +610 -0
  1685. package/tools/skills_guard.py +932 -0
  1686. package/tools/skills_hub.py +3263 -0
  1687. package/tools/skills_sync.py +432 -0
  1688. package/tools/skills_tool.py +1569 -0
  1689. package/tools/slash_confirm.py +167 -0
  1690. package/tools/terminal_tool.py +2376 -0
  1691. package/tools/tirith_security.py +775 -0
  1692. package/tools/todo_tool.py +277 -0
  1693. package/tools/tool_backend_helpers.py +144 -0
  1694. package/tools/tool_output_limits.py +92 -0
  1695. package/tools/tool_result_storage.py +232 -0
  1696. package/tools/transcription_tools.py +936 -0
  1697. package/tools/tts_tool.py +2285 -0
  1698. package/tools/url_safety.py +330 -0
  1699. package/tools/video_generation_tool.py +561 -0
  1700. package/tools/vision_tools.py +1422 -0
  1701. package/tools/voice_mode.py +1019 -0
  1702. package/tools/web_tools.py +1551 -0
  1703. package/tools/website_policy.py +283 -0
  1704. package/tools/x_search_tool.py +424 -0
  1705. package/tools/xai_http.py +83 -0
  1706. package/tools/yuanbao_tools.py +736 -0
  1707. package/toolset_distributions.py +364 -0
  1708. package/toolsets.py +866 -0
  1709. package/trajectory_compressor.py +1509 -0
  1710. package/tui_gateway/__init__.py +0 -0
  1711. package/tui_gateway/entry.py +251 -0
  1712. package/tui_gateway/event_publisher.py +126 -0
  1713. package/tui_gateway/render.py +49 -0
  1714. package/tui_gateway/server.py +6623 -0
  1715. package/tui_gateway/slash_worker.py +76 -0
  1716. package/tui_gateway/transport.py +219 -0
  1717. package/tui_gateway/ws.py +178 -0
  1718. package/utils.py +361 -0
@@ -0,0 +1,3151 @@
1
+ """
2
+ Unified tool configuration for Hermes Agent.
3
+
4
+ `hermes tools` and `hermes setup tools` both enter this module.
5
+ Select a platform → toggle toolsets on/off → for newly enabled tools
6
+ that need API keys, run through provider-aware configuration.
7
+
8
+ Saves per-platform tool configuration to ~/.hermes/config.yaml under
9
+ the `platform_toolsets` key.
10
+ """
11
+
12
+ import json as _json
13
+ import logging
14
+ import os
15
+ import shutil
16
+ import subprocess
17
+ import sys
18
+ from pathlib import Path
19
+ from typing import Dict, List, Optional, Set
20
+
21
+
22
+ from hermes_cli.config import (
23
+ cfg_get,
24
+ load_config, save_config, get_env_value, save_env_value,
25
+ )
26
+ from hermes_cli.colors import Colors, color
27
+ from hermes_cli.nous_subscription import (
28
+ apply_nous_managed_defaults,
29
+ get_nous_subscription_features,
30
+ )
31
+ from tools.tool_backend_helpers import fal_key_is_configured, managed_nous_tools_enabled
32
+ from utils import base_url_hostname, is_truthy_value
33
+
34
+ logger = logging.getLogger(__name__)
35
+
36
+ PROJECT_ROOT = Path(__file__).parent.parent.resolve()
37
+
38
+
39
+ # ─── UI Helpers (shared with setup.py) ────────────────────────────────────────
40
+
41
+ from hermes_cli.cli_output import ( # noqa: E402 — late import block
42
+ print_error as _print_error,
43
+ print_info as _print_info,
44
+ print_success as _print_success,
45
+ print_warning as _print_warning,
46
+ prompt as _prompt,
47
+ )
48
+
49
+ # ─── Toolset Registry ─────────────────────────────────────────────────────────
50
+
51
+ # Toolsets shown in the configurator, grouped for display.
52
+ # Each entry: (toolset_name, label, description)
53
+ # These map to keys in toolsets.py TOOLSETS dict.
54
+ CONFIGURABLE_TOOLSETS = [
55
+ ("web", "🔍 Web Search & Scraping", "web_search, web_extract"),
56
+ ("browser", "🌐 Browser Automation", "navigate, click, type, scroll"),
57
+ ("terminal", "💻 Terminal & Processes", "terminal, process"),
58
+ ("file", "📁 File Operations", "read, write, patch, search"),
59
+ ("code_execution", "⚡ Code Execution", "execute_code"),
60
+ ("vision", "👁️ Vision / Image Analysis", "vision_analyze"),
61
+ ("video", "🎬 Video Analysis", "video_analyze (requires video-capable model)"),
62
+ ("image_gen", "🎨 Image Generation", "image_generate"),
63
+ ("video_gen", "🎬 Video Generation", "video_generate (text-to-video + image-to-video)"),
64
+ ("x_search", "🐦 X (Twitter) Search", "x_search (requires xAI OAuth or XAI_API_KEY)"),
65
+ ("moa", "🧠 Mixture of Agents", "mixture_of_agents"),
66
+ ("tts", "🔊 Text-to-Speech", "text_to_speech"),
67
+ ("skills", "📚 Skills", "list, view, manage"),
68
+ ("todo", "📋 Task Planning", "todo"),
69
+ ("memory", "💾 Memory", "persistent memory across sessions"),
70
+ ("session_search", "🔎 Session Search", "search past conversations"),
71
+ ("clarify", "❓ Clarifying Questions", "clarify"),
72
+ ("delegation", "👥 Task Delegation", "delegate_task"),
73
+ ("cronjob", "⏰ Cron Jobs", "create/list/update/pause/resume/run, with optional attached skills"),
74
+ ("messaging", "📨 Cross-Platform Messaging", "send_message"),
75
+ ("homeassistant", "🏠 Home Assistant", "smart home device control"),
76
+ ("spotify", "🎵 Spotify", "playback, search, playlists, library"),
77
+ ("discord", "💬 Discord (read/participate)", "fetch messages, search members, create thread"),
78
+ ("discord_admin", "🛡️ Discord Server Admin", "list channels/roles, pin, assign roles"),
79
+ ("yuanbao", "🤖 Yuanbao", "group info, member queries, DM"),
80
+ ("computer_use", "🖱️ Computer Use (macOS)", "background desktop control via cua-driver"),
81
+ ]
82
+
83
+ # Toolsets that are OFF by default for new installs.
84
+ # They're still in _HERMES_CORE_TOOLS (available at runtime if enabled),
85
+ # but the setup checklist won't pre-select them for first-time users.
86
+ #
87
+ # Video gen is off by default — it's a niche, paid, slow feature. Users
88
+ # who want it opt in via `hermes tools` → Video Generation, which walks
89
+ # them through provider + model selection.
90
+ #
91
+ # X search is off by default — gated on xAI credentials (SuperGrok OAuth
92
+ # or XAI_API_KEY). Users opt in via `hermes tools` → X (Twitter) Search,
93
+ # which walks them through credential setup. The tool's check_fn means
94
+ # the schema won't appear to the model even if enabled without credentials.
95
+ _DEFAULT_OFF_TOOLSETS = {"moa", "homeassistant", "spotify", "discord", "discord_admin", "video", "video_gen", "x_search"}
96
+
97
+ # Platform-scoped toolsets: only appear in the `hermes tools` checklist for
98
+ # these platforms, and only resolve/save for these platforms. A toolset
99
+ # absent from this map is available on every platform (current behaviour).
100
+ #
101
+ # Use this for tools whose APIs only make sense on one platform (Discord
102
+ # server admin, Slack workspace admin, etc.). Keeps every other platform's
103
+ # checklist from filling up with irrelevant toggles.
104
+ _TOOLSET_PLATFORM_RESTRICTIONS: Dict[str, Set[str]] = {
105
+ "discord": {"discord"},
106
+ "discord_admin": {"discord"},
107
+ }
108
+
109
+
110
+ def _toolset_allowed_for_platform(ts_key: str, platform: str) -> bool:
111
+ """Return True if ``ts_key`` is configurable on ``platform``.
112
+
113
+ Toolsets without a restriction entry are allowed everywhere (the default).
114
+ """
115
+ allowed = _TOOLSET_PLATFORM_RESTRICTIONS.get(ts_key)
116
+ return allowed is None or platform in allowed
117
+
118
+
119
+ def _get_effective_configurable_toolsets():
120
+ """Return CONFIGURABLE_TOOLSETS + any plugin-provided toolsets.
121
+
122
+ Plugin toolsets are appended at the end so they appear after the
123
+ built-in toolsets in the TUI checklist. A plugin whose toolset key
124
+ already appears in ``CONFIGURABLE_TOOLSETS`` is skipped — bundled
125
+ plugins (e.g. ``plugins/spotify``) share their toolset key with the
126
+ built-in entry, and we want the built-in label/description to win.
127
+ Without the dedupe, ``hermes tools`` → "reconfigure existing" would
128
+ list the same toolset twice.
129
+ """
130
+ result = list(CONFIGURABLE_TOOLSETS)
131
+ seen = {ts_key for ts_key, _, _ in result}
132
+ try:
133
+ from hermes_cli.plugins import discover_plugins, get_plugin_toolsets
134
+ discover_plugins() # idempotent — ensures plugins are loaded
135
+ for entry in get_plugin_toolsets():
136
+ if entry[0] in seen:
137
+ continue
138
+ seen.add(entry[0])
139
+ result.append(entry)
140
+ except Exception:
141
+ pass
142
+ return result
143
+
144
+
145
+ def _get_plugin_toolset_keys() -> set:
146
+ """Return the set of toolset keys provided by plugins."""
147
+ try:
148
+ from hermes_cli.plugins import discover_plugins, get_plugin_toolsets
149
+ discover_plugins() # idempotent — ensures plugins are loaded
150
+ return {ts_key for ts_key, _, _ in get_plugin_toolsets()}
151
+ except Exception:
152
+ return set()
153
+
154
+ # Platform display config — derived from the canonical registry so every
155
+ # module shares the same data. Kept as dict-of-dicts for backward
156
+ # compatibility with existing ``PLATFORMS[key]["label"]`` access patterns.
157
+ from hermes_cli.platforms import PLATFORMS as _PLATFORMS_REGISTRY
158
+
159
+ PLATFORMS = {
160
+ k: {"label": info.label, "default_toolset": info.default_toolset}
161
+ for k, info in _PLATFORMS_REGISTRY.items()
162
+ }
163
+
164
+
165
+ # ─── Tool Categories (provider-aware configuration) ──────────────────────────
166
+ # Maps toolset keys to their provider options. When a toolset is newly enabled,
167
+ # we use this to show provider selection and prompt for the right API keys.
168
+ # Toolsets not in this map either need no config or use the simple fallback.
169
+
170
+ TOOL_CATEGORIES = {
171
+ "tts": {
172
+ "name": "Text-to-Speech",
173
+ "icon": "🔊",
174
+ "providers": [
175
+ {
176
+ "name": "Nous Subscription",
177
+ "badge": "subscription",
178
+ "tag": "Managed OpenAI TTS billed to your subscription",
179
+ "env_vars": [],
180
+ "tts_provider": "openai",
181
+ "requires_nous_auth": True,
182
+ "managed_nous_feature": "tts",
183
+ "override_env_vars": ["VOICE_TOOLS_OPENAI_KEY", "OPENAI_API_KEY"],
184
+ },
185
+ {
186
+ "name": "Microsoft Edge TTS",
187
+ "badge": "★ recommended · free",
188
+ "tag": "Good quality, no API key needed",
189
+ "env_vars": [],
190
+ "tts_provider": "edge",
191
+ },
192
+ {
193
+ "name": "OpenAI TTS",
194
+ "badge": "paid",
195
+ "tag": "High quality voices",
196
+ "env_vars": [
197
+ {"key": "VOICE_TOOLS_OPENAI_KEY", "prompt": "OpenAI API key", "url": "https://platform.openai.com/api-keys"},
198
+ ],
199
+ "tts_provider": "openai",
200
+ },
201
+ {
202
+ "name": "xAI TTS",
203
+ "tag": "Grok voices — uses xAI Grok OAuth or XAI_API_KEY",
204
+ "env_vars": [],
205
+ "tts_provider": "xai",
206
+ "post_setup": "xai_grok",
207
+ },
208
+ {
209
+ "name": "ElevenLabs",
210
+ "badge": "paid",
211
+ "tag": "Most natural voices",
212
+ "env_vars": [
213
+ {"key": "ELEVENLABS_API_KEY", "prompt": "ElevenLabs API key", "url": "https://elevenlabs.io/app/settings/api-keys"},
214
+ ],
215
+ "tts_provider": "elevenlabs",
216
+ },
217
+ # Mistral (Voxtral TTS) temporarily hidden — `mistralai` PyPI
218
+ # package is currently quarantined (malicious 2.4.6 release on
219
+ # 2026-05-12). Restore this entry once PyPI un-quarantines.
220
+ {
221
+ "name": "Google Gemini TTS",
222
+ "badge": "preview",
223
+ "tag": "30 prebuilt voices, controllable via prompts",
224
+ "env_vars": [
225
+ {"key": "GEMINI_API_KEY", "prompt": "Gemini API key", "url": "https://aistudio.google.com/app/apikey"},
226
+ ],
227
+ "tts_provider": "gemini",
228
+ },
229
+ {
230
+ "name": "KittenTTS",
231
+ "badge": "local · free",
232
+ "tag": "Lightweight local ONNX TTS (~25MB), no API key",
233
+ "env_vars": [],
234
+ "tts_provider": "kittentts",
235
+ "post_setup": "kittentts",
236
+ },
237
+ {
238
+ "name": "Piper",
239
+ "badge": "local · free",
240
+ "tag": "Local neural TTS, 44 languages (voices ~20-90MB)",
241
+ "env_vars": [],
242
+ "tts_provider": "piper",
243
+ "post_setup": "piper",
244
+ },
245
+ ],
246
+ },
247
+ "web": {
248
+ "name": "Web Search & Extract",
249
+ "setup_title": "Select Search Provider",
250
+ "setup_note": "A free DuckDuckGo search skill is also included — skip this if you don't need a premium provider.",
251
+ "icon": "🔍",
252
+ # Per-provider rows are injected at runtime from
253
+ # plugins.web.<vendor>.provider via _plugin_web_search_providers()
254
+ # in _visible_providers(). Only non-provider UX setup-flow rows
255
+ # for the firecrawl backend are listed here:
256
+ # - "Nous Subscription" — managed Firecrawl billed via Nous
257
+ # subscription (requires_nous_auth + override_env_vars).
258
+ # - "Firecrawl Self-Hosted" — points firecrawl at a private
259
+ # Docker instance via FIRECRAWL_API_URL only.
260
+ # See PR #25182 for the migration rationale.
261
+ "providers": [
262
+ {
263
+ "name": "Nous Subscription",
264
+ "badge": "subscription",
265
+ "tag": "Managed Firecrawl billed to your subscription",
266
+ "web_backend": "firecrawl",
267
+ "env_vars": [],
268
+ "requires_nous_auth": True,
269
+ "managed_nous_feature": "web",
270
+ "override_env_vars": ["FIRECRAWL_API_KEY", "FIRECRAWL_API_URL"],
271
+ },
272
+ {
273
+ "name": "Firecrawl Self-Hosted",
274
+ "badge": "free · self-hosted",
275
+ "tag": "Run your own Firecrawl instance (Docker)",
276
+ "web_backend": "firecrawl",
277
+ "env_vars": [
278
+ {"key": "FIRECRAWL_API_URL", "prompt": "Your Firecrawl instance URL (e.g., http://localhost:3002)"},
279
+ ],
280
+ },
281
+ ],
282
+ },
283
+ "image_gen": {
284
+ "name": "Image Generation",
285
+ "icon": "🎨",
286
+ "providers": [
287
+ {
288
+ "name": "Nous Subscription",
289
+ "badge": "subscription",
290
+ "tag": "Managed FAL image generation billed to your subscription",
291
+ "env_vars": [],
292
+ "requires_nous_auth": True,
293
+ "managed_nous_feature": "image_gen",
294
+ "override_env_vars": ["FAL_KEY"],
295
+ "imagegen_backend": "fal",
296
+ },
297
+ {
298
+ "name": "FAL.ai",
299
+ "badge": "paid",
300
+ "tag": "Pick from flux-2-klein, flux-2-pro, gpt-image, nano-banana, etc.",
301
+ "env_vars": [
302
+ {"key": "FAL_KEY", "prompt": "FAL API key", "url": "https://fal.ai/dashboard/keys"},
303
+ ],
304
+ "imagegen_backend": "fal",
305
+ },
306
+ ],
307
+ },
308
+ "video_gen": {
309
+ "name": "Video Generation",
310
+ "icon": "🎬",
311
+ # Providers list is intentionally empty — every video gen backend
312
+ # is a plugin, surfaced by ``_plugin_video_gen_providers()`` and
313
+ # injected by ``_visible_providers``. Mirrors the design we'll
314
+ # converge image_gen toward.
315
+ "providers": [],
316
+ },
317
+ "x_search": {
318
+ "name": "X (Twitter) Search",
319
+ "setup_title": "Select xAI Credential Source",
320
+ "setup_note": (
321
+ "Hermes routes X searches through xAI's built-in x_search "
322
+ "Responses tool. Both credential sources hit the same "
323
+ "https://api.x.ai/v1/responses endpoint — pick whichever you "
324
+ "already have. SuperGrok OAuth is preferred when both are set "
325
+ "(uses your subscription quota instead of API spend)."
326
+ ),
327
+ "icon": "🐦",
328
+ "providers": [
329
+ {
330
+ "name": "xAI Grok OAuth (SuperGrok Subscription)",
331
+ "badge": "subscription",
332
+ "tag": "Browser login at accounts.x.ai — no API key required",
333
+ "env_vars": [],
334
+ "post_setup": "xai_grok",
335
+ },
336
+ {
337
+ "name": "xAI API key",
338
+ "badge": "paid",
339
+ "tag": "Direct xAI API billing via XAI_API_KEY",
340
+ "env_vars": [
341
+ {
342
+ "key": "XAI_API_KEY",
343
+ "prompt": "xAI API key",
344
+ "url": "https://console.x.ai/",
345
+ },
346
+ ],
347
+ },
348
+ ],
349
+ },
350
+ "browser": {
351
+ "name": "Browser Automation",
352
+ "icon": "🌐",
353
+ "providers": [
354
+ {
355
+ "name": "Nous Subscription (Browser Use cloud)",
356
+ "badge": "subscription",
357
+ "tag": "Managed Browser Use billed to your subscription",
358
+ "env_vars": [],
359
+ "browser_provider": "browser-use",
360
+ "requires_nous_auth": True,
361
+ "managed_nous_feature": "browser",
362
+ "override_env_vars": ["BROWSER_USE_API_KEY"],
363
+ "post_setup": "agent_browser",
364
+ },
365
+ {
366
+ "name": "Local Browser",
367
+ "badge": "★ recommended · free",
368
+ "tag": "Headless Chromium, no API key needed",
369
+ "env_vars": [],
370
+ "browser_provider": "local",
371
+ "post_setup": "agent_browser",
372
+ },
373
+ {
374
+ "name": "Browserbase",
375
+ "badge": "paid",
376
+ "tag": "Cloud browser with stealth and proxies",
377
+ "env_vars": [
378
+ {"key": "BROWSERBASE_API_KEY", "prompt": "Browserbase API key", "url": "https://browserbase.com"},
379
+ {"key": "BROWSERBASE_PROJECT_ID", "prompt": "Browserbase project ID"},
380
+ ],
381
+ "browser_provider": "browserbase",
382
+ "post_setup": "agent_browser",
383
+ },
384
+ {
385
+ "name": "Browser Use",
386
+ "badge": "paid",
387
+ "tag": "Cloud browser with remote execution",
388
+ "env_vars": [
389
+ {"key": "BROWSER_USE_API_KEY", "prompt": "Browser Use API key", "url": "https://browser-use.com"},
390
+ ],
391
+ "browser_provider": "browser-use",
392
+ "post_setup": "agent_browser",
393
+ },
394
+ {
395
+ "name": "Firecrawl",
396
+ "badge": "paid",
397
+ "tag": "Cloud browser with remote execution",
398
+ "env_vars": [
399
+ {"key": "FIRECRAWL_API_KEY", "prompt": "Firecrawl API key", "url": "https://firecrawl.dev"},
400
+ ],
401
+ "browser_provider": "firecrawl",
402
+ "post_setup": "agent_browser",
403
+ },
404
+ {
405
+ "name": "Camofox",
406
+ "badge": "free · local",
407
+ "tag": "Anti-detection browser (Firefox/Camoufox)",
408
+ "env_vars": [
409
+ {"key": "CAMOFOX_URL", "prompt": "Camofox server URL", "default": "http://localhost:9377",
410
+ "url": "https://github.com/jo-inc/camofox-browser"},
411
+ ],
412
+ "browser_provider": "camofox",
413
+ "post_setup": "camofox",
414
+ },
415
+ ],
416
+ },
417
+ "homeassistant": {
418
+ "name": "Smart Home",
419
+ "icon": "🏠",
420
+ "providers": [
421
+ {
422
+ "name": "Home Assistant",
423
+ "tag": "REST API integration",
424
+ "env_vars": [
425
+ {"key": "HASS_TOKEN", "prompt": "Home Assistant Long-Lived Access Token"},
426
+ {"key": "HASS_URL", "prompt": "Home Assistant URL", "default": "http://homeassistant.local:8123"},
427
+ ],
428
+ },
429
+ ],
430
+ },
431
+ "spotify": {
432
+ "name": "Spotify",
433
+ "icon": "🎵",
434
+ "providers": [
435
+ {
436
+ "name": "Spotify Web API",
437
+ "tag": "PKCE OAuth — opens the setup wizard",
438
+ "env_vars": [],
439
+ "post_setup": "spotify",
440
+ },
441
+ ],
442
+ },
443
+ "computer_use": {
444
+ "name": "Computer Use (macOS)",
445
+ "icon": "🖱️",
446
+ "platform_gate": "darwin",
447
+ "providers": [
448
+ {
449
+ "name": "cua-driver (background)",
450
+ "badge": "★ recommended · free · local",
451
+ "tag": (
452
+ "macOS background computer-use via SkyLight SPIs — does "
453
+ "NOT steal your cursor or focus. Works with any model."
454
+ ),
455
+ "env_vars": [
456
+ # cua-driver reads HOME/TMPDIR from the process env, no
457
+ # extra keys required. HERMES_CUA_DRIVER_VERSION is an
458
+ # optional pin for reproducibility across macOS updates.
459
+ ],
460
+ "post_setup": "cua_driver",
461
+ },
462
+ ],
463
+ },
464
+ "langfuse": {
465
+ "name": "Langfuse Observability",
466
+ "icon": "📊",
467
+ "providers": [
468
+ {
469
+ "name": "Langfuse Cloud",
470
+ "tag": "Hosted Langfuse (cloud.langfuse.com)",
471
+ "env_vars": [
472
+ {"key": "HERMES_LANGFUSE_PUBLIC_KEY", "prompt": "Langfuse public key (pk-lf-...)", "url": "https://cloud.langfuse.com"},
473
+ {"key": "HERMES_LANGFUSE_SECRET_KEY", "prompt": "Langfuse secret key (sk-lf-...)", "url": "https://cloud.langfuse.com"},
474
+ ],
475
+ "post_setup": "langfuse",
476
+ },
477
+ {
478
+ "name": "Langfuse Self-Hosted",
479
+ "tag": "Self-hosted Langfuse instance",
480
+ "env_vars": [
481
+ {"key": "HERMES_LANGFUSE_PUBLIC_KEY", "prompt": "Langfuse public key (pk-lf-...)"},
482
+ {"key": "HERMES_LANGFUSE_SECRET_KEY", "prompt": "Langfuse secret key (sk-lf-...)"},
483
+ {"key": "HERMES_LANGFUSE_BASE_URL", "prompt": "Langfuse server URL (e.g. http://localhost:3000)", "default": "http://localhost:3000"},
484
+ ],
485
+ "post_setup": "langfuse",
486
+ },
487
+ ],
488
+ },
489
+ }
490
+
491
+ # Simple env-var requirements for toolsets NOT in TOOL_CATEGORIES.
492
+ # Used as a fallback for tools like vision/moa that just need an API key.
493
+ TOOLSET_ENV_REQUIREMENTS = {
494
+ "vision": [("OPENROUTER_API_KEY", "https://openrouter.ai/keys")],
495
+ "moa": [("OPENROUTER_API_KEY", "https://openrouter.ai/keys")],
496
+ }
497
+
498
+
499
+ # ─── Post-Setup Hooks ─────────────────────────────────────────────────────────
500
+
501
+
502
+ def _pip_install(
503
+ args: List[str],
504
+ *,
505
+ timeout: int = 300,
506
+ capture_output: bool = True,
507
+ ):
508
+ """Install Python packages from a post-setup hook.
509
+
510
+ Strategy (in order):
511
+ 1. ``uv pip install`` if uv is on PATH — fast, doesn't need pip in the venv.
512
+ 2. ``python -m pip install`` — works on stdlib venvs.
513
+ 3. ``python -m ensurepip --upgrade`` then retry pip — covers ``uv venv``
514
+ which creates a venv WITHOUT pip.
515
+
516
+ Why this exists: the Windows installer creates the venv via ``uv venv``,
517
+ which doesn't seed pip. Post-setup hooks that shelled out to
518
+ ``[sys.executable, '-m', 'pip', 'install', ...]`` failed with
519
+ ``No module named pip`` on every fresh install. uv-first sidesteps that.
520
+
521
+ Returns the ``subprocess.CompletedProcess`` from whichever tier succeeded
522
+ (or the last failure for the caller to inspect).
523
+ """
524
+ venv_root = Path(sys.executable).parent.parent
525
+ uv_env = {**os.environ, "VIRTUAL_ENV": str(venv_root)}
526
+
527
+ uv_bin = shutil.which("uv")
528
+ if uv_bin:
529
+ try:
530
+ result = subprocess.run(
531
+ [uv_bin, "pip", "install", *args],
532
+ capture_output=capture_output, text=True, timeout=timeout,
533
+ env=uv_env,
534
+ )
535
+ if result.returncode == 0:
536
+ return result
537
+ # Fall through to pip — uv may have failed for an unrelated reason
538
+ # (resolution conflict, network), and pip might handle it.
539
+ except (subprocess.TimeoutExpired, FileNotFoundError):
540
+ pass
541
+
542
+ pip_cmd = [sys.executable, "-m", "pip"]
543
+ try:
544
+ # Probe for pip; bootstrap via ensurepip if missing (uv venv lacks it).
545
+ probe = subprocess.run(
546
+ pip_cmd + ["--version"],
547
+ capture_output=True, text=True, timeout=15,
548
+ )
549
+ if probe.returncode != 0:
550
+ raise FileNotFoundError("pip not in venv")
551
+ except (subprocess.TimeoutExpired, FileNotFoundError):
552
+ try:
553
+ subprocess.run(
554
+ [sys.executable, "-m", "ensurepip", "--upgrade", "--default-pip"],
555
+ capture_output=True, text=True, timeout=120, check=True,
556
+ )
557
+ except (subprocess.CalledProcessError, subprocess.TimeoutExpired) as e:
558
+ # Synthesize a result so callers see a clean failure path.
559
+ return subprocess.CompletedProcess(
560
+ pip_cmd, returncode=1, stdout="",
561
+ stderr=f"pip not available and ensurepip failed: {e}",
562
+ )
563
+
564
+ return subprocess.run(
565
+ pip_cmd + ["install", *args],
566
+ capture_output=capture_output, text=True, timeout=timeout,
567
+ )
568
+
569
+
570
+ def install_cua_driver(upgrade: bool = False) -> bool:
571
+ """Install or refresh the cua-driver binary used by Computer Use.
572
+
573
+ The upstream installer always pulls the latest release tag, so re-running
574
+ it is the canonical way to upgrade. We expose two modes:
575
+
576
+ * ``upgrade=False`` — original post-setup behaviour: skip if already
577
+ installed, install otherwise. Used by the toolset enable flow where
578
+ we don't want to surprise the user with a network fetch.
579
+ * ``upgrade=True`` — always re-run the installer (or call ``cua-driver
580
+ update`` if the binary supports it). Used by ``hermes update`` and
581
+ by ``hermes computer-use install --upgrade``.
582
+
583
+ Returns True iff cua-driver is installed (or successfully refreshed)
584
+ when the function returns. macOS-only — silently returns False on
585
+ other platforms.
586
+ """
587
+ import platform as _plat
588
+ import shutil
589
+ import subprocess
590
+
591
+ if _plat.system() != "Darwin":
592
+ if upgrade:
593
+ # Silent on non-macOS — `hermes update` calls this for every
594
+ # user; only macOS users with cua-driver care.
595
+ return False
596
+ _print_warning(" Computer Use (cua-driver) is macOS-only; skipping.")
597
+ return False
598
+
599
+ binary = shutil.which("cua-driver")
600
+
601
+ # Not installed → fresh install path (only when caller asked for it).
602
+ if not binary and not upgrade:
603
+ if not shutil.which("curl"):
604
+ _print_warning(" curl not found — install manually:")
605
+ _print_info(" https://github.com/trycua/cua/blob/main/libs/cua-driver/README.md")
606
+ return False
607
+ return _run_cua_driver_installer(label="Installing")
608
+
609
+ # Already installed and caller didn't ask to upgrade → just confirm.
610
+ if binary and not upgrade:
611
+ try:
612
+ version = subprocess.run(
613
+ ["cua-driver", "--version"],
614
+ capture_output=True, text=True, timeout=5,
615
+ ).stdout.strip()
616
+ _print_success(f" cua-driver already installed: {version or 'unknown version'}")
617
+ except Exception:
618
+ _print_success(" cua-driver already installed.")
619
+ _print_info(" Grant macOS permissions if not done yet:")
620
+ _print_info(" System Settings > Privacy & Security > Accessibility")
621
+ _print_info(" System Settings > Privacy & Security > Screen Recording")
622
+ return True
623
+
624
+ # upgrade=True path — refresh to the latest upstream release.
625
+ if not shutil.which("curl"):
626
+ _print_warning(" curl not found — cannot refresh cua-driver.")
627
+ return bool(binary)
628
+
629
+ if binary:
630
+ # Show before/after version when we have a baseline. Best-effort.
631
+ try:
632
+ before = subprocess.run(
633
+ ["cua-driver", "--version"],
634
+ capture_output=True, text=True, timeout=5,
635
+ ).stdout.strip()
636
+ except Exception:
637
+ before = ""
638
+ else:
639
+ before = ""
640
+
641
+ ok = _run_cua_driver_installer(label="Refreshing", verbose=False)
642
+ if ok and before:
643
+ try:
644
+ after = subprocess.run(
645
+ ["cua-driver", "--version"],
646
+ capture_output=True, text=True, timeout=5,
647
+ ).stdout.strip()
648
+ if after and after != before:
649
+ _print_success(f" cua-driver upgraded: {before} → {after}")
650
+ elif after:
651
+ _print_info(f" cua-driver up to date: {after}")
652
+ except Exception:
653
+ pass
654
+ return ok
655
+
656
+
657
+ def _run_cua_driver_installer(label: str = "Installing", verbose: bool = True) -> bool:
658
+ """Run the upstream cua-driver install.sh. Returns True on success.
659
+
660
+ The script is idempotent: it always downloads the latest release, so
661
+ re-running it on an already-installed system performs an upgrade.
662
+ """
663
+ import shutil
664
+ import subprocess
665
+
666
+ install_cmd = (
667
+ "/bin/bash -c \"$(curl -fsSL "
668
+ "https://raw.githubusercontent.com/trycua/cua/main/"
669
+ "libs/cua-driver/scripts/install.sh)\""
670
+ )
671
+ if verbose:
672
+ _print_info(f" {label} cua-driver (macOS background computer-use)...")
673
+ else:
674
+ _print_info(f" {label} cua-driver...")
675
+ try:
676
+ result = subprocess.run(install_cmd, shell=True, timeout=300)
677
+ if result.returncode == 0 and shutil.which("cua-driver"):
678
+ if verbose:
679
+ _print_success(" cua-driver installed.")
680
+ _print_info(" IMPORTANT — grant macOS permissions now:")
681
+ _print_info(" System Settings > Privacy & Security > Accessibility")
682
+ _print_info(" System Settings > Privacy & Security > Screen Recording")
683
+ _print_info(" Both must allow the terminal / Hermes process.")
684
+ return True
685
+ _print_warning(f" cua-driver {label.lower()} did not complete. Re-run manually:")
686
+ _print_info(f" {install_cmd}")
687
+ return False
688
+ except subprocess.TimeoutExpired:
689
+ _print_warning(f" cua-driver {label.lower()} timed out. Re-run manually.")
690
+ return False
691
+ except Exception as e:
692
+ _print_warning(f" cua-driver {label.lower()} failed: {e}")
693
+ return False
694
+
695
+
696
+ def _run_post_setup(post_setup_key: str):
697
+ """Run post-setup hooks for tools that need extra installation steps."""
698
+ import shutil
699
+ if post_setup_key in {"agent_browser", "browserbase"}:
700
+ node_modules = PROJECT_ROOT / "node_modules" / "agent-browser"
701
+ npm_bin = shutil.which("npm")
702
+ npx_bin = shutil.which("npx")
703
+ # Step 1: install the agent-browser npm package into node_modules/
704
+ if not node_modules.exists() and npm_bin:
705
+ _print_info(" Installing Node.js dependencies for browser tools...")
706
+ import subprocess
707
+ # Use the resolved npm_bin absolute path so subprocess.Popen can
708
+ # execute npm.cmd on Windows (CreateProcessW otherwise rejects
709
+ # batch shims). On POSIX npm_bin is the plain path — same
710
+ # behaviour as before.
711
+ result = subprocess.run(
712
+ [npm_bin, "install", "--silent"],
713
+ capture_output=True, text=True, cwd=str(PROJECT_ROOT)
714
+ )
715
+ if result.returncode == 0:
716
+ _print_success(" Node.js dependencies installed")
717
+ else:
718
+ from calvyn_constants import display_hermes_home
719
+ _print_warning(f" npm install failed - run manually: cd {display_hermes_home()}/hermes-agent && npm install")
720
+ if result.stderr:
721
+ _print_info(f" {result.stderr.strip()[:200]}")
722
+ elif not node_modules.exists():
723
+ _print_warning(" Node.js not found - browser tools require: npm install (in hermes-agent directory)")
724
+ return
725
+
726
+ # Step 2: only the local browser provider actually needs Chromium on
727
+ # disk. Cloud providers (Browserbase, Browser Use, Firecrawl) host
728
+ # their own Chromium and don't need the local install.
729
+ if post_setup_key != "agent_browser":
730
+ return
731
+
732
+ # Step 3: ensure the Chromium / headless-shell build agent-browser
733
+ # drives is actually installed. Without it the CLI hangs on first
734
+ # use until the command timeout fires. Skip inside Docker — the
735
+ # image bakes Chromium in at build time, and runtime users usually
736
+ # can't write to PLAYWRIGHT_BROWSERS_PATH anyway.
737
+ try:
738
+ # Import lazily so the tools_config UI doesn't pull in the full
739
+ # browser_tool module at import time.
740
+ from tools.browser_tool import (
741
+ _chromium_installed,
742
+ _running_in_docker,
743
+ )
744
+ except Exception as exc: # pragma: no cover — defensive
745
+ _print_warning(f" Could not check Chromium status: {exc}")
746
+ return
747
+
748
+ if _chromium_installed():
749
+ _print_success(" Chromium browser already installed")
750
+ return
751
+
752
+ if _running_in_docker():
753
+ _print_warning(
754
+ " Chromium is missing but you're running in Docker."
755
+ )
756
+ _print_info(
757
+ " Pull the latest image to get the bundled Chromium:"
758
+ )
759
+ _print_info(
760
+ " docker pull ghcr.io/nousresearch/hermes-agent:latest"
761
+ )
762
+ return
763
+
764
+ if not npx_bin:
765
+ _print_warning(
766
+ " npx not found - install Chromium manually: npx agent-browser install --with-deps"
767
+ )
768
+ return
769
+
770
+ _print_info(" Installing Chromium (~170MB one-time download)...")
771
+ import subprocess
772
+ # Prefer the bundled agent-browser install subcommand so the
773
+ # version of Chromium matches the CLI. Fall back to npx shim on
774
+ # setups where the local bin stub isn't present.
775
+ local_ab = PROJECT_ROOT / "node_modules" / ".bin" / "agent-browser"
776
+ if sys.platform == "win32":
777
+ local_ab_win = local_ab.with_suffix(".cmd")
778
+ if local_ab_win.exists():
779
+ local_ab = local_ab_win
780
+ install_cmd = (
781
+ [str(local_ab), "install", "--with-deps"]
782
+ if local_ab.exists()
783
+ else [npx_bin, "-y", "agent-browser", "install", "--with-deps"]
784
+ )
785
+ try:
786
+ result = subprocess.run(
787
+ install_cmd,
788
+ capture_output=True, text=True, cwd=str(PROJECT_ROOT), timeout=600,
789
+ )
790
+ if result.returncode == 0:
791
+ _print_success(" Chromium installed")
792
+ # Invalidate the cached "missing" result so subsequent
793
+ # check_browser_requirements() calls see the new install.
794
+ import tools.browser_tool as _bt
795
+ _bt._cached_chromium_installed = None
796
+ else:
797
+ _print_warning(" Chromium install failed:")
798
+ tail = (result.stderr or result.stdout or "").strip().splitlines()[-3:]
799
+ for line in tail:
800
+ _print_info(f" {line[:200]}")
801
+ _print_info(" Run manually: npx agent-browser install --with-deps")
802
+ except subprocess.TimeoutExpired:
803
+ _print_warning(" Chromium install timed out (>10min)")
804
+ _print_info(" Run manually: npx agent-browser install --with-deps")
805
+ except Exception as exc:
806
+ _print_warning(f" Chromium install failed: {exc}")
807
+ _print_info(" Run manually: npx agent-browser install --with-deps")
808
+
809
+ elif post_setup_key == "camofox":
810
+ camofox_dir = PROJECT_ROOT / "node_modules" / "@askjo" / "camofox-browser"
811
+ _npm_bin = shutil.which("npm")
812
+ if not camofox_dir.exists() and _npm_bin:
813
+ _print_info(" Installing Camofox browser server...")
814
+ import subprocess
815
+ # Absolute npm path so .cmd shim executes on Windows.
816
+ result = subprocess.run(
817
+ [_npm_bin, "install", "--silent"],
818
+ capture_output=True, text=True, cwd=str(PROJECT_ROOT)
819
+ )
820
+ if result.returncode == 0:
821
+ _print_success(" Camofox installed")
822
+ else:
823
+ _print_warning(" npm install failed - run manually: npm install")
824
+ if camofox_dir.exists():
825
+ _print_info(" Start the Camofox server:")
826
+ _print_info(" npx @askjo/camofox-browser")
827
+ _print_info(" First run downloads the Camoufox engine (~300MB)")
828
+ _print_info(" Or use Docker: docker run -p 9377:9377 -e CAMOFOX_PORT=9377 jo-inc/camofox-browser")
829
+ elif not shutil.which("npm"):
830
+ _print_warning(" Node.js not found. Install Camofox via Docker:")
831
+ _print_info(" docker run -p 9377:9377 -e CAMOFOX_PORT=9377 jo-inc/camofox-browser")
832
+
833
+ elif post_setup_key == "cua_driver":
834
+ install_cua_driver(upgrade=False)
835
+
836
+ elif post_setup_key == "kittentts":
837
+ try:
838
+ __import__("kittentts")
839
+ _print_success(" kittentts is already installed")
840
+ return
841
+ except ImportError:
842
+ pass
843
+ _print_info(" Installing kittentts (~25-80MB model, CPU-only)...")
844
+ wheel_url = (
845
+ "https://github.com/KittenML/KittenTTS/releases/download/"
846
+ "0.8.1/kittentts-0.8.1-py3-none-any.whl"
847
+ )
848
+ try:
849
+ result = _pip_install(["-U", wheel_url, "soundfile", "--quiet"], timeout=300)
850
+ if result.returncode == 0:
851
+ _print_success(" kittentts installed")
852
+ _print_info(" Voices: Jasper, Bella, Luna, Bruno, Rosie, Hugo, Kiki, Leo")
853
+ _print_info(" Models: KittenML/kitten-tts-nano-0.8-int8 (25MB), micro (41MB), mini (80MB)")
854
+ else:
855
+ _print_warning(" kittentts install failed:")
856
+ _print_info(f" {(result.stderr or '').strip()[:300]}")
857
+ _print_info(f" Run manually: uv pip install -U '{wheel_url}' soundfile")
858
+ except subprocess.TimeoutExpired:
859
+ _print_warning(" kittentts install timed out (>5min)")
860
+ _print_info(f" Run manually: uv pip install -U '{wheel_url}' soundfile")
861
+
862
+ elif post_setup_key == "piper":
863
+ try:
864
+ __import__("piper")
865
+ _print_success(" piper-tts is already installed")
866
+ except ImportError:
867
+ _print_info(" Installing piper-tts (~14MB wheel, voices downloaded on first use)...")
868
+ try:
869
+ result = _pip_install(["-U", "piper-tts", "--quiet"], timeout=300)
870
+ if result.returncode == 0:
871
+ _print_success(" piper-tts installed")
872
+ else:
873
+ _print_warning(" piper-tts install failed:")
874
+ _print_info(f" {(result.stderr or '').strip()[:300]}")
875
+ _print_info(" Run manually: uv pip install -U piper-tts")
876
+ return
877
+ except subprocess.TimeoutExpired:
878
+ _print_warning(" piper-tts install timed out (>5min)")
879
+ _print_info(" Run manually: uv pip install -U piper-tts")
880
+ return
881
+ _print_info(" Default voice: en_US-lessac-medium (downloaded on first TTS call)")
882
+ _print_info(" Full voice list: https://github.com/OHF-Voice/piper1-gpl/blob/main/docs/VOICES.md")
883
+ _print_info(" Switch voices by setting tts.piper.voice in ~/.hermes/config.yaml")
884
+
885
+ elif post_setup_key == "ddgs":
886
+ try:
887
+ __import__("ddgs")
888
+ _print_success(" ddgs is already installed")
889
+ except ImportError:
890
+ _print_info(" Installing ddgs (DuckDuckGo search package)...")
891
+ try:
892
+ result = _pip_install(["-U", "ddgs", "--quiet"], timeout=300)
893
+ if result.returncode == 0:
894
+ _print_success(" ddgs installed")
895
+ else:
896
+ _print_warning(" ddgs install failed:")
897
+ _print_info(f" {(result.stderr or '').strip()[:300]}")
898
+ _print_info(" Run manually: uv pip install -U ddgs")
899
+ return
900
+ except subprocess.TimeoutExpired:
901
+ _print_warning(" ddgs install timed out (>5min)")
902
+ _print_info(" Run manually: uv pip install -U ddgs")
903
+ return
904
+ _print_info(" No API key required. DuckDuckGo enforces server-side rate limits.")
905
+ _print_info(" Pair with an extract provider if you also need web_extract.")
906
+
907
+ elif post_setup_key == "spotify":
908
+ # Run the full `hermes auth spotify` flow — if the user has no
909
+ # client_id yet, this drops them into the interactive wizard
910
+ # (opens the Spotify dashboard, prompts for client_id, persists
911
+ # to ~/.hermes/.env), then continues straight into PKCE. If they
912
+ # already have an app, it skips the wizard and just does OAuth.
913
+ from types import SimpleNamespace
914
+ try:
915
+ from hermes_cli.auth import login_spotify_command
916
+ except Exception as exc:
917
+ _print_warning(f" Could not load Spotify auth: {exc}")
918
+ _print_info(" Run manually: hermes auth spotify")
919
+ return
920
+ _print_info(" Starting Spotify login...")
921
+ try:
922
+ login_spotify_command(SimpleNamespace(
923
+ client_id=None, redirect_uri=None, scope=None,
924
+ no_browser=False, timeout=None,
925
+ ))
926
+ _print_success(" Spotify authenticated")
927
+ except SystemExit as exc:
928
+ # User aborted the wizard, or OAuth failed — don't fail the
929
+ # toolset enable; they can retry with `hermes auth spotify`.
930
+ _print_warning(f" Spotify login did not complete: {exc}")
931
+ _print_info(" Run later: hermes auth spotify")
932
+ except Exception as exc:
933
+ _print_warning(f" Spotify login failed: {exc}")
934
+ _print_info(" Run manually: hermes auth spotify")
935
+
936
+ elif post_setup_key == "langfuse":
937
+ # Install the langfuse SDK.
938
+ try:
939
+ __import__("langfuse")
940
+ _print_success(" langfuse SDK already installed")
941
+ except ImportError:
942
+ _print_info(" Installing langfuse SDK...")
943
+ result = _pip_install(["langfuse", "--quiet"], timeout=120)
944
+ if result.returncode == 0:
945
+ _print_success(" langfuse SDK installed")
946
+ else:
947
+ _print_warning(" langfuse SDK install failed — run manually: uv pip install langfuse")
948
+ # Opt the bundled observability/langfuse plugin into plugins.enabled.
949
+ # The plugin ships in the repo but doesn't load until the user enables
950
+ # it (standalone plugins are opt-in).
951
+ try:
952
+ from hermes_cli.plugins_cmd import _get_enabled_set, _save_enabled_set
953
+ enabled = _get_enabled_set()
954
+ if "observability/langfuse" in enabled or "langfuse" in enabled:
955
+ _print_success(" Plugin observability/langfuse already enabled")
956
+ else:
957
+ enabled.add("observability/langfuse")
958
+ _save_enabled_set(enabled)
959
+ _print_success(" Plugin observability/langfuse enabled")
960
+ except Exception as exc:
961
+ _print_warning(f" Could not enable plugin automatically: {exc}")
962
+ _print_info(" Run manually: hermes plugins enable observability/langfuse")
963
+ _print_info(" Restart Hermes for tracing to take effect.")
964
+ _print_info(" Verify: hermes plugins list")
965
+
966
+ elif post_setup_key == "xai_grok":
967
+ # Shared credential bootstrap for any picker entry that talks to xAI
968
+ # (TTS, Video Gen, future Image Gen, etc.). Accepts either a
969
+ # SuperGrok-tier OAuth bearer token (preferred — billed against the
970
+ # user's existing subscription) or a raw XAI_API_KEY from
971
+ # console.x.ai. The picker entries declare empty env_vars so we
972
+ # drive the full auth UX here.
973
+ try:
974
+ from hermes_cli.auth import get_xai_oauth_auth_status
975
+ oauth_logged_in = bool(get_xai_oauth_auth_status().get("logged_in"))
976
+ except Exception:
977
+ oauth_logged_in = False
978
+ existing_api_key = get_env_value("XAI_API_KEY")
979
+
980
+ if oauth_logged_in:
981
+ _print_success(
982
+ " xAI will use your xAI Grok OAuth (SuperGrok Subscription) credentials"
983
+ )
984
+ return
985
+ if existing_api_key:
986
+ _print_success(" xAI will use your existing XAI_API_KEY")
987
+ return
988
+
989
+ _print_info(" xAI needs credentials. Choose one:")
990
+ try:
991
+ from hermes_cli.setup import (
992
+ _run_xai_oauth_login_from_setup,
993
+ prompt_choice,
994
+ prompt as _setup_prompt,
995
+ )
996
+ from hermes_cli.config import save_env_value
997
+ except Exception as exc:
998
+ _print_warning(f" Could not load setup helpers: {exc}")
999
+ _print_info(" Run later: hermes auth add xai-oauth (or set XAI_API_KEY)")
1000
+ return
1001
+
1002
+ idx = prompt_choice(
1003
+ " How do you want xAI to authenticate?",
1004
+ choices=[
1005
+ "Sign in with xAI Grok OAuth (SuperGrok Subscription) — browser login",
1006
+ "Paste an xAI API key (console.x.ai)",
1007
+ "Skip — configure later via `hermes auth add xai-oauth`",
1008
+ ],
1009
+ default=0,
1010
+ )
1011
+ if idx == 0:
1012
+ if _run_xai_oauth_login_from_setup():
1013
+ _print_success(
1014
+ " Logged in — xAI will use these OAuth credentials"
1015
+ )
1016
+ else:
1017
+ _print_warning(
1018
+ " xAI Grok OAuth login did not complete. "
1019
+ "Run later: hermes auth add xai-oauth"
1020
+ )
1021
+ elif idx == 1:
1022
+ api_key = _setup_prompt(" xAI API key", password=True)
1023
+ if api_key:
1024
+ save_env_value("XAI_API_KEY", api_key)
1025
+ _print_success(" XAI_API_KEY saved")
1026
+ else:
1027
+ _print_warning(
1028
+ " No API key provided. Run later: hermes auth add xai-oauth"
1029
+ )
1030
+ else:
1031
+ _print_info(" xAI will remain inactive until credentials are configured.")
1032
+
1033
+
1034
+ # ─── Platform / Toolset Helpers ───────────────────────────────────────────────
1035
+
1036
+ def _get_enabled_platforms() -> List[str]:
1037
+ """Return platform keys that are configured (have tokens or are CLI)."""
1038
+ enabled = ["cli"]
1039
+ if get_env_value("TELEGRAM_BOT_TOKEN"):
1040
+ enabled.append("telegram")
1041
+ if get_env_value("DISCORD_BOT_TOKEN"):
1042
+ enabled.append("discord")
1043
+ if get_env_value("SLACK_BOT_TOKEN"):
1044
+ enabled.append("slack")
1045
+ if get_env_value("WHATSAPP_ENABLED"):
1046
+ enabled.append("whatsapp")
1047
+ if get_env_value("QQ_APP_ID"):
1048
+ enabled.append("qqbot")
1049
+ return enabled
1050
+
1051
+
1052
+ def _platform_toolset_summary(config: dict, platforms: Optional[List[str]] = None) -> Dict[str, Set[str]]:
1053
+ """Return a summary of enabled toolsets per platform.
1054
+
1055
+ When ``platforms`` is None, this uses ``_get_enabled_platforms`` to
1056
+ auto-detect platforms. Tests can pass an explicit list to avoid relying
1057
+ on environment variables.
1058
+ """
1059
+ if platforms is None:
1060
+ platforms = _get_enabled_platforms()
1061
+
1062
+ summary: Dict[str, Set[str]] = {}
1063
+ for pkey in platforms:
1064
+ summary[pkey] = _get_platform_tools(config, pkey)
1065
+ return summary
1066
+
1067
+
1068
+ def _parse_enabled_flag(value, default: bool = True) -> bool:
1069
+ """Parse bool-like config values used by tool/platform settings."""
1070
+ if value is None:
1071
+ return default
1072
+ if isinstance(value, bool):
1073
+ return value
1074
+ if isinstance(value, int):
1075
+ return value != 0
1076
+ if isinstance(value, str):
1077
+ lowered = value.strip().lower()
1078
+ if lowered in {"true", "1", "yes", "on"}:
1079
+ return True
1080
+ if lowered in {"false", "0", "no", "off"}:
1081
+ return False
1082
+ return default
1083
+
1084
+
1085
+ def _get_platform_tools(
1086
+ config: dict,
1087
+ platform: str,
1088
+ *,
1089
+ include_default_mcp_servers: bool = True,
1090
+ ) -> Set[str]:
1091
+ """Resolve which individual toolset names are enabled for a platform."""
1092
+ from toolsets import resolve_toolset, TOOLSETS
1093
+
1094
+ platform_toolsets = config.get("platform_toolsets") or {}
1095
+ toolset_names = platform_toolsets.get(platform)
1096
+
1097
+ if toolset_names is None or not isinstance(toolset_names, list):
1098
+ plat_info = PLATFORMS.get(platform)
1099
+ if plat_info:
1100
+ default_ts = plat_info["default_toolset"]
1101
+ else:
1102
+ # Plugin platform — derive toolset name from platform key
1103
+ default_ts = f"hermes-{platform}"
1104
+ toolset_names = [default_ts]
1105
+
1106
+ # YAML may parse bare numeric names (e.g. ``12306:``) as int.
1107
+ # Normalise to str so downstream sorted() never mixes types.
1108
+ toolset_names = [str(ts) for ts in toolset_names]
1109
+
1110
+ configurable_keys = {ts_key for ts_key, _, _ in CONFIGURABLE_TOOLSETS}
1111
+ plugin_ts_keys = _get_plugin_toolset_keys()
1112
+ platform_default_keys = {p["default_toolset"] for p in PLATFORMS.values()}
1113
+
1114
+ # If the saved list contains any configurable keys directly, the user
1115
+ # has explicitly configured this platform — use direct membership.
1116
+ # This avoids the subset-inference bug where composite toolsets like
1117
+ # "hermes-cli" (which include all _HERMES_CORE_TOOLS) cause disabled
1118
+ # toolsets to re-appear as enabled.
1119
+ has_explicit_config = any(ts in configurable_keys for ts in toolset_names)
1120
+
1121
+ if has_explicit_config:
1122
+ enabled_toolsets = {
1123
+ ts for ts in toolset_names
1124
+ if ts in configurable_keys and _toolset_allowed_for_platform(ts, platform)
1125
+ }
1126
+ # Mixed config: composite toolset alongside configurables (e.g.
1127
+ # ``[hermes-cli, spotify]`` after enabling Spotify via ``hermes
1128
+ # tools``). Without expansion the composite name is silently dropped,
1129
+ # leaving sessions with only the configurable opt-ins and no native
1130
+ # tools. Mirror the else-branch's subset inference, but apply
1131
+ # _DEFAULT_OFF_TOOLSETS only to the implicit expansion — anything the
1132
+ # user explicitly listed (e.g. ``spotify``) must survive.
1133
+ composite_tools = set()
1134
+ for ts_name in toolset_names:
1135
+ if ts_name in configurable_keys or ts_name in plugin_ts_keys:
1136
+ continue
1137
+ if ts_name not in TOOLSETS:
1138
+ continue
1139
+ composite_tools.update(resolve_toolset(ts_name))
1140
+
1141
+ if composite_tools:
1142
+ expanded = set()
1143
+ for ts_key, _, _ in CONFIGURABLE_TOOLSETS:
1144
+ if not _toolset_allowed_for_platform(ts_key, platform):
1145
+ continue
1146
+ ts_tools = set(resolve_toolset(ts_key))
1147
+ if ts_tools and ts_tools.issubset(composite_tools):
1148
+ expanded.add(ts_key)
1149
+
1150
+ default_off = set(_DEFAULT_OFF_TOOLSETS)
1151
+ if platform in default_off and platform not in _TOOLSET_PLATFORM_RESTRICTIONS:
1152
+ default_off.remove(platform)
1153
+ if "homeassistant" in default_off and os.getenv("HASS_TOKEN"):
1154
+ default_off.remove("homeassistant")
1155
+ expanded -= default_off
1156
+
1157
+ enabled_toolsets |= expanded
1158
+ else:
1159
+ # No explicit config — fall back to resolving composite toolset names
1160
+ # (e.g. "hermes-cli") to individual tool names and reverse-mapping.
1161
+ all_tool_names = set()
1162
+ for ts_name in toolset_names:
1163
+ all_tool_names.update(resolve_toolset(ts_name))
1164
+
1165
+ enabled_toolsets = set()
1166
+ for ts_key, _, _ in CONFIGURABLE_TOOLSETS:
1167
+ if not _toolset_allowed_for_platform(ts_key, platform):
1168
+ continue
1169
+ ts_tools = set(resolve_toolset(ts_key))
1170
+ if ts_tools and ts_tools.issubset(all_tool_names):
1171
+ enabled_toolsets.add(ts_key)
1172
+
1173
+ default_off = set(_DEFAULT_OFF_TOOLSETS)
1174
+ # Legacy safety: if the platform's own name matches a default-off
1175
+ # toolset (e.g. `homeassistant` platform + `homeassistant` toolset),
1176
+ # keep that toolset enabled on first install. Skip this dodge for
1177
+ # platform-restricted toolsets — those are always opt-in even on
1178
+ # their own platform (e.g. `discord` + `discord` should stay OFF).
1179
+ if platform in default_off and platform not in _TOOLSET_PLATFORM_RESTRICTIONS:
1180
+ default_off.remove(platform)
1181
+ # Home Assistant is already runtime-gated by its check_fn (requires
1182
+ # HASS_TOKEN to register any tools). When a user has configured
1183
+ # HASS_TOKEN, they've explicitly opted in — don't also strip it via
1184
+ # _DEFAULT_OFF_TOOLSETS, which would silently drop HA from platforms
1185
+ # (e.g. cron) that run through _get_platform_tools without an
1186
+ # explicit saved toolset list. Without this, Norbert's HA cron jobs
1187
+ # regressed after #14798 made cron honor per-platform tool config.
1188
+ if "homeassistant" in default_off and os.getenv("HASS_TOKEN"):
1189
+ default_off.remove("homeassistant")
1190
+ enabled_toolsets -= default_off
1191
+
1192
+ # Recover non-configurable platform toolsets (e.g. discord, feishu_doc,
1193
+ # feishu_drive). These are part of the platform's default composite but
1194
+ # absent from CONFIGURABLE_TOOLSETS, so they can't appear in the TUI
1195
+ # checklist or in a user-saved config. Must run in BOTH branches —
1196
+ # otherwise saving via `hermes tools` (which flips has_explicit_config
1197
+ # to True) silently drops them.
1198
+ _plat_info = PLATFORMS.get(platform)
1199
+ _default_ts = _plat_info["default_toolset"] if _plat_info else f"hermes-{platform}"
1200
+ platform_tool_universe = set(resolve_toolset(_default_ts))
1201
+ configurable_tool_universe = set()
1202
+ for ck in configurable_keys:
1203
+ configurable_tool_universe.update(resolve_toolset(ck))
1204
+ claimed = set()
1205
+ for ts_key in enabled_toolsets:
1206
+ claimed.update(resolve_toolset(ts_key))
1207
+ skip = configurable_keys | plugin_ts_keys | platform_default_keys
1208
+ skip |= {k for k in TOOLSETS if k.startswith("hermes-")}
1209
+ skip |= set(_DEFAULT_OFF_TOOLSETS) - {platform}
1210
+ for ts_key, ts_def in TOOLSETS.items():
1211
+ if ts_key in skip:
1212
+ continue
1213
+ if ts_def.get("includes"):
1214
+ continue
1215
+ ts_tools = set(resolve_toolset(ts_key))
1216
+ if not ts_tools or not ts_tools.issubset(platform_tool_universe):
1217
+ continue
1218
+ if ts_tools.issubset(configurable_tool_universe):
1219
+ continue
1220
+ if not ts_tools.issubset(claimed):
1221
+ enabled_toolsets.add(ts_key)
1222
+ claimed.update(ts_tools)
1223
+
1224
+ # Plugin toolsets: enabled by default unless explicitly disabled, or
1225
+ # unless the toolset is in _DEFAULT_OFF_TOOLSETS (e.g. spotify —
1226
+ # shipped as a bundled plugin but user must opt in via `hermes tools`
1227
+ # so we don't ship 7 Spotify tool schemas to users who don't use it).
1228
+ # A plugin toolset is "known" for a platform once `hermes tools`
1229
+ # has been saved for that platform (tracked via known_plugin_toolsets).
1230
+ # Unknown plugins default to enabled; known-but-absent = disabled.
1231
+ if plugin_ts_keys:
1232
+ known_map = config.get("known_plugin_toolsets", {})
1233
+ known_for_platform = set(known_map.get(platform, []))
1234
+ for pts in plugin_ts_keys:
1235
+ if pts in toolset_names:
1236
+ # Explicitly listed in config — enabled
1237
+ enabled_toolsets.add(pts)
1238
+ elif pts in _DEFAULT_OFF_TOOLSETS:
1239
+ # Opt-in plugin toolset — stay off until user picks it
1240
+ continue
1241
+ elif pts not in known_for_platform:
1242
+ # New plugin not yet seen by hermes tools — default enabled
1243
+ enabled_toolsets.add(pts)
1244
+ # else: known but not in config = user disabled it
1245
+
1246
+ # Preserve any explicit non-configurable toolset entries (for example,
1247
+ # custom toolsets or MCP server names saved in platform_toolsets).
1248
+ explicit_passthrough = {
1249
+ ts
1250
+ for ts in toolset_names
1251
+ if ts not in configurable_keys
1252
+ and ts not in plugin_ts_keys
1253
+ and ts not in platform_default_keys
1254
+ }
1255
+
1256
+ # MCP servers are expected to be available on all platforms by default.
1257
+ # If the platform explicitly lists one or more MCP server names, treat that
1258
+ # as an allowlist. Otherwise include every globally enabled MCP server.
1259
+ # Special sentinel: "no_mcp" in the toolset list disables all MCP servers.
1260
+ mcp_servers = config.get("mcp_servers") or {}
1261
+ enabled_mcp_servers = {
1262
+ str(name)
1263
+ for name, server_cfg in mcp_servers.items()
1264
+ if isinstance(server_cfg, dict)
1265
+ and _parse_enabled_flag(server_cfg.get("enabled", True), default=True)
1266
+ }
1267
+ # Allow "no_mcp" sentinel to opt out of all MCP servers for this platform
1268
+ if "no_mcp" in toolset_names:
1269
+ explicit_mcp_servers = set()
1270
+ enabled_toolsets.update(explicit_passthrough - enabled_mcp_servers - {"no_mcp"})
1271
+ else:
1272
+ explicit_mcp_servers = explicit_passthrough & enabled_mcp_servers
1273
+ enabled_toolsets.update(explicit_passthrough - enabled_mcp_servers)
1274
+ if include_default_mcp_servers:
1275
+ if explicit_mcp_servers or "no_mcp" in toolset_names:
1276
+ enabled_toolsets.update(explicit_mcp_servers)
1277
+ else:
1278
+ enabled_toolsets.update(enabled_mcp_servers)
1279
+ else:
1280
+ enabled_toolsets.update(explicit_mcp_servers)
1281
+
1282
+ # Honor agent.disabled_toolsets from config.yaml — allows users to
1283
+ # globally suppress specific toolsets (e.g. "memory") across all
1284
+ # platforms without per-platform toolset configuration. This runs
1285
+ # last so it overrides everything above.
1286
+ agent_cfg = config.get("agent") or {}
1287
+ disabled_toolsets = agent_cfg.get("disabled_toolsets") or []
1288
+ if disabled_toolsets:
1289
+ disabled_set = {str(ts) for ts in disabled_toolsets}
1290
+ enabled_toolsets -= disabled_set
1291
+
1292
+ return enabled_toolsets
1293
+
1294
+
1295
+ def _save_platform_tools(config: dict, platform: str, enabled_toolset_keys: Set[str]):
1296
+ """Save the selected toolset keys for a platform to config.
1297
+
1298
+ Preserves any non-configurable toolset entries (like MCP server names)
1299
+ that were already in the config for this platform.
1300
+ """
1301
+ config.setdefault("platform_toolsets", {})
1302
+
1303
+ # Drop platform-scoped toolsets that don't apply here. Prevents the
1304
+ # "Configure all platforms" checklist (or a hand-edited config.yaml)
1305
+ # from turning on, say, the `discord` toolset for Telegram.
1306
+ enabled_toolset_keys = {
1307
+ ts for ts in enabled_toolset_keys
1308
+ if _toolset_allowed_for_platform(ts, platform)
1309
+ }
1310
+
1311
+ # Get the set of all configurable toolset keys (built-in + plugin)
1312
+ configurable_keys = {ts_key for ts_key, _, _ in CONFIGURABLE_TOOLSETS}
1313
+ plugin_keys = _get_plugin_toolset_keys()
1314
+ configurable_keys |= plugin_keys
1315
+
1316
+ # Also exclude platform default toolsets (hermes-cli, hermes-telegram, etc.)
1317
+ # These are "super" toolsets that resolve to ALL tools, so preserving them
1318
+ # would silently override the user's unchecked selections on the next read.
1319
+ platform_default_keys = {p["default_toolset"] for p in PLATFORMS.values()}
1320
+
1321
+ # Get existing toolsets for this platform
1322
+ existing_toolsets = cfg_get(config, "platform_toolsets", platform, default=[])
1323
+ if not isinstance(existing_toolsets, list):
1324
+ existing_toolsets = []
1325
+ existing_toolsets = [str(ts) for ts in existing_toolsets]
1326
+
1327
+ # Preserve any entries that are NOT configurable toolsets and NOT platform
1328
+ # defaults (i.e. only MCP server names should be preserved)
1329
+ preserved_entries = {
1330
+ entry for entry in existing_toolsets
1331
+ if entry not in configurable_keys and entry not in platform_default_keys
1332
+ }
1333
+ # Opening `hermes tools` is the user's opt-in to reconfigure tools, so treat
1334
+ # saving from the picker as consent to clear the "no_mcp" sentinel. The
1335
+ # picker has no checkbox for no_mcp, so without this users who once set it
1336
+ # by hand could never re-enable MCP servers through the UI.
1337
+ preserved_entries.discard("no_mcp")
1338
+
1339
+ # Merge preserved entries with new enabled toolsets
1340
+ config["platform_toolsets"][platform] = sorted(enabled_toolset_keys | preserved_entries)
1341
+
1342
+ # Track which plugin toolsets are "known" for this platform so we can
1343
+ # distinguish "new plugin, default enabled" from "user disabled it".
1344
+ if plugin_keys:
1345
+ config.setdefault("known_plugin_toolsets", {})
1346
+ config["known_plugin_toolsets"][platform] = sorted(plugin_keys)
1347
+
1348
+ save_config(config)
1349
+
1350
+
1351
+ def _toolset_has_keys(ts_key: str, config: dict = None) -> bool:
1352
+ """Check if a toolset's required API keys are configured."""
1353
+ if config is None:
1354
+ config = load_config()
1355
+
1356
+ if ts_key == "vision":
1357
+ try:
1358
+ from agent.auxiliary_client import resolve_vision_provider_client
1359
+
1360
+ _provider, client, _model = resolve_vision_provider_client()
1361
+ return client is not None
1362
+ except Exception:
1363
+ return False
1364
+
1365
+ if ts_key in {"web", "image_gen", "tts", "browser"}:
1366
+ features = get_nous_subscription_features(config)
1367
+ feature = features.features.get(ts_key)
1368
+ if feature and (feature.available or feature.managed_by_nous):
1369
+ return True
1370
+
1371
+ # Check TOOL_CATEGORIES first (provider-aware)
1372
+ cat = TOOL_CATEGORIES.get(ts_key)
1373
+ if cat:
1374
+ for provider in _visible_providers(cat, config):
1375
+ env_vars = provider.get("env_vars", [])
1376
+ if not env_vars:
1377
+ return True # No-key provider (e.g. Local Browser, Edge TTS)
1378
+ if all(get_env_value(e["key"]) for e in env_vars):
1379
+ return True
1380
+ return False
1381
+
1382
+ # Fallback to simple requirements
1383
+ requirements = TOOLSET_ENV_REQUIREMENTS.get(ts_key, [])
1384
+ if not requirements:
1385
+ return True
1386
+ return all(get_env_value(var) for var, _ in requirements)
1387
+
1388
+
1389
+ # ─── Menu Helpers ─────────────────────────────────────────────────────────────
1390
+
1391
+ def _prompt_choice(question: str, choices: list, default: int = 0) -> int:
1392
+ """Single-select menu (arrow keys). Delegates to curses_radiolist."""
1393
+ from hermes_cli.curses_ui import curses_radiolist
1394
+ return curses_radiolist(question, choices, selected=default, cancel_returns=default)
1395
+
1396
+
1397
+ # ─── Token Estimation ────────────────────────────────────────────────────────
1398
+
1399
+ # Module-level cache so discovery + tokenization runs at most once per process.
1400
+ _tool_token_cache: Optional[Dict[str, int]] = None
1401
+
1402
+
1403
+ def _estimate_tool_tokens() -> Dict[str, int]:
1404
+ """Return estimated token counts per individual tool name.
1405
+
1406
+ Uses tiktoken (cl100k_base) to count tokens in the JSON-serialised
1407
+ OpenAI-format tool schema. Triggers tool discovery on first call,
1408
+ then caches the result for the rest of the process.
1409
+
1410
+ Returns an empty dict when tiktoken or the registry is unavailable.
1411
+ """
1412
+ global _tool_token_cache
1413
+ if _tool_token_cache is not None:
1414
+ return _tool_token_cache
1415
+
1416
+ try:
1417
+ import tiktoken
1418
+ enc = tiktoken.get_encoding("cl100k_base")
1419
+ except Exception:
1420
+ logger.debug("tiktoken unavailable; skipping tool token estimation")
1421
+ _tool_token_cache = {}
1422
+ return _tool_token_cache
1423
+
1424
+ try:
1425
+ # Trigger full tool discovery (imports all tool modules).
1426
+ import model_tools # noqa: F401
1427
+ from tools.registry import registry
1428
+ except Exception:
1429
+ logger.debug("Tool registry unavailable; skipping token estimation")
1430
+ _tool_token_cache = {}
1431
+ return _tool_token_cache
1432
+
1433
+ counts: Dict[str, int] = {}
1434
+ for name in registry.get_all_tool_names():
1435
+ schema = registry.get_schema(name)
1436
+ if schema:
1437
+ # Mirror what gets sent to the API:
1438
+ # {"type": "function", "function": <schema>}
1439
+ text = _json.dumps({"type": "function", "function": schema})
1440
+ counts[name] = len(enc.encode(text))
1441
+ _tool_token_cache = counts
1442
+ return _tool_token_cache
1443
+
1444
+
1445
+ def _prompt_toolset_checklist(platform_label: str, enabled: Set[str], platform: str = "cli") -> Set[str]:
1446
+ """Multi-select checklist of toolsets. Returns set of selected toolset keys."""
1447
+ from hermes_cli.curses_ui import curses_checklist
1448
+ from toolsets import resolve_toolset
1449
+
1450
+ # Pre-compute per-tool token counts (cached after first call).
1451
+ tool_tokens = _estimate_tool_tokens()
1452
+
1453
+ effective_all = _get_effective_configurable_toolsets()
1454
+ # Drop platform-scoped toolsets that don't apply to this platform.
1455
+ effective = [
1456
+ (k, l, d) for (k, l, d) in effective_all
1457
+ if _toolset_allowed_for_platform(k, platform)
1458
+ ]
1459
+
1460
+ labels = []
1461
+ for ts_key, ts_label, ts_desc in effective:
1462
+ suffix = ""
1463
+ if not _toolset_has_keys(ts_key) and (TOOL_CATEGORIES.get(ts_key) or TOOLSET_ENV_REQUIREMENTS.get(ts_key)):
1464
+ suffix = " [no API key]"
1465
+ labels.append(f"{ts_label} ({ts_desc}){suffix}")
1466
+
1467
+ pre_selected = {
1468
+ i for i, (ts_key, _, _) in enumerate(effective)
1469
+ if ts_key in enabled
1470
+ }
1471
+
1472
+ # Build a live status function that shows deduplicated total token cost.
1473
+ status_fn = None
1474
+ if tool_tokens:
1475
+ ts_keys = [ts_key for ts_key, _, _ in effective]
1476
+
1477
+ def status_fn(chosen: set) -> str:
1478
+ # Collect unique tool names across all selected toolsets
1479
+ all_tools: set = set()
1480
+ for idx in chosen:
1481
+ all_tools.update(resolve_toolset(ts_keys[idx]))
1482
+ total = sum(tool_tokens.get(name, 0) for name in all_tools)
1483
+ if total >= 1000:
1484
+ return f"Est. tool context: ~{total / 1000:.1f}k tokens"
1485
+ return f"Est. tool context: ~{total} tokens"
1486
+
1487
+ chosen = curses_checklist(
1488
+ f"Tools for {platform_label}",
1489
+ labels,
1490
+ pre_selected,
1491
+ cancel_returns=pre_selected,
1492
+ status_fn=status_fn,
1493
+ )
1494
+ return {effective[i][0] for i in chosen}
1495
+
1496
+
1497
+ # ─── Provider-Aware Configuration ────────────────────────────────────────────
1498
+
1499
+ def _configure_toolset(ts_key: str, config: dict):
1500
+ """Configure a toolset - provider selection + API keys.
1501
+
1502
+ Uses TOOL_CATEGORIES for provider-aware config, falls back to simple
1503
+ env var prompts for toolsets not in TOOL_CATEGORIES.
1504
+ """
1505
+ cat = TOOL_CATEGORIES.get(ts_key)
1506
+
1507
+ if cat:
1508
+ _configure_tool_category(ts_key, cat, config)
1509
+ else:
1510
+ # Simple fallback for vision, moa, etc.
1511
+ _configure_simple_requirements(ts_key)
1512
+
1513
+
1514
+ def _plugin_image_gen_providers() -> list[dict]:
1515
+ """Build picker-row dicts from plugin-registered image gen providers.
1516
+
1517
+ Each returned dict looks like a regular ``TOOL_CATEGORIES`` provider
1518
+ row but carries an ``image_gen_plugin_name`` marker so downstream
1519
+ code (config writing, model picker) knows to route through the
1520
+ plugin registry instead of the in-tree FAL backend.
1521
+
1522
+ FAL is skipped — it's already exposed by the hardcoded
1523
+ ``TOOL_CATEGORIES["image_gen"]`` entries. When FAL gets ported to
1524
+ a plugin in a follow-up PR, the hardcoded entries go away and this
1525
+ function surfaces it alongside OpenAI automatically.
1526
+ """
1527
+ try:
1528
+ from agent.image_gen_registry import list_providers
1529
+ from hermes_cli.plugins import _ensure_plugins_discovered
1530
+
1531
+ _ensure_plugins_discovered()
1532
+ providers = list_providers()
1533
+ except Exception:
1534
+ return []
1535
+
1536
+ rows: list[dict] = []
1537
+ for provider in providers:
1538
+ if getattr(provider, "name", None) == "fal":
1539
+ # FAL has its own hardcoded rows today.
1540
+ continue
1541
+ try:
1542
+ schema = provider.get_setup_schema()
1543
+ except Exception:
1544
+ continue
1545
+ if not isinstance(schema, dict):
1546
+ continue
1547
+ row = {
1548
+ "name": schema.get("name", provider.display_name),
1549
+ "badge": schema.get("badge", ""),
1550
+ "tag": schema.get("tag", ""),
1551
+ "env_vars": schema.get("env_vars", []),
1552
+ "image_gen_plugin_name": provider.name,
1553
+ }
1554
+ if schema.get("post_setup"):
1555
+ row["post_setup"] = schema["post_setup"]
1556
+ rows.append(row)
1557
+ return rows
1558
+
1559
+
1560
+ def _plugin_video_gen_providers() -> list[dict]:
1561
+ """Build picker-row dicts from plugin-registered video gen providers.
1562
+
1563
+ Mirrors ``_plugin_image_gen_providers`` exactly — every video backend
1564
+ is a plugin, so this function is the *only* source of provider rows
1565
+ for the Video Generation category. The hardcoded ``TOOL_CATEGORIES``
1566
+ entry for ``video_gen`` keeps an empty providers list.
1567
+ """
1568
+ try:
1569
+ from agent.video_gen_registry import list_providers
1570
+ from hermes_cli.plugins import _ensure_plugins_discovered
1571
+
1572
+ _ensure_plugins_discovered()
1573
+ providers = list_providers()
1574
+ except Exception:
1575
+ return []
1576
+
1577
+ rows: list[dict] = []
1578
+ for provider in providers:
1579
+ try:
1580
+ schema = provider.get_setup_schema()
1581
+ except Exception:
1582
+ continue
1583
+ if not isinstance(schema, dict):
1584
+ continue
1585
+ row = {
1586
+ "name": schema.get("name", provider.display_name),
1587
+ "badge": schema.get("badge", ""),
1588
+ "tag": schema.get("tag", ""),
1589
+ "env_vars": schema.get("env_vars", []),
1590
+ "video_gen_plugin_name": provider.name,
1591
+ }
1592
+ if schema.get("post_setup"):
1593
+ row["post_setup"] = schema["post_setup"]
1594
+ rows.append(row)
1595
+ return rows
1596
+
1597
+
1598
+ # Mirror of _plugin_image_gen_providers for web search backends. Surfaces
1599
+ # every plugin-registered web provider so it appears in the
1600
+ # "Web Search & Extract" picker. All seven providers (brave-free, ddgs,
1601
+ # searxng, exa, parallel, tavily, firecrawl) live as plugins after
1602
+ # PR #25182 — this helper is the sole source of truth for the category's
1603
+ # provider rows. The hardcoded entries that used to drive the category
1604
+ # were deleted in the same PR; only the two non-provider UX rows
1605
+ # ("Nous Subscription" managed-gateway entry, "Firecrawl Self-Hosted")
1606
+ # remain in TOOL_CATEGORIES because they describe alternative *setup
1607
+ # flows* for the firecrawl backend rather than distinct providers.
1608
+ def _plugin_web_search_providers() -> list[dict]:
1609
+ """Build picker-row dicts from plugin-registered web search providers.
1610
+
1611
+ Each returned dict is a regular ``TOOL_CATEGORIES`` provider row. It
1612
+ populates both ``web_backend`` (legacy field consumed by setup +
1613
+ selection helpers) and ``web_search_plugin_name`` (informational
1614
+ marker) so the picker behaves identically whether a provider is
1615
+ hardcoded or plugin-registered.
1616
+
1617
+ After PR #25182, all seven web providers (brave-free, ddgs, searxng,
1618
+ exa, parallel, tavily, firecrawl) are plugins; this helper is the sole
1619
+ source of provider rows for the Web Search & Extract category.
1620
+ """
1621
+ try:
1622
+ from agent.web_search_registry import list_providers as _list_web_providers
1623
+ from hermes_cli.plugins import _ensure_plugins_discovered
1624
+
1625
+ _ensure_plugins_discovered()
1626
+ providers = _list_web_providers()
1627
+ except Exception:
1628
+ return []
1629
+
1630
+ rows: list[dict] = []
1631
+ for provider in providers:
1632
+ name = getattr(provider, "name", None)
1633
+ if not name:
1634
+ continue
1635
+ try:
1636
+ schema = provider.get_setup_schema()
1637
+ except Exception:
1638
+ continue
1639
+ if not isinstance(schema, dict):
1640
+ continue
1641
+ row = {
1642
+ "name": schema.get("name", provider.display_name),
1643
+ "badge": schema.get("badge", ""),
1644
+ "tag": schema.get("tag", ""),
1645
+ "env_vars": schema.get("env_vars", []),
1646
+ "web_backend": name,
1647
+ "web_search_plugin_name": name,
1648
+ }
1649
+ # Optional pass-through fields the schema can opt into.
1650
+ if schema.get("post_setup"):
1651
+ row["post_setup"] = schema["post_setup"]
1652
+ rows.append(row)
1653
+ return rows
1654
+
1655
+
1656
+ def _visible_providers(cat: dict, config: dict) -> list[dict]:
1657
+ """Return provider entries visible for the current auth/config state."""
1658
+ features = get_nous_subscription_features(config)
1659
+ visible = []
1660
+ for provider in cat.get("providers", []):
1661
+ if provider.get("managed_nous_feature") and not managed_nous_tools_enabled():
1662
+ continue
1663
+ if provider.get("requires_nous_auth") and not features.nous_auth_present:
1664
+ continue
1665
+ visible.append(provider)
1666
+
1667
+ # Inject plugin-registered image_gen backends (OpenAI today, more
1668
+ # later) so the picker lists them alongside FAL / Nous Subscription.
1669
+ if cat.get("name") == "Image Generation":
1670
+ visible.extend(_plugin_image_gen_providers())
1671
+
1672
+ # Inject plugin-registered video_gen backends. Unlike image_gen,
1673
+ # video_gen has NO hardcoded providers — every backend is a plugin.
1674
+ if cat.get("name") == "Video Generation":
1675
+ visible.extend(_plugin_video_gen_providers())
1676
+
1677
+ # Inject plugin-registered web search backends. After PR #25182, this
1678
+ # is the SOLE source of provider rows for the Web Search & Extract
1679
+ # category — the per-provider hardcoded entries were deleted. The two
1680
+ # remaining hardcoded rows ("Nous Subscription", "Firecrawl
1681
+ # Self-Hosted") are non-provider UX setup-flow rows for firecrawl.
1682
+ if cat.get("name") == "Web Search & Extract":
1683
+ visible.extend(_plugin_web_search_providers())
1684
+
1685
+ return visible
1686
+
1687
+
1688
+ _POST_SETUP_INSTALLED: dict = {
1689
+ # post_setup_key -> predicate(): True when the install side-effect
1690
+ # is already satisfied. Used by `_toolset_needs_configuration_prompt`
1691
+ # to force the provider-setup flow when a no-key provider still needs
1692
+ # a binary/dependency install (otherwise an already-configured user
1693
+ # who toggles the toolset on via `hermes tools` gets a silent no-op
1694
+ # because the gate sees "no env vars to ask about" and skips the
1695
+ # provider-setup flow that would have run the post_setup hook).
1696
+ #
1697
+ # Only entries here are gated; other post_setup hooks (kittentts,
1698
+ # piper, agent_browser, etc.) keep their existing behaviour. Add an
1699
+ # entry when (a) the post_setup is the ONLY install side-effect for
1700
+ # a no-key provider, and (b) an installed-state check is cheap and
1701
+ # doesn't trigger a heavy import.
1702
+ "cua_driver": lambda: bool(shutil.which("cua-driver")),
1703
+ }
1704
+
1705
+
1706
+ def _post_setup_already_installed(post_setup_key: str) -> bool:
1707
+ """Return True when the post_setup install side-effect is satisfied."""
1708
+ predicate = _POST_SETUP_INSTALLED.get(post_setup_key)
1709
+ if predicate is None:
1710
+ # No install-state check registered → assume satisfied (don't
1711
+ # change behaviour for hooks we haven't explicitly opted in).
1712
+ return True
1713
+ try:
1714
+ return bool(predicate())
1715
+ except Exception:
1716
+ return True
1717
+
1718
+
1719
+ def _toolset_needs_configuration_prompt(ts_key: str, config: dict) -> bool:
1720
+ """Return True when enabling this toolset should open provider setup."""
1721
+ cat = TOOL_CATEGORIES.get(ts_key)
1722
+ if not cat:
1723
+ return not _toolset_has_keys(ts_key, config)
1724
+
1725
+ # If any visible provider has a registered post_setup install-state
1726
+ # check that hasn't been satisfied (e.g. cua-driver binary not on
1727
+ # PATH yet), force the configuration flow so `_configure_provider`
1728
+ # invokes `_run_post_setup` and the install actually runs.
1729
+ for provider in _visible_providers(cat, config):
1730
+ post_setup = provider.get("post_setup")
1731
+ if post_setup and not _post_setup_already_installed(post_setup):
1732
+ return True
1733
+
1734
+ if ts_key == "tts":
1735
+ tts_cfg = config.get("tts", {})
1736
+ return not isinstance(tts_cfg, dict) or "provider" not in tts_cfg
1737
+ if ts_key == "web":
1738
+ web_cfg = config.get("web", {})
1739
+ return not isinstance(web_cfg, dict) or "backend" not in web_cfg
1740
+ if ts_key == "browser":
1741
+ browser_cfg = config.get("browser", {})
1742
+ return not isinstance(browser_cfg, dict) or "cloud_provider" not in browser_cfg
1743
+ if ts_key == "image_gen":
1744
+ # Satisfied when the in-tree FAL backend is configured OR any
1745
+ # plugin-registered image gen provider is available.
1746
+ if fal_key_is_configured():
1747
+ return False
1748
+ try:
1749
+ from agent.image_gen_registry import list_providers
1750
+ from hermes_cli.plugins import _ensure_plugins_discovered
1751
+
1752
+ _ensure_plugins_discovered()
1753
+ for provider in list_providers():
1754
+ try:
1755
+ if provider.is_available():
1756
+ return False
1757
+ except Exception:
1758
+ continue
1759
+ except Exception:
1760
+ pass
1761
+ return True
1762
+ if ts_key == "video_gen":
1763
+ # Satisfied when any plugin-registered video gen provider reports
1764
+ # available — no in-tree fallback (every backend is a plugin).
1765
+ try:
1766
+ from agent.video_gen_registry import list_providers
1767
+ from hermes_cli.plugins import _ensure_plugins_discovered
1768
+
1769
+ _ensure_plugins_discovered()
1770
+ for provider in list_providers():
1771
+ try:
1772
+ if provider.is_available():
1773
+ return False
1774
+ except Exception:
1775
+ continue
1776
+ except Exception:
1777
+ pass
1778
+ return True
1779
+
1780
+ return not _toolset_has_keys(ts_key, config)
1781
+
1782
+
1783
+ def _configure_tool_category(ts_key: str, cat: dict, config: dict):
1784
+ """Configure a tool category with provider selection."""
1785
+ icon = cat.get("icon", "")
1786
+ name = cat["name"]
1787
+ providers = _visible_providers(cat, config)
1788
+
1789
+ # Check Python version requirement
1790
+ if cat.get("requires_python"):
1791
+ req = cat["requires_python"]
1792
+ if sys.version_info < req:
1793
+ print()
1794
+ _print_error(f" {name} requires Python {req[0]}.{req[1]}+ (current: {sys.version_info.major}.{sys.version_info.minor})")
1795
+ _print_info(" Upgrade Python and reinstall to enable this tool.")
1796
+ return
1797
+
1798
+ if len(providers) == 1:
1799
+ # Single provider - configure directly
1800
+ provider = providers[0]
1801
+ print()
1802
+ print(color(f" --- {icon} {name} ({provider['name']}) ---", Colors.CYAN))
1803
+ if provider.get("tag"):
1804
+ _print_info(f" {provider['tag']}")
1805
+ # For single-provider tools, show a note if available
1806
+ if cat.get("setup_note"):
1807
+ _print_info(f" {cat['setup_note']}")
1808
+ _configure_provider(provider, config)
1809
+ else:
1810
+ # Multiple providers - let user choose
1811
+ print()
1812
+ # Use custom title if provided (e.g. "Select Search Provider")
1813
+ title = cat.get("setup_title", "Choose a provider")
1814
+ print(color(f" --- {icon} {name} - {title} ---", Colors.CYAN))
1815
+ if cat.get("setup_note"):
1816
+ _print_info(f" {cat['setup_note']}")
1817
+ print()
1818
+
1819
+ # Plain text labels only (no ANSI codes in menu items)
1820
+ provider_choices = []
1821
+ for p in providers:
1822
+ badge = f" [{p['badge']}]" if p.get("badge") else ""
1823
+ tag = f" — {p['tag']}" if p.get("tag") else ""
1824
+ configured = ""
1825
+ env_vars = p.get("env_vars", [])
1826
+ if not env_vars or all(get_env_value(v["key"]) for v in env_vars):
1827
+ if _is_provider_active(p, config):
1828
+ configured = " [active]"
1829
+ elif not env_vars:
1830
+ configured = ""
1831
+ else:
1832
+ configured = " [configured]"
1833
+ provider_choices.append(f"{p['name']}{badge}{tag}{configured}")
1834
+
1835
+ # Add skip option
1836
+ provider_choices.append("Skip — keep defaults / configure later")
1837
+
1838
+ # Detect current provider as default
1839
+ default_idx = _detect_active_provider_index(providers, config)
1840
+
1841
+ provider_idx = _prompt_choice(f" {title}:", provider_choices, default_idx)
1842
+
1843
+ # Skip selected
1844
+ if provider_idx >= len(providers):
1845
+ _print_info(f" Skipped {name}")
1846
+ return
1847
+
1848
+ _configure_provider(providers[provider_idx], config)
1849
+
1850
+
1851
+ def _is_provider_active(provider: dict, config: dict) -> bool:
1852
+ """Check if a provider entry matches the currently active config."""
1853
+ plugin_name = provider.get("image_gen_plugin_name")
1854
+ if plugin_name:
1855
+ image_cfg = config.get("image_gen", {})
1856
+ return isinstance(image_cfg, dict) and image_cfg.get("provider") == plugin_name
1857
+
1858
+ video_plugin_name = provider.get("video_gen_plugin_name")
1859
+ if video_plugin_name:
1860
+ video_cfg = config.get("video_gen", {})
1861
+ return isinstance(video_cfg, dict) and video_cfg.get("provider") == video_plugin_name
1862
+
1863
+ managed_feature = provider.get("managed_nous_feature")
1864
+ if managed_feature:
1865
+ features = get_nous_subscription_features(config)
1866
+ feature = features.features.get(managed_feature)
1867
+ if feature is None:
1868
+ return False
1869
+ if managed_feature == "image_gen":
1870
+ image_cfg = config.get("image_gen", {})
1871
+ if isinstance(image_cfg, dict):
1872
+ configured_provider = image_cfg.get("provider")
1873
+ if configured_provider not in {None, "", "fal"}:
1874
+ return False
1875
+ if image_cfg.get("use_gateway") is not None and not is_truthy_value(image_cfg.get("use_gateway"), default=False):
1876
+ return False
1877
+ return feature.managed_by_nous
1878
+ if provider.get("tts_provider"):
1879
+ return (
1880
+ feature.managed_by_nous
1881
+ and cfg_get(config, "tts", "provider") == provider["tts_provider"]
1882
+ )
1883
+ if "browser_provider" in provider:
1884
+ current = cfg_get(config, "browser", "cloud_provider")
1885
+ return feature.managed_by_nous and provider["browser_provider"] == current
1886
+ if provider.get("web_backend"):
1887
+ current = cfg_get(config, "web", "backend")
1888
+ return feature.managed_by_nous and current == provider["web_backend"]
1889
+ return feature.managed_by_nous
1890
+
1891
+ if provider.get("tts_provider"):
1892
+ return cfg_get(config, "tts", "provider") == provider["tts_provider"]
1893
+ if "browser_provider" in provider:
1894
+ current = cfg_get(config, "browser", "cloud_provider")
1895
+ return provider["browser_provider"] == current
1896
+ if provider.get("web_backend"):
1897
+ current = cfg_get(config, "web", "backend")
1898
+ return current == provider["web_backend"]
1899
+ if provider.get("imagegen_backend"):
1900
+ image_cfg = config.get("image_gen", {})
1901
+ if not isinstance(image_cfg, dict):
1902
+ return False
1903
+ configured_provider = image_cfg.get("provider")
1904
+ return (
1905
+ provider["imagegen_backend"] == "fal"
1906
+ and configured_provider in {None, "", "fal"}
1907
+ and not is_truthy_value(image_cfg.get("use_gateway"), default=False)
1908
+ )
1909
+ return False
1910
+
1911
+
1912
+ def _detect_active_provider_index(providers: list, config: dict) -> int:
1913
+ """Return the index of the currently active provider, or 0."""
1914
+ for i, p in enumerate(providers):
1915
+ if _is_provider_active(p, config):
1916
+ return i
1917
+ # Fallback: env vars present → likely configured
1918
+ env_vars = p.get("env_vars", [])
1919
+ if env_vars and all(get_env_value(v["key"]) for v in env_vars):
1920
+ return i
1921
+ return 0
1922
+
1923
+
1924
+ # ─── Image Generation Model Pickers ───────────────────────────────────────────
1925
+ #
1926
+ # IMAGEGEN_BACKENDS is a per-backend catalog. Each entry exposes:
1927
+ # - config_key: top-level config.yaml key for this backend's settings
1928
+ # - model_catalog_fn: returns an OrderedDict-like {model_id: metadata}
1929
+ # - default_model: fallback when nothing is configured
1930
+ #
1931
+ # This prepares for future imagegen backends (Replicate, Stability, etc.):
1932
+ # each new backend registers its own entry; the FAL provider entry in
1933
+ # TOOL_CATEGORIES tags itself with `imagegen_backend: "fal"` to select the
1934
+ # right catalog at picker time.
1935
+
1936
+
1937
+ def _fal_model_catalog():
1938
+ """Lazy-load the FAL model catalog from the tool module."""
1939
+ from tools.image_generation_tool import FAL_MODELS, DEFAULT_MODEL
1940
+ return FAL_MODELS, DEFAULT_MODEL
1941
+
1942
+
1943
+ IMAGEGEN_BACKENDS = {
1944
+ "fal": {
1945
+ "display": "FAL.ai",
1946
+ "config_key": "image_gen",
1947
+ "catalog_fn": _fal_model_catalog,
1948
+ },
1949
+ }
1950
+
1951
+
1952
+ def _format_imagegen_model_row(model_id: str, meta: dict, widths: dict) -> str:
1953
+ """Format a single picker row with column-aligned speed / strengths / price."""
1954
+ return (
1955
+ f"{model_id:<{widths['model']}} "
1956
+ f"{meta.get('speed', ''):<{widths['speed']}} "
1957
+ f"{meta.get('strengths', ''):<{widths['strengths']}} "
1958
+ f"{meta.get('price', '')}"
1959
+ )
1960
+
1961
+
1962
+ def _configure_imagegen_model(backend_name: str, config: dict) -> None:
1963
+ """Prompt the user to pick a model for the given imagegen backend.
1964
+
1965
+ Writes selection to ``config[backend_config_key]["model"]``. Safe to
1966
+ call even when stdin is not a TTY — curses_radiolist falls back to
1967
+ keeping the current selection.
1968
+ """
1969
+ backend = IMAGEGEN_BACKENDS.get(backend_name)
1970
+ if not backend:
1971
+ return
1972
+
1973
+ catalog, default_model = backend["catalog_fn"]()
1974
+ if not catalog:
1975
+ return
1976
+
1977
+ cfg_key = backend["config_key"]
1978
+ cur_cfg = config.setdefault(cfg_key, {})
1979
+ if not isinstance(cur_cfg, dict):
1980
+ cur_cfg = {}
1981
+ config[cfg_key] = cur_cfg
1982
+ current_model = cur_cfg.get("model") or default_model
1983
+ if current_model not in catalog:
1984
+ current_model = default_model
1985
+
1986
+ model_ids = list(catalog.keys())
1987
+ # Put current model at the top so the cursor lands on it by default.
1988
+ ordered = [current_model] + [m for m in model_ids if m != current_model]
1989
+
1990
+ # Column widths
1991
+ widths = {
1992
+ "model": max(len(m) for m in model_ids),
1993
+ "speed": max((len(catalog[m].get("speed", "")) for m in model_ids), default=6),
1994
+ "strengths": max((len(catalog[m].get("strengths", "")) for m in model_ids), default=0),
1995
+ }
1996
+
1997
+ print()
1998
+ header = (
1999
+ f" {'Model':<{widths['model']}} "
2000
+ f"{'Speed':<{widths['speed']}} "
2001
+ f"{'Strengths':<{widths['strengths']}} "
2002
+ f"Price"
2003
+ )
2004
+ print(color(header, Colors.CYAN))
2005
+
2006
+ rows = []
2007
+ for mid in ordered:
2008
+ row = _format_imagegen_model_row(mid, catalog[mid], widths)
2009
+ if mid == current_model:
2010
+ row += " ← currently in use"
2011
+ rows.append(row)
2012
+
2013
+ idx = _prompt_choice(
2014
+ f" Choose {backend['display']} model:",
2015
+ rows,
2016
+ default=0,
2017
+ )
2018
+
2019
+ chosen = ordered[idx]
2020
+ cur_cfg["model"] = chosen
2021
+ _print_success(f" Model set to: {chosen}")
2022
+
2023
+
2024
+ def _plugin_image_gen_catalog(plugin_name: str):
2025
+ """Return ``(catalog_dict, default_model_id)`` for a plugin provider.
2026
+
2027
+ ``catalog_dict`` is shaped like the legacy ``FAL_MODELS`` table —
2028
+ ``{model_id: {"display", "speed", "strengths", "price", ...}}`` —
2029
+ so the existing picker code paths work without change. Returns
2030
+ ``({}, None)`` if the provider isn't registered or has no models.
2031
+ """
2032
+ try:
2033
+ from agent.image_gen_registry import get_provider
2034
+ from hermes_cli.plugins import _ensure_plugins_discovered
2035
+
2036
+ _ensure_plugins_discovered()
2037
+ provider = get_provider(plugin_name)
2038
+ except Exception:
2039
+ return {}, None
2040
+ if provider is None:
2041
+ return {}, None
2042
+ try:
2043
+ models = provider.list_models() or []
2044
+ default = provider.default_model()
2045
+ except Exception:
2046
+ return {}, None
2047
+ catalog = {m["id"]: m for m in models if isinstance(m, dict) and "id" in m}
2048
+ return catalog, default
2049
+
2050
+
2051
+ def _configure_imagegen_model_for_plugin(plugin_name: str, config: dict) -> None:
2052
+ """Prompt the user to pick a model for a plugin-registered backend.
2053
+
2054
+ Writes selection to ``image_gen.model``. Mirrors
2055
+ :func:`_configure_imagegen_model` but sources its catalog from the
2056
+ plugin registry instead of :data:`IMAGEGEN_BACKENDS`.
2057
+ """
2058
+ catalog, default_model = _plugin_image_gen_catalog(plugin_name)
2059
+ if not catalog:
2060
+ return
2061
+
2062
+ cur_cfg = config.setdefault("image_gen", {})
2063
+ if not isinstance(cur_cfg, dict):
2064
+ cur_cfg = {}
2065
+ config["image_gen"] = cur_cfg
2066
+ current_model = cur_cfg.get("model") or default_model
2067
+ if current_model not in catalog:
2068
+ current_model = default_model
2069
+
2070
+ model_ids = list(catalog.keys())
2071
+ ordered = [current_model] + [m for m in model_ids if m != current_model]
2072
+
2073
+ widths = {
2074
+ "model": max(len(m) for m in model_ids),
2075
+ "speed": max((len(catalog[m].get("speed", "")) for m in model_ids), default=6),
2076
+ "strengths": max((len(catalog[m].get("strengths", "")) for m in model_ids), default=0),
2077
+ }
2078
+
2079
+ print()
2080
+ header = (
2081
+ f" {'Model':<{widths['model']}} "
2082
+ f"{'Speed':<{widths['speed']}} "
2083
+ f"{'Strengths':<{widths['strengths']}} "
2084
+ f"Price"
2085
+ )
2086
+ print(color(header, Colors.CYAN))
2087
+
2088
+ rows = []
2089
+ for mid in ordered:
2090
+ row = _format_imagegen_model_row(mid, catalog[mid], widths)
2091
+ if mid == current_model:
2092
+ row += " ← currently in use"
2093
+ rows.append(row)
2094
+
2095
+ idx = _prompt_choice(
2096
+ f" Choose {plugin_name} model:",
2097
+ rows,
2098
+ default=0,
2099
+ )
2100
+
2101
+ chosen = ordered[idx]
2102
+ cur_cfg["model"] = chosen
2103
+ _print_success(f" Model set to: {chosen}")
2104
+
2105
+
2106
+ def _select_plugin_image_gen_provider(plugin_name: str, config: dict) -> None:
2107
+ """Persist a plugin-backed image generation provider selection."""
2108
+ img_cfg = config.setdefault("image_gen", {})
2109
+ if not isinstance(img_cfg, dict):
2110
+ img_cfg = {}
2111
+ config["image_gen"] = img_cfg
2112
+ img_cfg["provider"] = plugin_name
2113
+ img_cfg["use_gateway"] = False
2114
+ _print_success(f" image_gen.provider set to: {plugin_name}")
2115
+ _configure_imagegen_model_for_plugin(plugin_name, config)
2116
+
2117
+
2118
+ # ─── Video Generation Model Pickers ───────────────────────────────────────────
2119
+
2120
+
2121
+ def _plugin_video_gen_catalog(plugin_name: str):
2122
+ """Return ``(catalog_dict, default_model_id)`` for a video gen plugin.
2123
+
2124
+ Mirrors :func:`_plugin_image_gen_catalog`. Returns ``({}, None)`` when
2125
+ the plugin isn't registered or has no models.
2126
+ """
2127
+ try:
2128
+ from agent.video_gen_registry import get_provider
2129
+ from hermes_cli.plugins import _ensure_plugins_discovered
2130
+
2131
+ _ensure_plugins_discovered()
2132
+ provider = get_provider(plugin_name)
2133
+ except Exception:
2134
+ return {}, None
2135
+ if provider is None:
2136
+ return {}, None
2137
+ try:
2138
+ models = provider.list_models() or []
2139
+ default = provider.default_model()
2140
+ except Exception:
2141
+ return {}, None
2142
+ catalog = {m["id"]: m for m in models if isinstance(m, dict) and "id" in m}
2143
+ return catalog, default
2144
+
2145
+
2146
+ def _configure_videogen_model_for_plugin(plugin_name: str, config: dict) -> None:
2147
+ """Prompt for a video gen model from a plugin's catalog.
2148
+
2149
+ Mirrors :func:`_configure_imagegen_model_for_plugin`. Writes the
2150
+ selection to ``video_gen.model``.
2151
+ """
2152
+ catalog, default_model = _plugin_video_gen_catalog(plugin_name)
2153
+ if not catalog:
2154
+ return
2155
+
2156
+ cur_cfg = config.setdefault("video_gen", {})
2157
+ if not isinstance(cur_cfg, dict):
2158
+ cur_cfg = {}
2159
+ config["video_gen"] = cur_cfg
2160
+ current_model = cur_cfg.get("model") or default_model
2161
+ if current_model not in catalog:
2162
+ current_model = default_model
2163
+
2164
+ model_ids = list(catalog.keys())
2165
+ ordered = [current_model] + [m for m in model_ids if m != current_model]
2166
+
2167
+ widths = {
2168
+ "model": max(len(m) for m in model_ids),
2169
+ "speed": max((len(catalog[m].get("speed", "")) for m in model_ids), default=6),
2170
+ "strengths": max((len(catalog[m].get("strengths", "")) for m in model_ids), default=0),
2171
+ }
2172
+
2173
+ print()
2174
+ header = (
2175
+ f" {'Model':<{widths['model']}} "
2176
+ f"{'Speed':<{widths['speed']}} "
2177
+ f"{'Strengths':<{widths['strengths']}} "
2178
+ f"Price"
2179
+ )
2180
+ print(color(header, Colors.CYAN))
2181
+
2182
+ rows = []
2183
+ for mid in ordered:
2184
+ meta = catalog[mid]
2185
+ row = (
2186
+ f" {mid:<{widths['model']}} "
2187
+ f"{meta.get('speed', ''):<{widths['speed']}} "
2188
+ f"{meta.get('strengths', ''):<{widths['strengths']}} "
2189
+ f"{meta.get('price', '')}"
2190
+ )
2191
+ if mid == current_model:
2192
+ row += " ← currently in use"
2193
+ rows.append(row)
2194
+
2195
+ idx = _prompt_choice(
2196
+ f" Choose {plugin_name} model:",
2197
+ rows,
2198
+ default=0,
2199
+ )
2200
+
2201
+ chosen = ordered[idx]
2202
+ cur_cfg["model"] = chosen
2203
+ _print_success(f" Model set to: {chosen}")
2204
+
2205
+
2206
+ def _select_plugin_video_gen_provider(plugin_name: str, config: dict) -> None:
2207
+ """Persist a plugin-backed video generation provider selection."""
2208
+ vid_cfg = config.setdefault("video_gen", {})
2209
+ if not isinstance(vid_cfg, dict):
2210
+ vid_cfg = {}
2211
+ config["video_gen"] = vid_cfg
2212
+ vid_cfg["provider"] = plugin_name
2213
+ vid_cfg["use_gateway"] = False
2214
+ _print_success(f" video_gen.provider set to: {plugin_name}")
2215
+ _configure_videogen_model_for_plugin(plugin_name, config)
2216
+
2217
+
2218
+ def _configure_provider(provider: dict, config: dict):
2219
+ """Configure a single provider - prompt for API keys and set config."""
2220
+ env_vars = provider.get("env_vars", [])
2221
+ managed_feature = provider.get("managed_nous_feature")
2222
+
2223
+ if provider.get("requires_nous_auth"):
2224
+ features = get_nous_subscription_features(config)
2225
+ if not features.nous_auth_present:
2226
+ _print_warning(" Nous Subscription is only available after logging into Nous Portal.")
2227
+ return
2228
+
2229
+ # Set TTS provider in config if applicable
2230
+ if provider.get("tts_provider"):
2231
+ tts_cfg = config.setdefault("tts", {})
2232
+ tts_cfg["provider"] = provider["tts_provider"]
2233
+ tts_cfg["use_gateway"] = bool(managed_feature)
2234
+
2235
+ # Set browser cloud provider in config if applicable
2236
+ if "browser_provider" in provider:
2237
+ bp = provider["browser_provider"]
2238
+ browser_cfg = config.setdefault("browser", {})
2239
+ if bp == "local":
2240
+ browser_cfg["cloud_provider"] = "local"
2241
+ _print_success(" Browser set to local mode")
2242
+ elif bp:
2243
+ browser_cfg["cloud_provider"] = bp
2244
+ _print_success(f" Browser cloud provider set to: {bp}")
2245
+ browser_cfg["use_gateway"] = bool(managed_feature)
2246
+
2247
+ # Set web search backend in config if applicable
2248
+ if provider.get("web_backend"):
2249
+ web_cfg = config.setdefault("web", {})
2250
+ web_cfg["backend"] = provider["web_backend"]
2251
+ web_cfg["use_gateway"] = bool(managed_feature)
2252
+ _print_success(f" Web backend set to: {provider['web_backend']}")
2253
+
2254
+ # For tools without a specific config key (e.g. image_gen), still
2255
+ # track use_gateway so the runtime knows the user's intent.
2256
+ if managed_feature and managed_feature not in {"web", "tts", "browser"}:
2257
+ config.setdefault(managed_feature, {})["use_gateway"] = True
2258
+ elif not managed_feature:
2259
+ # User picked a non-gateway provider — find which category this
2260
+ # belongs to and clear use_gateway if it was previously set.
2261
+ for cat_key, cat in TOOL_CATEGORIES.items():
2262
+ if provider in cat.get("providers", []):
2263
+ section = config.get(cat_key)
2264
+ if isinstance(section, dict) and section.get("use_gateway"):
2265
+ section["use_gateway"] = False
2266
+ break
2267
+
2268
+ if not env_vars:
2269
+ if provider.get("post_setup"):
2270
+ _run_post_setup(provider["post_setup"])
2271
+ _print_success(f" {provider['name']} - no configuration needed!")
2272
+ if managed_feature:
2273
+ _print_info(" Requests for this tool will be billed to your Nous subscription.")
2274
+ # Plugin-registered image_gen provider: write image_gen.provider
2275
+ # and route model selection to the plugin's own catalog.
2276
+ plugin_name = provider.get("image_gen_plugin_name")
2277
+ if plugin_name:
2278
+ _select_plugin_image_gen_provider(plugin_name, config)
2279
+ return
2280
+ # Plugin-registered video_gen provider — same flow, different
2281
+ # registry.
2282
+ video_plugin = provider.get("video_gen_plugin_name")
2283
+ if video_plugin:
2284
+ _select_plugin_video_gen_provider(video_plugin, config)
2285
+ return
2286
+ # Imagegen backends prompt for model selection after backend pick.
2287
+ backend = provider.get("imagegen_backend")
2288
+ if backend:
2289
+ _configure_imagegen_model(backend, config)
2290
+ # In-tree FAL is the only non-plugin backend today. Keep
2291
+ # image_gen.provider clear so the dispatch shim falls through
2292
+ # to the legacy FAL path.
2293
+ img_cfg = config.setdefault("image_gen", {})
2294
+ if isinstance(img_cfg, dict) and img_cfg.get("provider") not in {None, "", "fal"}:
2295
+ img_cfg["provider"] = "fal"
2296
+ return
2297
+
2298
+ # Prompt for each required env var
2299
+ all_configured = True
2300
+ for var in env_vars:
2301
+ existing = get_env_value(var["key"])
2302
+ if existing:
2303
+ _print_success(f" {var['key']}: already configured")
2304
+ # Don't ask to update - this is a new enable flow.
2305
+ # Reconfigure is handled separately.
2306
+ else:
2307
+ url = var.get("url", "")
2308
+ if url:
2309
+ _print_info(f" Get yours at: {url}")
2310
+
2311
+ default_val = var.get("default", "")
2312
+ if default_val:
2313
+ value = _prompt(f" {var.get('prompt', var['key'])}", default_val)
2314
+ else:
2315
+ value = _prompt(f" {var.get('prompt', var['key'])}", password=True)
2316
+
2317
+ if value:
2318
+ save_env_value(var["key"], value)
2319
+ _print_success(" Saved")
2320
+ else:
2321
+ _print_warning(" Skipped")
2322
+ all_configured = False
2323
+
2324
+ # Run post-setup hooks if needed
2325
+ if provider.get("post_setup") and all_configured:
2326
+ _run_post_setup(provider["post_setup"])
2327
+
2328
+ if all_configured:
2329
+ _print_success(f" {provider['name']} configured!")
2330
+ plugin_name = provider.get("image_gen_plugin_name")
2331
+ if plugin_name:
2332
+ _select_plugin_image_gen_provider(plugin_name, config)
2333
+ return
2334
+ video_plugin = provider.get("video_gen_plugin_name")
2335
+ if video_plugin:
2336
+ _select_plugin_video_gen_provider(video_plugin, config)
2337
+ return
2338
+ # Imagegen backends prompt for model selection after env vars are in.
2339
+ backend = provider.get("imagegen_backend")
2340
+ if backend:
2341
+ _configure_imagegen_model(backend, config)
2342
+ img_cfg = config.setdefault("image_gen", {})
2343
+ if isinstance(img_cfg, dict) and img_cfg.get("provider") not in {None, "", "fal"}:
2344
+ img_cfg["provider"] = "fal"
2345
+
2346
+
2347
+ def _configure_simple_requirements(ts_key: str):
2348
+ """Simple fallback for toolsets that just need env vars (no provider selection)."""
2349
+ if ts_key == "vision":
2350
+ if _toolset_has_keys("vision"):
2351
+ return
2352
+ print()
2353
+ print(color(" Vision / Image Analysis requires a multimodal backend:", Colors.YELLOW))
2354
+ choices = [
2355
+ "OpenRouter — uses Gemini",
2356
+ "OpenAI-compatible endpoint — base URL, API key, and vision model",
2357
+ "Skip",
2358
+ ]
2359
+ idx = _prompt_choice(" Configure vision backend", choices, 2)
2360
+ if idx == 0:
2361
+ _print_info(" Get key at: https://openrouter.ai/keys")
2362
+ value = _prompt(" OPENROUTER_API_KEY", password=True)
2363
+ if value and value.strip():
2364
+ save_env_value("OPENROUTER_API_KEY", value.strip())
2365
+ _print_success(" Saved")
2366
+ else:
2367
+ _print_warning(" Skipped")
2368
+ elif idx == 1:
2369
+ base_url = _prompt(" OPENAI_BASE_URL (blank for OpenAI)").strip() or "https://api.openai.com/v1"
2370
+ is_native_openai = base_url_hostname(base_url) == "api.openai.com"
2371
+ key_label = " OPENAI_API_KEY" if is_native_openai else " API key"
2372
+ api_key = _prompt(key_label, password=True)
2373
+ if api_key and api_key.strip():
2374
+ save_env_value("OPENAI_API_KEY", api_key.strip())
2375
+ # Save vision base URL to config (not .env — only secrets go there)
2376
+ _cfg = load_config()
2377
+ _aux = _cfg.setdefault("auxiliary", {}).setdefault("vision", {})
2378
+ _aux["base_url"] = base_url
2379
+ save_config(_cfg)
2380
+ if is_native_openai:
2381
+ save_env_value("AUXILIARY_VISION_MODEL", "gpt-4o-mini")
2382
+ _print_success(" Saved")
2383
+ else:
2384
+ _print_warning(" Skipped")
2385
+ return
2386
+
2387
+ requirements = TOOLSET_ENV_REQUIREMENTS.get(ts_key, [])
2388
+ if not requirements:
2389
+ return
2390
+
2391
+ missing = [(var, url) for var, url in requirements if not get_env_value(var)]
2392
+ if not missing:
2393
+ return
2394
+
2395
+ ts_label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts_key), ts_key)
2396
+ print()
2397
+ print(color(f" {ts_label} requires configuration:", Colors.YELLOW))
2398
+
2399
+ for var, url in missing:
2400
+ if url:
2401
+ _print_info(f" Get key at: {url}")
2402
+ value = _prompt(f" {var}", password=True)
2403
+ if value and value.strip():
2404
+ save_env_value(var, value.strip())
2405
+ _print_success(" Saved")
2406
+ else:
2407
+ _print_warning(" Skipped")
2408
+
2409
+
2410
+ def _reconfigure_tool(config: dict):
2411
+ """Let user reconfigure an existing tool's provider or API key."""
2412
+ # Build list of configurable tools that are currently set up
2413
+ configurable = []
2414
+ for ts_key, ts_label, _ in _get_effective_configurable_toolsets():
2415
+ cat = TOOL_CATEGORIES.get(ts_key)
2416
+ reqs = TOOLSET_ENV_REQUIREMENTS.get(ts_key)
2417
+ if cat or reqs:
2418
+ if _toolset_has_keys(ts_key, config) or _toolset_enabled_for_reconfigure(ts_key, config):
2419
+ configurable.append((ts_key, ts_label))
2420
+
2421
+ if not configurable:
2422
+ _print_info("No configured tools to reconfigure.")
2423
+ return
2424
+
2425
+ choices = [label for _, label in configurable]
2426
+ choices.append("Cancel")
2427
+
2428
+ idx = _prompt_choice(" Which tool would you like to reconfigure?", choices, len(choices) - 1)
2429
+
2430
+ if idx >= len(configurable):
2431
+ return # Cancel
2432
+
2433
+ ts_key, ts_label = configurable[idx]
2434
+ cat = TOOL_CATEGORIES.get(ts_key)
2435
+
2436
+ if cat:
2437
+ _configure_tool_category_for_reconfig(ts_key, cat, config)
2438
+ else:
2439
+ _reconfigure_simple_requirements(ts_key)
2440
+
2441
+ save_config(config)
2442
+
2443
+
2444
+ def _toolset_enabled_for_reconfigure(ts_key: str, config: dict) -> bool:
2445
+ """Return True if a configurable toolset is enabled anywhere.
2446
+
2447
+ Reconfigure must include enabled-but-unconfigured categories so users can
2448
+ finish provider/API-key setup without disabling and re-enabling the toolset.
2449
+ """
2450
+ for platform in PLATFORMS:
2451
+ if not _toolset_allowed_for_platform(ts_key, platform):
2452
+ continue
2453
+ try:
2454
+ enabled = _get_platform_tools(
2455
+ config,
2456
+ platform,
2457
+ include_default_mcp_servers=False,
2458
+ )
2459
+ except Exception:
2460
+ continue
2461
+ if ts_key in enabled:
2462
+ return True
2463
+ return False
2464
+
2465
+
2466
+ def _configure_tool_category_for_reconfig(ts_key: str, cat: dict, config: dict):
2467
+ """Reconfigure a tool category - provider selection + API key update."""
2468
+ icon = cat.get("icon", "")
2469
+ name = cat["name"]
2470
+ providers = _visible_providers(cat, config)
2471
+
2472
+ if len(providers) == 1:
2473
+ provider = providers[0]
2474
+ print()
2475
+ print(color(f" --- {icon} {name} ({provider['name']}) ---", Colors.CYAN))
2476
+ _reconfigure_provider(provider, config)
2477
+ else:
2478
+ print()
2479
+ print(color(f" --- {icon} {name} - Choose a provider ---", Colors.CYAN))
2480
+ print()
2481
+
2482
+ provider_choices = []
2483
+ for p in providers:
2484
+ badge = f" [{p['badge']}]" if p.get("badge") else ""
2485
+ tag = f" — {p['tag']}" if p.get("tag") else ""
2486
+ configured = ""
2487
+ env_vars = p.get("env_vars", [])
2488
+ if not env_vars or all(get_env_value(v["key"]) for v in env_vars):
2489
+ if _is_provider_active(p, config):
2490
+ configured = " [active]"
2491
+ elif not env_vars:
2492
+ configured = ""
2493
+ else:
2494
+ configured = " [configured]"
2495
+ provider_choices.append(f"{p['name']}{badge}{tag}{configured}")
2496
+
2497
+ default_idx = _detect_active_provider_index(providers, config)
2498
+
2499
+ provider_idx = _prompt_choice(" Select provider:", provider_choices, default_idx)
2500
+ _reconfigure_provider(providers[provider_idx], config)
2501
+
2502
+
2503
+ def _reconfigure_provider(provider: dict, config: dict):
2504
+ """Reconfigure a provider - update API keys."""
2505
+ env_vars = provider.get("env_vars", [])
2506
+ managed_feature = provider.get("managed_nous_feature")
2507
+
2508
+ if provider.get("requires_nous_auth"):
2509
+ features = get_nous_subscription_features(config)
2510
+ if not features.nous_auth_present:
2511
+ _print_warning(" Nous Subscription is only available after logging into Nous Portal.")
2512
+ return
2513
+
2514
+ if provider.get("tts_provider"):
2515
+ tts_cfg = config.setdefault("tts", {})
2516
+ tts_cfg["provider"] = provider["tts_provider"]
2517
+ tts_cfg["use_gateway"] = bool(managed_feature)
2518
+ _print_success(f" TTS provider set to: {provider['tts_provider']}")
2519
+
2520
+ if "browser_provider" in provider:
2521
+ bp = provider["browser_provider"]
2522
+ browser_cfg = config.setdefault("browser", {})
2523
+ if bp == "local":
2524
+ browser_cfg["cloud_provider"] = "local"
2525
+ _print_success(" Browser set to local mode")
2526
+ elif bp:
2527
+ browser_cfg["cloud_provider"] = bp
2528
+ _print_success(f" Browser cloud provider set to: {bp}")
2529
+ browser_cfg["use_gateway"] = bool(managed_feature)
2530
+
2531
+ # Set web search backend in config if applicable
2532
+ if provider.get("web_backend"):
2533
+ web_cfg = config.setdefault("web", {})
2534
+ web_cfg["backend"] = provider["web_backend"]
2535
+ web_cfg["use_gateway"] = bool(managed_feature)
2536
+ _print_success(f" Web backend set to: {provider['web_backend']}")
2537
+
2538
+ if managed_feature and managed_feature not in {"web", "tts", "browser"}:
2539
+ section = config.setdefault(managed_feature, {})
2540
+ if not isinstance(section, dict):
2541
+ section = {}
2542
+ config[managed_feature] = section
2543
+ section["use_gateway"] = True
2544
+ elif not managed_feature:
2545
+ for cat_key, cat in TOOL_CATEGORIES.items():
2546
+ if provider in cat.get("providers", []):
2547
+ section = config.get(cat_key)
2548
+ if isinstance(section, dict) and section.get("use_gateway"):
2549
+ section["use_gateway"] = False
2550
+ break
2551
+
2552
+ if not env_vars:
2553
+ if provider.get("post_setup"):
2554
+ _run_post_setup(provider["post_setup"])
2555
+ _print_success(f" {provider['name']} - no configuration needed!")
2556
+ if managed_feature:
2557
+ _print_info(" Requests for this tool will be billed to your Nous subscription.")
2558
+ plugin_name = provider.get("image_gen_plugin_name")
2559
+ if plugin_name:
2560
+ _select_plugin_image_gen_provider(plugin_name, config)
2561
+ return
2562
+ # Plugin-registered video_gen provider — same flow, different registry.
2563
+ video_plugin = provider.get("video_gen_plugin_name")
2564
+ if video_plugin:
2565
+ _select_plugin_video_gen_provider(video_plugin, config)
2566
+ return
2567
+ # Imagegen backends prompt for model selection on reconfig too.
2568
+ backend = provider.get("imagegen_backend")
2569
+ if backend:
2570
+ _configure_imagegen_model(backend, config)
2571
+ if backend == "fal":
2572
+ img_cfg = config.setdefault("image_gen", {})
2573
+ if isinstance(img_cfg, dict):
2574
+ img_cfg["provider"] = "fal"
2575
+ img_cfg["use_gateway"] = False
2576
+ return
2577
+
2578
+ for var in env_vars:
2579
+ existing = get_env_value(var["key"])
2580
+ if existing:
2581
+ _print_info(f" {var['key']}: configured ({existing[:8]}...)")
2582
+ url = var.get("url", "")
2583
+ if url:
2584
+ _print_info(f" Get yours at: {url}")
2585
+ default_val = var.get("default", "")
2586
+ value = _prompt(f" {var.get('prompt', var['key'])} (Enter to keep current)", password=not default_val)
2587
+ if value and value.strip():
2588
+ save_env_value(var["key"], value.strip())
2589
+ _print_success(" Updated")
2590
+ else:
2591
+ _print_info(" Kept current")
2592
+
2593
+ # Imagegen backends prompt for model selection on reconfig too.
2594
+ plugin_name = provider.get("image_gen_plugin_name")
2595
+ if plugin_name:
2596
+ _select_plugin_image_gen_provider(plugin_name, config)
2597
+ return
2598
+
2599
+ # Plugin-registered video_gen provider — same flow, different registry.
2600
+ video_plugin = provider.get("video_gen_plugin_name")
2601
+ if video_plugin:
2602
+ _select_plugin_video_gen_provider(video_plugin, config)
2603
+ return
2604
+
2605
+ backend = provider.get("imagegen_backend")
2606
+ if backend:
2607
+ _configure_imagegen_model(backend, config)
2608
+ if backend == "fal":
2609
+ img_cfg = config.setdefault("image_gen", {})
2610
+ if isinstance(img_cfg, dict):
2611
+ img_cfg["provider"] = "fal"
2612
+ img_cfg["use_gateway"] = False
2613
+
2614
+
2615
+ def _reconfigure_simple_requirements(ts_key: str):
2616
+ """Reconfigure simple env var requirements."""
2617
+ requirements = TOOLSET_ENV_REQUIREMENTS.get(ts_key, [])
2618
+ if not requirements:
2619
+ return
2620
+
2621
+ ts_label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts_key), ts_key)
2622
+ print()
2623
+ print(color(f" {ts_label}:", Colors.CYAN))
2624
+
2625
+ for var, url in requirements:
2626
+ existing = get_env_value(var)
2627
+ if existing:
2628
+ _print_info(f" {var}: configured ({existing[:8]}...)")
2629
+ if url:
2630
+ _print_info(f" Get key at: {url}")
2631
+ value = _prompt(f" {var} (Enter to keep current)", password=True)
2632
+ if value and value.strip():
2633
+ save_env_value(var, value.strip())
2634
+ _print_success(" Updated")
2635
+ else:
2636
+ _print_info(" Kept current")
2637
+
2638
+
2639
+ # ─── Main Entry Point ─────────────────────────────────────────────────────────
2640
+
2641
+ def tools_command(args=None, first_install: bool = False, config: dict = None):
2642
+ """Entry point for `hermes tools` and `hermes setup tools`.
2643
+
2644
+ Args:
2645
+ first_install: When True (set by the setup wizard on fresh installs),
2646
+ skip the platform menu, go straight to the CLI checklist, and
2647
+ prompt for API keys on all enabled tools that need them.
2648
+ config: Optional config dict to use. When called from the setup
2649
+ wizard, the wizard passes its own dict so that platform_toolsets
2650
+ are written into it and survive the wizard's final save_config().
2651
+ """
2652
+ if config is None:
2653
+ config = load_config()
2654
+ enabled_platforms = _get_enabled_platforms()
2655
+
2656
+ print()
2657
+
2658
+ # Non-interactive summary mode for CLI usage
2659
+ if getattr(args, "summary", False):
2660
+ total = len(_get_effective_configurable_toolsets())
2661
+ print(color("⚕ Tool Summary", Colors.CYAN, Colors.BOLD))
2662
+ print()
2663
+ summary = _platform_toolset_summary(config, enabled_platforms)
2664
+ for pkey in enabled_platforms:
2665
+ pinfo = PLATFORMS[pkey]
2666
+ enabled = summary.get(pkey, set())
2667
+ count = len(enabled)
2668
+ print(color(f" {pinfo['label']}", Colors.BOLD) + color(f" ({count}/{total})", Colors.DIM))
2669
+ if enabled:
2670
+ for ts_key in sorted(enabled):
2671
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts_key), ts_key)
2672
+ print(color(f" ✓ {label}", Colors.GREEN))
2673
+ else:
2674
+ print(color(" (none enabled)", Colors.DIM))
2675
+ print()
2676
+ return
2677
+ print(color("⚕ Hermes Tool Configuration", Colors.CYAN, Colors.BOLD))
2678
+ print(color(" Enable or disable tools per platform.", Colors.DIM))
2679
+ print(color(" Tools that need API keys will be configured when enabled.", Colors.DIM))
2680
+ print(color(" Guide: https://hermes-agent.nousresearch.com/docs/user-guide/features/tools", Colors.DIM))
2681
+ print()
2682
+
2683
+ # ── First-time install: linear flow, no platform menu ──
2684
+ if first_install:
2685
+ for pkey in enabled_platforms:
2686
+ pinfo = PLATFORMS[pkey]
2687
+ current_enabled = _get_platform_tools(config, pkey, include_default_mcp_servers=False)
2688
+
2689
+ # Uncheck toolsets that should be off by default
2690
+ checklist_preselected = current_enabled - _DEFAULT_OFF_TOOLSETS
2691
+
2692
+ # Show checklist
2693
+ new_enabled = _prompt_toolset_checklist(pinfo["label"], checklist_preselected, pkey)
2694
+
2695
+ added = new_enabled - current_enabled
2696
+ removed = current_enabled - new_enabled
2697
+ if added:
2698
+ for ts in sorted(added):
2699
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2700
+ print(color(f" + {label}", Colors.GREEN))
2701
+ if removed:
2702
+ for ts in sorted(removed):
2703
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2704
+ print(color(f" - {label}", Colors.RED))
2705
+
2706
+ auto_configured = apply_nous_managed_defaults(
2707
+ config,
2708
+ enabled_toolsets=new_enabled,
2709
+ )
2710
+ if managed_nous_tools_enabled():
2711
+ for ts_key in sorted(auto_configured):
2712
+ label = next((l for k, l, _ in CONFIGURABLE_TOOLSETS if k == ts_key), ts_key)
2713
+ print(color(f" ✓ {label}: using your Nous subscription defaults", Colors.GREEN))
2714
+
2715
+ # Walk through ALL selected tools that have provider options or
2716
+ # need API keys. This ensures browser (Local vs Browserbase),
2717
+ # TTS (Edge vs OpenAI vs ElevenLabs), etc. are shown even when
2718
+ # a free provider exists.
2719
+ to_configure = [
2720
+ ts_key for ts_key in sorted(new_enabled)
2721
+ if (TOOL_CATEGORIES.get(ts_key) or TOOLSET_ENV_REQUIREMENTS.get(ts_key))
2722
+ and ts_key not in auto_configured
2723
+ ]
2724
+
2725
+ if to_configure:
2726
+ print()
2727
+ print(color(f" Configuring {len(to_configure)} tool(s):", Colors.YELLOW))
2728
+ for ts_key in to_configure:
2729
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts_key), ts_key)
2730
+ print(color(f" • {label}", Colors.DIM))
2731
+ print(color(" You can skip any tool you don't need right now.", Colors.DIM))
2732
+ print()
2733
+ for ts_key in to_configure:
2734
+ _configure_toolset(ts_key, config)
2735
+
2736
+ _save_platform_tools(config, pkey, new_enabled)
2737
+ save_config(config)
2738
+ print(color(f" ✓ Saved {pinfo['label']} tool configuration", Colors.GREEN))
2739
+ print()
2740
+
2741
+ return
2742
+
2743
+ # ── Returning user: platform menu loop ──
2744
+ # Build platform choices
2745
+ platform_choices = []
2746
+ platform_keys = []
2747
+ for pkey in enabled_platforms:
2748
+ pinfo = PLATFORMS[pkey]
2749
+ current = _get_platform_tools(config, pkey, include_default_mcp_servers=False)
2750
+ count = len(current)
2751
+ total = len(_get_effective_configurable_toolsets())
2752
+ platform_choices.append(f"Configure {pinfo['label']} ({count}/{total} enabled)")
2753
+ platform_keys.append(pkey)
2754
+
2755
+ if len(platform_keys) > 1:
2756
+ platform_choices.append("Configure all platforms (global)")
2757
+ platform_choices.append("Reconfigure an existing tool's provider or API key")
2758
+
2759
+ # Show MCP option if any MCP servers are configured
2760
+ _has_mcp = bool(config.get("mcp_servers"))
2761
+ if _has_mcp:
2762
+ platform_choices.append("Configure MCP server tools")
2763
+
2764
+ platform_choices.append("Done")
2765
+
2766
+ # Index offsets for the extra options after per-platform entries
2767
+ _global_idx = len(platform_keys) if len(platform_keys) > 1 else -1
2768
+ _reconfig_idx = len(platform_keys) + (1 if len(platform_keys) > 1 else 0)
2769
+ _mcp_idx = (_reconfig_idx + 1) if _has_mcp else -1
2770
+ _done_idx = _reconfig_idx + (2 if _has_mcp else 1)
2771
+
2772
+ while True:
2773
+ idx = _prompt_choice("Select an option:", platform_choices, default=0)
2774
+
2775
+ # "Done" selected
2776
+ if idx == _done_idx:
2777
+ break
2778
+
2779
+ # "Reconfigure" selected
2780
+ if idx == _reconfig_idx:
2781
+ _reconfigure_tool(config)
2782
+ print()
2783
+ continue
2784
+
2785
+ # "Configure MCP tools" selected
2786
+ if idx == _mcp_idx:
2787
+ _configure_mcp_tools_interactive(config)
2788
+ print()
2789
+ continue
2790
+
2791
+ # "Configure all platforms (global)" selected
2792
+ if idx == _global_idx:
2793
+ # Use the union of all platforms' current tools as the starting state
2794
+ all_current = set()
2795
+ for pk in platform_keys:
2796
+ all_current |= _get_platform_tools(config, pk, include_default_mcp_servers=False)
2797
+ new_enabled = _prompt_toolset_checklist("All platforms", all_current)
2798
+ if new_enabled != all_current:
2799
+ for pk in platform_keys:
2800
+ prev = _get_platform_tools(config, pk, include_default_mcp_servers=False)
2801
+ added = new_enabled - prev
2802
+ removed = prev - new_enabled
2803
+ pinfo_inner = PLATFORMS[pk]
2804
+ if added or removed:
2805
+ print(color(f" {pinfo_inner['label']}:", Colors.DIM))
2806
+ for ts in sorted(added):
2807
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2808
+ print(color(f" + {label}", Colors.GREEN))
2809
+ for ts in sorted(removed):
2810
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2811
+ print(color(f" - {label}", Colors.RED))
2812
+ # Configure API keys for newly enabled tools
2813
+ for ts_key in sorted(added):
2814
+ if (TOOL_CATEGORIES.get(ts_key) or TOOLSET_ENV_REQUIREMENTS.get(ts_key)):
2815
+ if _toolset_needs_configuration_prompt(ts_key, config):
2816
+ _configure_toolset(ts_key, config)
2817
+ _save_platform_tools(config, pk, new_enabled)
2818
+ save_config(config)
2819
+ print(color(" ✓ Saved configuration for all platforms", Colors.GREEN))
2820
+ # Update choice labels
2821
+ for ci, pk in enumerate(platform_keys):
2822
+ new_count = len(_get_platform_tools(config, pk, include_default_mcp_servers=False))
2823
+ total = len(_get_effective_configurable_toolsets())
2824
+ platform_choices[ci] = f"Configure {PLATFORMS[pk]['label']} ({new_count}/{total} enabled)"
2825
+ else:
2826
+ print(color(" No changes", Colors.DIM))
2827
+ print()
2828
+ continue
2829
+
2830
+ pkey = platform_keys[idx]
2831
+ pinfo = PLATFORMS[pkey]
2832
+
2833
+ # Get current enabled toolsets for this platform
2834
+ current_enabled = _get_platform_tools(config, pkey, include_default_mcp_servers=False)
2835
+
2836
+ # Show checklist
2837
+ new_enabled = _prompt_toolset_checklist(pinfo["label"], current_enabled)
2838
+
2839
+ if new_enabled != current_enabled:
2840
+ added = new_enabled - current_enabled
2841
+ removed = current_enabled - new_enabled
2842
+
2843
+ if added:
2844
+ for ts in sorted(added):
2845
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2846
+ print(color(f" + {label}", Colors.GREEN))
2847
+ if removed:
2848
+ for ts in sorted(removed):
2849
+ label = next((l for k, l, _ in _get_effective_configurable_toolsets() if k == ts), ts)
2850
+ print(color(f" - {label}", Colors.RED))
2851
+
2852
+ # Configure newly enabled toolsets that need API keys
2853
+ for ts_key in sorted(added):
2854
+ if (TOOL_CATEGORIES.get(ts_key) or TOOLSET_ENV_REQUIREMENTS.get(ts_key)):
2855
+ if _toolset_needs_configuration_prompt(ts_key, config):
2856
+ _configure_toolset(ts_key, config)
2857
+
2858
+ _save_platform_tools(config, pkey, new_enabled)
2859
+ save_config(config)
2860
+ print(color(f" ✓ Saved {pinfo['label']} configuration", Colors.GREEN))
2861
+ else:
2862
+ print(color(f" No changes to {pinfo['label']}", Colors.DIM))
2863
+
2864
+ print()
2865
+
2866
+ # Update the choice label with new count
2867
+ new_count = len(_get_platform_tools(config, pkey, include_default_mcp_servers=False))
2868
+ total = len(_get_effective_configurable_toolsets())
2869
+ platform_choices[idx] = f"Configure {pinfo['label']} ({new_count}/{total} enabled)"
2870
+
2871
+ print()
2872
+ from calvyn_constants import display_hermes_home
2873
+ print(color(f" Tool configuration saved to {display_hermes_home()}/config.yaml", Colors.DIM))
2874
+ print(color(" Changes take effect on next 'hermes' or gateway restart.", Colors.DIM))
2875
+ print()
2876
+
2877
+
2878
+ # ─── MCP Tools Interactive Configuration ─────────────────────────────────────
2879
+
2880
+
2881
+ def _configure_mcp_tools_interactive(config: dict):
2882
+ """Probe MCP servers for available tools and let user toggle them on/off.
2883
+
2884
+ Connects to each configured MCP server, discovers tools, then shows
2885
+ a per-server curses checklist. Writes changes back as ``tools.exclude``
2886
+ entries in config.yaml.
2887
+ """
2888
+ from hermes_cli.curses_ui import curses_checklist
2889
+
2890
+ mcp_servers = config.get("mcp_servers") or {}
2891
+ if not mcp_servers:
2892
+ _print_info("No MCP servers configured.")
2893
+ return
2894
+
2895
+ # Count enabled servers
2896
+ enabled_names = [
2897
+ k for k, v in mcp_servers.items()
2898
+ if v.get("enabled", True) not in {False, "false", "0", "no", "off"}
2899
+ ]
2900
+ if not enabled_names:
2901
+ _print_info("All MCP servers are disabled.")
2902
+ return
2903
+
2904
+ print()
2905
+ print(color(" Discovering tools from MCP servers...", Colors.YELLOW))
2906
+ print(color(f" Connecting to {len(enabled_names)} server(s): {', '.join(enabled_names)}", Colors.DIM))
2907
+
2908
+ try:
2909
+ from tools.mcp_tool import probe_mcp_server_tools
2910
+ server_tools = probe_mcp_server_tools()
2911
+ except Exception as exc:
2912
+ _print_error(f"Failed to probe MCP servers: {exc}")
2913
+ return
2914
+
2915
+ if not server_tools:
2916
+ _print_warning("Could not discover tools from any MCP server.")
2917
+ _print_info("Check that server commands/URLs are correct and dependencies are installed.")
2918
+ return
2919
+
2920
+ # Report discovery results
2921
+ failed = [n for n in enabled_names if n not in server_tools]
2922
+ if failed:
2923
+ for name in failed:
2924
+ _print_warning(f" Could not connect to '{name}'")
2925
+
2926
+ total_tools = sum(len(tools) for tools in server_tools.values())
2927
+ print(color(f" Found {total_tools} tool(s) across {len(server_tools)} server(s)", Colors.GREEN))
2928
+ print()
2929
+
2930
+ any_changes = False
2931
+
2932
+ for server_name, tools in server_tools.items():
2933
+ if not tools:
2934
+ _print_info(f" {server_name}: no tools found")
2935
+ continue
2936
+
2937
+ srv_cfg = mcp_servers.get(server_name, {})
2938
+ tools_cfg = srv_cfg.get("tools") or {}
2939
+ include_list = tools_cfg.get("include") or []
2940
+ exclude_list = tools_cfg.get("exclude") or []
2941
+
2942
+ # Build checklist labels
2943
+ labels = []
2944
+ for tool_name, description in tools:
2945
+ desc_short = description[:70] + "..." if len(description) > 70 else description
2946
+ if desc_short:
2947
+ labels.append(f"{tool_name} ({desc_short})")
2948
+ else:
2949
+ labels.append(tool_name)
2950
+
2951
+ # Determine which tools are currently enabled
2952
+ pre_selected: Set[int] = set()
2953
+ tool_names = [t[0] for t in tools]
2954
+ for i, tool_name in enumerate(tool_names):
2955
+ if include_list:
2956
+ # Include mode: only included tools are selected
2957
+ if tool_name in include_list:
2958
+ pre_selected.add(i)
2959
+ elif exclude_list:
2960
+ # Exclude mode: everything except excluded
2961
+ if tool_name not in exclude_list:
2962
+ pre_selected.add(i)
2963
+ else:
2964
+ # No filter: all enabled
2965
+ pre_selected.add(i)
2966
+
2967
+ chosen = curses_checklist(
2968
+ f"MCP Server: {server_name} ({len(tools)} tools)",
2969
+ labels,
2970
+ pre_selected,
2971
+ cancel_returns=pre_selected,
2972
+ )
2973
+
2974
+ if chosen == pre_selected:
2975
+ _print_info(f" {server_name}: no changes")
2976
+ continue
2977
+
2978
+ # Compute new exclude list based on unchecked tools
2979
+ new_exclude = [tool_names[i] for i in range(len(tool_names)) if i not in chosen]
2980
+
2981
+ # Update config
2982
+ srv_cfg = mcp_servers.setdefault(server_name, {})
2983
+ tools_cfg = srv_cfg.setdefault("tools", {})
2984
+
2985
+ if new_exclude:
2986
+ tools_cfg["exclude"] = new_exclude
2987
+ # Remove include if present — we're switching to exclude mode
2988
+ tools_cfg.pop("include", None)
2989
+ else:
2990
+ # All tools enabled — clear filters
2991
+ tools_cfg.pop("exclude", None)
2992
+ tools_cfg.pop("include", None)
2993
+
2994
+ enabled_count = len(chosen)
2995
+ disabled_count = len(tools) - enabled_count
2996
+ _print_success(
2997
+ f" {server_name}: {enabled_count} enabled, {disabled_count} disabled"
2998
+ )
2999
+ any_changes = True
3000
+
3001
+ if any_changes:
3002
+ save_config(config)
3003
+ print()
3004
+ print(color(" ✓ MCP tool configuration saved", Colors.GREEN))
3005
+ else:
3006
+ print(color(" No changes to MCP tools", Colors.DIM))
3007
+
3008
+
3009
+ # ─── Non-interactive disable/enable ──────────────────────────────────────────
3010
+
3011
+
3012
+ def _apply_toolset_change(config: dict, platform: str, toolset_names: List[str], action: str):
3013
+ """Add or remove built-in toolsets for a platform."""
3014
+ enabled = _get_platform_tools(config, platform, include_default_mcp_servers=False)
3015
+ if action == "disable":
3016
+ updated = enabled - set(toolset_names)
3017
+ else:
3018
+ updated = enabled | set(toolset_names)
3019
+ _save_platform_tools(config, platform, updated)
3020
+
3021
+
3022
+ def _apply_mcp_change(config: dict, targets: List[str], action: str) -> Set[str]:
3023
+ """Add or remove specific MCP tools from a server's exclude list.
3024
+
3025
+ Returns the set of server names that were not found in config.
3026
+ """
3027
+ failed_servers: Set[str] = set()
3028
+ mcp_servers = config.get("mcp_servers") or {}
3029
+
3030
+ for target in targets:
3031
+ server_name, tool_name = target.split(":", 1)
3032
+ if server_name not in mcp_servers:
3033
+ failed_servers.add(server_name)
3034
+ continue
3035
+ tools_cfg = mcp_servers[server_name].setdefault("tools", {})
3036
+ exclude = list(tools_cfg.get("exclude") or [])
3037
+ if action == "disable":
3038
+ if tool_name not in exclude:
3039
+ exclude.append(tool_name)
3040
+ else:
3041
+ exclude = [t for t in exclude if t != tool_name]
3042
+ tools_cfg["exclude"] = exclude
3043
+
3044
+ return failed_servers
3045
+
3046
+
3047
+ def _print_tools_list(enabled_toolsets: set, mcp_servers: dict, platform: str = "cli"):
3048
+ """Print a summary of enabled/disabled toolsets and MCP tool filters."""
3049
+ effective_all = _get_effective_configurable_toolsets()
3050
+ effective = [
3051
+ (k, l, d) for (k, l, d) in effective_all
3052
+ if _toolset_allowed_for_platform(k, platform)
3053
+ ]
3054
+ builtin_keys = {ts_key for ts_key, _, _ in CONFIGURABLE_TOOLSETS}
3055
+
3056
+ print(f"Built-in toolsets ({platform}):")
3057
+ for ts_key, label, _ in effective:
3058
+ if ts_key not in builtin_keys:
3059
+ continue
3060
+ status = (color("✓ enabled", Colors.GREEN) if ts_key in enabled_toolsets
3061
+ else color("✗ disabled", Colors.RED))
3062
+ print(f" {status} {ts_key} {color(label, Colors.DIM)}")
3063
+
3064
+ # Plugin toolsets
3065
+ plugin_entries = [(k, l) for k, l, _ in effective if k not in builtin_keys]
3066
+ if plugin_entries:
3067
+ print()
3068
+ print(f"Plugin toolsets ({platform}):")
3069
+ for ts_key, label in plugin_entries:
3070
+ status = (color("✓ enabled", Colors.GREEN) if ts_key in enabled_toolsets
3071
+ else color("✗ disabled", Colors.RED))
3072
+ print(f" {status} {ts_key} {color(label, Colors.DIM)}")
3073
+
3074
+ if mcp_servers:
3075
+ print()
3076
+ print("MCP servers:")
3077
+ for srv_name, srv_cfg in mcp_servers.items():
3078
+ tools_cfg = srv_cfg.get("tools") or {}
3079
+ exclude = tools_cfg.get("exclude") or []
3080
+ include = tools_cfg.get("include") or []
3081
+ if include:
3082
+ _print_info(f"{srv_name} [include only: {', '.join(include)}]")
3083
+ elif exclude:
3084
+ _print_info(f"{srv_name} [excluded: {color(', '.join(exclude), Colors.YELLOW)}]")
3085
+ else:
3086
+ _print_info(f"{srv_name} {color('all tools enabled', Colors.DIM)}")
3087
+
3088
+
3089
+ def tools_disable_enable_command(args):
3090
+ """Enable, disable, or list tools for a platform.
3091
+
3092
+ Built-in toolsets use plain names (e.g. ``web``, ``memory``).
3093
+ MCP tools use ``server:tool`` notation (e.g. ``github:create_issue``).
3094
+ """
3095
+ action = args.tools_action
3096
+ platform = getattr(args, "platform", "cli")
3097
+ config = load_config()
3098
+
3099
+ if platform not in PLATFORMS:
3100
+ _print_error(f"Unknown platform '{platform}'. Valid: {', '.join(PLATFORMS)}")
3101
+ return
3102
+
3103
+ if action == "list":
3104
+ _print_tools_list(_get_platform_tools(config, platform, include_default_mcp_servers=False),
3105
+ config.get("mcp_servers") or {}, platform)
3106
+ return
3107
+
3108
+ targets: List[str] = args.names
3109
+ toolset_targets = [t for t in targets if ":" not in t]
3110
+ mcp_targets = [t for t in targets if ":" in t]
3111
+
3112
+ valid_toolsets = {ts_key for ts_key, _, _ in CONFIGURABLE_TOOLSETS} | _get_plugin_toolset_keys()
3113
+ unknown_toolsets = [t for t in toolset_targets if t not in valid_toolsets]
3114
+ if unknown_toolsets:
3115
+ for name in unknown_toolsets:
3116
+ _print_error(f"Unknown toolset '{name}'")
3117
+ toolset_targets = [t for t in toolset_targets if t in valid_toolsets]
3118
+
3119
+ # Reject platform-scoped toolsets on platforms that don't allow them.
3120
+ restricted_targets = [
3121
+ t for t in toolset_targets
3122
+ if not _toolset_allowed_for_platform(t, platform)
3123
+ ]
3124
+ if restricted_targets:
3125
+ for name in restricted_targets:
3126
+ allowed = sorted(_TOOLSET_PLATFORM_RESTRICTIONS.get(name) or set())
3127
+ _print_error(
3128
+ f"Toolset '{name}' is not available on platform '{platform}' "
3129
+ f"(only: {', '.join(allowed)})"
3130
+ )
3131
+ toolset_targets = [t for t in toolset_targets if t not in restricted_targets]
3132
+
3133
+ if toolset_targets:
3134
+ _apply_toolset_change(config, platform, toolset_targets, action)
3135
+
3136
+ failed_servers: Set[str] = set()
3137
+ if mcp_targets:
3138
+ failed_servers = _apply_mcp_change(config, mcp_targets, action)
3139
+ for srv in failed_servers:
3140
+ _print_error(f"MCP server '{srv}' not found in config")
3141
+
3142
+ save_config(config)
3143
+
3144
+ successful = [
3145
+ t for t in targets
3146
+ if t not in unknown_toolsets and (":" not in t or t.split(":")[0] not in failed_servers)
3147
+ ]
3148
+ if successful:
3149
+ verb = "Disabled" if action == "disable" else "Enabled"
3150
+ _print_success(f"{verb}: {', '.join(successful)}")
3151
+