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,1534 @@
1
+ """
2
+ Process Registry -- In-memory registry for managed background processes.
3
+
4
+ Tracks processes spawned via terminal(background=true), providing:
5
+ - Output buffering (rolling 200KB window)
6
+ - Status polling and log retrieval
7
+ - Blocking wait with interrupt support
8
+ - Process killing
9
+ - Crash recovery via JSON checkpoint file
10
+ - Session-scoped tracking for gateway reset protection
11
+
12
+ Background processes execute THROUGH the environment interface -- nothing
13
+ runs on the host machine unless TERMINAL_ENV=local. For Docker, Singularity,
14
+ Modal, Daytona, and SSH backends, the command runs inside the sandbox.
15
+
16
+ Usage:
17
+ from tools.process_registry import process_registry
18
+
19
+ # Spawn a background process (called from terminal_tool)
20
+ session = process_registry.spawn(env, "pytest -v", task_id="task_123")
21
+
22
+ # Poll for status
23
+ result = process_registry.poll(session.id)
24
+
25
+ # Block until done
26
+ result = process_registry.wait(session.id, timeout=300)
27
+
28
+ # Kill it
29
+ process_registry.kill(session.id)
30
+ """
31
+
32
+ import json
33
+ import logging
34
+ import os
35
+ import platform
36
+ import shlex
37
+ import signal
38
+ import subprocess
39
+ import threading
40
+ import time
41
+ import uuid
42
+
43
+ _IS_WINDOWS = platform.system() == "Windows"
44
+ from tools.environments.local import _find_shell, _resolve_safe_cwd, _sanitize_subprocess_env
45
+ from dataclasses import dataclass, field
46
+ from typing import Any, Dict, List, Optional
47
+
48
+ from hermes_cli.config import get_hermes_home
49
+
50
+ logger = logging.getLogger(__name__)
51
+
52
+
53
+ # Checkpoint file for crash recovery (gateway only)
54
+ CHECKPOINT_PATH = get_hermes_home() / "processes.json"
55
+
56
+ # Limits
57
+ MAX_OUTPUT_CHARS = 200_000 # 200KB rolling output buffer
58
+ FINISHED_TTL_SECONDS = 1800 # Keep finished processes for 30 minutes
59
+ MAX_PROCESSES = 64 # Max concurrent tracked processes (LRU pruning)
60
+
61
+ # Watch pattern rate limiting — PER SESSION.
62
+ # Hard rule: at most ONE watch-match notification every WATCH_MIN_INTERVAL_SECONDS.
63
+ # Any match arriving inside that cooldown window is dropped and counted as a strike.
64
+ # After WATCH_STRIKE_LIMIT consecutive strike windows, watch_patterns for that
65
+ # session is permanently disabled and the session falls back to notify_on_complete
66
+ # semantics (one notification when the process actually exits).
67
+ WATCH_MIN_INTERVAL_SECONDS = 15 # Minimum spacing between consecutive watch matches
68
+ WATCH_STRIKE_LIMIT = 3 # Strikes in a row → disable watch + promote to notify_on_complete
69
+
70
+ # Global circuit breaker — across all sessions. Secondary safety net so concurrent
71
+ # siblings can't collectively flood the user even when each is under its own cap.
72
+ WATCH_GLOBAL_MAX_PER_WINDOW = 15
73
+ WATCH_GLOBAL_WINDOW_SECONDS = 10
74
+ WATCH_GLOBAL_COOLDOWN_SECONDS = 30
75
+
76
+
77
+ def format_uptime_short(seconds: int) -> str:
78
+ s = max(0, int(seconds))
79
+ if s < 60:
80
+ return f"{s}s"
81
+ mins, secs = divmod(s, 60)
82
+ if mins < 60:
83
+ return f"{mins}m {secs}s"
84
+ hours, mins = divmod(mins, 60)
85
+ return f"{hours}h {mins}m"
86
+
87
+
88
+ @dataclass
89
+ class ProcessSession:
90
+ """A tracked background process with output buffering."""
91
+ id: str # Unique session ID ("proc_xxxxxxxxxxxx")
92
+ command: str # Original command string
93
+ task_id: str = "" # Task/sandbox isolation key
94
+ session_key: str = "" # Gateway session key (for reset protection)
95
+ pid: Optional[int] = None # OS process ID
96
+ process: Optional[subprocess.Popen] = None # Popen handle (local only)
97
+ env_ref: Any = None # Reference to the environment object
98
+ cwd: Optional[str] = None # Working directory
99
+ started_at: float = 0.0 # time.time() of spawn
100
+ exited: bool = False # Whether the process has finished
101
+ exit_code: Optional[int] = None # Exit code (None if still running)
102
+ output_buffer: str = "" # Rolling output (last MAX_OUTPUT_CHARS)
103
+ max_output_chars: int = MAX_OUTPUT_CHARS
104
+ detached: bool = False # True if recovered from crash (no pipe)
105
+ pid_scope: str = "host" # "host" for local/PTY PIDs, "sandbox" for env-local PIDs
106
+ # Watcher/notification metadata (persisted for crash recovery)
107
+ watcher_platform: str = ""
108
+ watcher_chat_id: str = ""
109
+ watcher_user_id: str = ""
110
+ watcher_user_name: str = ""
111
+ watcher_thread_id: str = ""
112
+ watcher_interval: int = 0 # 0 = no watcher configured
113
+ notify_on_complete: bool = False # Queue agent notification on exit
114
+ # Watch patterns — trigger agent notification when output matches any pattern
115
+ watch_patterns: List[str] = field(default_factory=list)
116
+ _watch_hits: int = field(default=0, repr=False) # total matches delivered
117
+ _watch_suppressed: int = field(default=0, repr=False) # matches dropped by rate limit
118
+ _watch_disabled: bool = field(default=False, repr=False) # permanently killed after strike limit
119
+ # Per-session rate limit state: at most one match every WATCH_MIN_INTERVAL_SECONDS.
120
+ # When an emission happens, _watch_cooldown_until is set to now + interval and
121
+ # _watch_strike_candidate becomes True. The next match to arrive before that
122
+ # deadline counts as one strike (regardless of how many matches were dropped in
123
+ # between — a strike is a window, not a match). After WATCH_STRIKE_LIMIT strikes
124
+ # in a row, watch_patterns is disabled and the session promotes to
125
+ # notify_on_complete.
126
+ _watch_last_emit_at: float = field(default=0.0, repr=False)
127
+ _watch_cooldown_until: float = field(default=0.0, repr=False)
128
+ _watch_strike_candidate: bool = field(default=False, repr=False)
129
+ _watch_consecutive_strikes: int = field(default=0, repr=False)
130
+ _lock: threading.Lock = field(default_factory=threading.Lock)
131
+ _reader_thread: Optional[threading.Thread] = field(default=None, repr=False)
132
+ _pty: Any = field(default=None, repr=False) # ptyprocess handle (when use_pty=True)
133
+
134
+
135
+ class ProcessRegistry:
136
+ """
137
+ In-memory registry of running and finished background processes.
138
+
139
+ Thread-safe. Accessed from:
140
+ - Executor threads (terminal_tool, process tool handlers)
141
+ - Gateway asyncio loop (watcher tasks, session reset checks)
142
+ - Cleanup thread (sandbox reaping coordination)
143
+ """
144
+
145
+ _SHELL_NOISE_SUBSTRINGS = (
146
+ "bash: cannot set terminal process group",
147
+ "bash: no job control in this shell",
148
+ "no job control in this shell",
149
+ "cannot set terminal process group",
150
+ "tcsetattr: Inappropriate ioctl for device",
151
+ )
152
+
153
+ def __init__(self):
154
+ self._running: Dict[str, ProcessSession] = {}
155
+ self._finished: Dict[str, ProcessSession] = {}
156
+ self._lock = threading.Lock()
157
+
158
+ # Side-channel for check_interval watchers (gateway reads after agent run)
159
+ self.pending_watchers: List[Dict[str, Any]] = []
160
+
161
+ # Notification queue — unified queue for all background process events.
162
+ # Completion notifications (notify_on_complete) and watch pattern matches
163
+ # both land here, distinguished by "type" field. CLI process_loop and
164
+ # gateway drain this after each agent turn to auto-trigger new turns.
165
+ import queue as _queue_mod
166
+ self.completion_queue: _queue_mod.Queue = _queue_mod.Queue()
167
+
168
+ # Track sessions whose completion was already consumed by the agent
169
+ # via wait/poll/log. Drain loops skip notifications for these.
170
+ self._completion_consumed: set = set()
171
+
172
+ # Global watch-match circuit breaker — across all sessions.
173
+ # Prevents sibling processes from collectively flooding the user even
174
+ # when each stays under its own per-session cap.
175
+ self._global_watch_lock = threading.Lock()
176
+ self._global_watch_window_start: float = 0.0
177
+ self._global_watch_window_hits: int = 0
178
+ self._global_watch_tripped_until: float = 0.0
179
+ self._global_watch_suppressed_during_trip: int = 0
180
+
181
+ @staticmethod
182
+ def _clean_shell_noise(text: str) -> str:
183
+ """Strip shell startup warnings from the beginning of output."""
184
+ lines = text.split("\n")
185
+ while lines and any(noise in lines[0] for noise in ProcessRegistry._SHELL_NOISE_SUBSTRINGS):
186
+ lines.pop(0)
187
+ return "\n".join(lines)
188
+
189
+ def _check_watch_patterns(self, session: ProcessSession, new_text: str) -> None:
190
+ """Scan new output for watch patterns and queue notifications.
191
+
192
+ Called from reader threads with new_text being the freshly-read chunk.
193
+
194
+ Per-session rate limit: at most ONE watch-match notification per
195
+ WATCH_MIN_INTERVAL_SECONDS. Any match arriving inside the cooldown
196
+ window is dropped and counts as ONE strike for that window. After
197
+ WATCH_STRIKE_LIMIT consecutive strike windows, watch_patterns is
198
+ disabled for this session and the session is promoted to
199
+ notify_on_complete semantics — one notification when the process
200
+ actually exits, no more mid-process spam.
201
+ """
202
+ if not session.watch_patterns or session._watch_disabled:
203
+ return
204
+ # Suppress-after-exit: once the reader loop has declared the process
205
+ # exited, any late chunk we still see is post-exit noise. Dropping these
206
+ # prevents the "stale notifications delivered minutes after the process
207
+ # ended" spam when completion_queue consumers run async.
208
+ if session.exited:
209
+ return
210
+
211
+ # Scan new text line-by-line for pattern matches
212
+ matched_lines = []
213
+ matched_pattern = None
214
+ for line in new_text.splitlines():
215
+ for pat in session.watch_patterns:
216
+ if pat in line:
217
+ matched_lines.append(line.rstrip())
218
+ if matched_pattern is None:
219
+ matched_pattern = pat
220
+ break # one match per line is enough
221
+
222
+ if not matched_lines:
223
+ return
224
+
225
+ now = time.time()
226
+ should_disable = False
227
+ with session._lock:
228
+ # Case 1: still inside the cooldown from the last emission.
229
+ # Count this as a strike for the current window (only once per window)
230
+ # and drop the event. If we've hit the strike limit, disable watch
231
+ # and promote to notify_on_complete.
232
+ if session._watch_cooldown_until and now < session._watch_cooldown_until:
233
+ session._watch_suppressed += len(matched_lines)
234
+ if not session._watch_strike_candidate:
235
+ # First drop in this window — count one strike.
236
+ session._watch_strike_candidate = True
237
+ session._watch_consecutive_strikes += 1
238
+ if session._watch_consecutive_strikes >= WATCH_STRIKE_LIMIT:
239
+ session._watch_disabled = True
240
+ # Promote to notify_on_complete so the agent still gets
241
+ # exactly one notification when the process actually ends.
242
+ session.notify_on_complete = True
243
+ should_disable = True
244
+ return_early = True
245
+ else:
246
+ # Case 2: cooldown has expired.
247
+ # Decide whether this window was a "clean" one (no drops) or a
248
+ # strike window. If no strike candidate was set during the prior
249
+ # cooldown, reset the consecutive-strike counter — we're back to
250
+ # healthy emission cadence.
251
+ if (
252
+ session._watch_cooldown_until
253
+ and not session._watch_strike_candidate
254
+ ):
255
+ session._watch_consecutive_strikes = 0
256
+ session._watch_strike_candidate = False
257
+
258
+ # Emit the notification and start a new cooldown window.
259
+ session._watch_last_emit_at = now
260
+ session._watch_cooldown_until = now + WATCH_MIN_INTERVAL_SECONDS
261
+ session._watch_hits += 1
262
+ suppressed = session._watch_suppressed
263
+ session._watch_suppressed = 0
264
+ return_early = False
265
+
266
+ if return_early:
267
+ if should_disable:
268
+ # Emit exactly one "watch disabled, falling back to notify_on_complete"
269
+ # summary event so the agent/user sees why things went quiet.
270
+ self.completion_queue.put({
271
+ "session_id": session.id,
272
+ "session_key": session.session_key,
273
+ "command": session.command,
274
+ "type": "watch_disabled",
275
+ "suppressed": session._watch_suppressed,
276
+ "platform": session.watcher_platform,
277
+ "chat_id": session.watcher_chat_id,
278
+ "user_id": session.watcher_user_id,
279
+ "user_name": session.watcher_user_name,
280
+ "thread_id": session.watcher_thread_id,
281
+ "message": (
282
+ f"Watch patterns disabled for process {session.id} — "
283
+ f"{WATCH_STRIKE_LIMIT} consecutive rate-limit windows triggered "
284
+ f"(min spacing {WATCH_MIN_INTERVAL_SECONDS}s). "
285
+ f"Falling back to notify_on_complete semantics; you'll get "
286
+ f"exactly one notification when the process exits."
287
+ ),
288
+ })
289
+ return
290
+
291
+ # Trim matched output to a reasonable size
292
+ output = "\n".join(matched_lines[:20])
293
+ if len(output) > 2000:
294
+ output = output[:2000] + "\n...(truncated)"
295
+
296
+ # Global circuit breaker — across all sessions (secondary safety net).
297
+ if not self._global_watch_admit(now):
298
+ return
299
+
300
+ self.completion_queue.put({
301
+ "session_id": session.id,
302
+ "session_key": session.session_key,
303
+ "command": session.command,
304
+ "type": "watch_match",
305
+ "pattern": matched_pattern,
306
+ "output": output,
307
+ "suppressed": suppressed,
308
+ "platform": session.watcher_platform,
309
+ "chat_id": session.watcher_chat_id,
310
+ "user_id": session.watcher_user_id,
311
+ "user_name": session.watcher_user_name,
312
+ "thread_id": session.watcher_thread_id,
313
+ })
314
+
315
+ def _global_watch_admit(self, now: float) -> bool:
316
+ """Return True if this watch_match event is allowed through the global breaker.
317
+
318
+ Semantics:
319
+ - If we're currently in a cooldown period, drop the event and count it.
320
+ - Otherwise, slide the rolling window and check the global cap.
321
+ - If the cap is exceeded, trip the breaker for WATCH_GLOBAL_COOLDOWN_SECONDS
322
+ and emit ONE summary event so the agent/user sees "N notifications were
323
+ suppressed" instead of getting them individually.
324
+ - When the cooldown ends, emit a release summary and reset counters.
325
+ """
326
+ with self._global_watch_lock:
327
+ # Handle cooldown expiry first so we can emit the release summary.
328
+ if self._global_watch_tripped_until and now >= self._global_watch_tripped_until:
329
+ suppressed = self._global_watch_suppressed_during_trip
330
+ self._global_watch_tripped_until = 0.0
331
+ self._global_watch_suppressed_during_trip = 0
332
+ self._global_watch_window_start = now
333
+ self._global_watch_window_hits = 0
334
+ if suppressed > 0:
335
+ # Queue a summary event outside the lock (below).
336
+ release_msg = {
337
+ "session_id": "",
338
+ "session_key": "",
339
+ "command": "",
340
+ "type": "watch_overflow_released",
341
+ "suppressed": suppressed,
342
+ "message": (
343
+ f"Watch-pattern notifications resumed. "
344
+ f"{suppressed} match event(s) were suppressed during the flood."
345
+ ),
346
+ "platform": "",
347
+ "chat_id": "",
348
+ "user_id": "",
349
+ "user_name": "",
350
+ "thread_id": "",
351
+ }
352
+ else:
353
+ release_msg = None
354
+ else:
355
+ release_msg = None
356
+
357
+ # Still in cooldown — drop and count.
358
+ if self._global_watch_tripped_until and now < self._global_watch_tripped_until:
359
+ self._global_watch_suppressed_during_trip += 1
360
+ admit = False
361
+ trip_now = None
362
+ else:
363
+ # Slide the window.
364
+ if now - self._global_watch_window_start >= WATCH_GLOBAL_WINDOW_SECONDS:
365
+ self._global_watch_window_start = now
366
+ self._global_watch_window_hits = 0
367
+
368
+ if self._global_watch_window_hits >= WATCH_GLOBAL_MAX_PER_WINDOW:
369
+ # Trip the breaker.
370
+ self._global_watch_tripped_until = now + WATCH_GLOBAL_COOLDOWN_SECONDS
371
+ self._global_watch_suppressed_during_trip += 1
372
+ trip_now = now
373
+ admit = False
374
+ else:
375
+ self._global_watch_window_hits += 1
376
+ trip_now = None
377
+ admit = True
378
+
379
+ # Queue summary events outside the lock.
380
+ if release_msg is not None:
381
+ self.completion_queue.put(release_msg)
382
+ if trip_now is not None:
383
+ self.completion_queue.put({
384
+ "session_id": "",
385
+ "session_key": "",
386
+ "command": "",
387
+ "type": "watch_overflow_tripped",
388
+ "message": (
389
+ f"Watch-pattern overflow: >{WATCH_GLOBAL_MAX_PER_WINDOW} "
390
+ f"notifications in {WATCH_GLOBAL_WINDOW_SECONDS}s across all processes. "
391
+ f"Suppressing further watch_match events for "
392
+ f"{WATCH_GLOBAL_COOLDOWN_SECONDS}s."
393
+ ),
394
+ "platform": "",
395
+ "chat_id": "",
396
+ "user_id": "",
397
+ "user_name": "",
398
+ "thread_id": "",
399
+ })
400
+ return admit
401
+
402
+ @staticmethod
403
+ def _is_host_pid_alive(pid: Optional[int]) -> bool:
404
+ """Best-effort liveness check for host-visible PIDs."""
405
+ if not pid:
406
+ return False
407
+ # ``os.kill(pid, 0)`` is NOT a no-op on Windows (bpo-14484) — use
408
+ # the cross-platform existence check.
409
+ from gateway.status import _pid_exists
410
+ return _pid_exists(pid)
411
+
412
+ def _refresh_detached_session(self, session: Optional[ProcessSession]) -> Optional[ProcessSession]:
413
+ """Update recovered host-PID sessions when the underlying process has exited."""
414
+ if session is None or session.exited or not session.detached or session.pid_scope != "host":
415
+ return session
416
+
417
+ if self._is_host_pid_alive(session.pid):
418
+ return session
419
+
420
+ with session._lock:
421
+ if session.exited:
422
+ return session
423
+ session.exited = True
424
+ # Recovered sessions no longer have a waitable handle, so the real
425
+ # exit code is unavailable once the original process object is gone.
426
+ session.exit_code = None
427
+
428
+ self._move_to_finished(session)
429
+ return session
430
+
431
+ @staticmethod
432
+ def _terminate_host_pid(pid: int) -> None:
433
+ """Terminate a host-visible PID without requiring the original process handle."""
434
+ if _IS_WINDOWS:
435
+ os.kill(pid, signal.SIGTERM)
436
+ return
437
+
438
+ import psutil
439
+ try:
440
+ parent = psutil.Process(pid)
441
+ for child in parent.children(recursive=True):
442
+ try:
443
+ child.terminate()
444
+ except psutil.NoSuchProcess:
445
+ pass
446
+ parent.terminate()
447
+ except psutil.NoSuchProcess:
448
+ return
449
+ except (OSError, PermissionError):
450
+ try:
451
+ os.kill(pid, signal.SIGTERM)
452
+ except (OSError, ProcessLookupError, PermissionError):
453
+ pass
454
+
455
+ # ----- Spawn -----
456
+
457
+ @staticmethod
458
+ def _env_temp_dir(env: Any) -> str:
459
+ """Return the writable sandbox temp dir for env-backed background tasks."""
460
+ get_temp_dir = getattr(env, "get_temp_dir", None)
461
+ if callable(get_temp_dir):
462
+ try:
463
+ temp_dir = get_temp_dir()
464
+ if isinstance(temp_dir, str) and temp_dir.startswith("/"):
465
+ return temp_dir.rstrip("/") or "/"
466
+ except Exception as exc:
467
+ logger.debug("Could not resolve environment temp dir: %s", exc)
468
+ return "/tmp"
469
+
470
+ def spawn_local(
471
+ self,
472
+ command: str,
473
+ cwd: str = None,
474
+ task_id: str = "",
475
+ session_key: str = "",
476
+ env_vars: dict = None,
477
+ use_pty: bool = False,
478
+ ) -> ProcessSession:
479
+ """
480
+ Spawn a background process locally.
481
+
482
+ Only for TERMINAL_ENV=local. Other backends use spawn_via_env().
483
+
484
+ Args:
485
+ use_pty: If True, use a pseudo-terminal via ptyprocess for interactive
486
+ CLI tools (Codex, Claude Code, Python REPL). Falls back to
487
+ subprocess.Popen if ptyprocess is not installed.
488
+ """
489
+ session = ProcessSession(
490
+ id=f"proc_{uuid.uuid4().hex[:12]}",
491
+ command=command,
492
+ task_id=task_id,
493
+ session_key=session_key,
494
+ cwd=_resolve_safe_cwd(cwd or os.getcwd()),
495
+ started_at=time.time(),
496
+ )
497
+
498
+ if use_pty:
499
+ # Try PTY mode for interactive CLI tools
500
+ try:
501
+ if _IS_WINDOWS:
502
+ from winpty import PtyProcess as _PtyProcessCls
503
+ else:
504
+ from ptyprocess import PtyProcess as _PtyProcessCls
505
+ user_shell = _find_shell()
506
+ pty_env = _sanitize_subprocess_env(os.environ, env_vars)
507
+ pty_env["PYTHONUNBUFFERED"] = "1"
508
+ pty_proc = _PtyProcessCls.spawn(
509
+ [user_shell, "-lic", f"set +m; {command}"],
510
+ cwd=session.cwd,
511
+ env=pty_env,
512
+ dimensions=(30, 120),
513
+ )
514
+ session.pid = pty_proc.pid
515
+ # Store the pty handle on the session for read/write
516
+ session._pty = pty_proc
517
+
518
+ # PTY reader thread
519
+ reader = threading.Thread(
520
+ target=self._pty_reader_loop,
521
+ args=(session,),
522
+ daemon=True,
523
+ name=f"proc-pty-reader-{session.id}",
524
+ )
525
+ session._reader_thread = reader
526
+ reader.start()
527
+
528
+ with self._lock:
529
+ self._prune_if_needed()
530
+ self._running[session.id] = session
531
+
532
+ self._write_checkpoint()
533
+ return session
534
+
535
+ except ImportError:
536
+ logger.warning("ptyprocess not installed, falling back to pipe mode")
537
+ except Exception as e:
538
+ logger.warning("PTY spawn failed (%s), falling back to pipe mode", e)
539
+
540
+ # Standard Popen path (non-PTY or PTY fallback)
541
+ # Use the user's login shell for consistency with LocalEnvironment --
542
+ # ensures rc files are sourced and user tools are available.
543
+ user_shell = _find_shell()
544
+ # Force unbuffered output for Python scripts so progress is visible
545
+ # during background execution (libraries like tqdm/datasets buffer when
546
+ # stdout is a pipe, hiding output from process(action="poll")).
547
+ bg_env = _sanitize_subprocess_env(os.environ, env_vars)
548
+ bg_env["PYTHONUNBUFFERED"] = "1"
549
+ proc = subprocess.Popen(
550
+ [user_shell, "-lic", f"set +m; {command}"],
551
+ text=True,
552
+ cwd=session.cwd,
553
+ env=bg_env,
554
+ encoding="utf-8",
555
+ errors="replace",
556
+ stdout=subprocess.PIPE,
557
+ stderr=subprocess.STDOUT,
558
+ stdin=subprocess.PIPE,
559
+ preexec_fn=None if _IS_WINDOWS else os.setsid,
560
+ )
561
+
562
+ session.process = proc
563
+ session.pid = proc.pid
564
+
565
+ try:
566
+ # Start output reader thread
567
+ reader = threading.Thread(
568
+ target=self._reader_loop,
569
+ args=(session,),
570
+ daemon=True,
571
+ name=f"proc-reader-{session.id}",
572
+ )
573
+ session._reader_thread = reader
574
+ reader.start()
575
+
576
+ with self._lock:
577
+ self._prune_if_needed()
578
+ self._running[session.id] = session
579
+
580
+ self._write_checkpoint()
581
+ except Exception:
582
+ # Post-Popen setup failed — kill the orphaned subprocess (and any
583
+ # descendants spawned via setsid) before re-raising so they do not
584
+ # leak as untracked background processes.
585
+ try:
586
+ if not _IS_WINDOWS:
587
+ try:
588
+ os.killpg(os.getpgid(proc.pid), signal.SIGKILL) # windows-footgun: ok — guarded by _IS_WINDOWS check above
589
+ except (ProcessLookupError, PermissionError, OSError):
590
+ proc.kill()
591
+ else:
592
+ proc.kill()
593
+ except Exception:
594
+ pass
595
+ try:
596
+ proc.wait(timeout=5)
597
+ except Exception:
598
+ pass
599
+ raise
600
+
601
+ return session
602
+
603
+ def spawn_via_env(
604
+ self,
605
+ env: Any,
606
+ command: str,
607
+ cwd: str = None,
608
+ task_id: str = "",
609
+ session_key: str = "",
610
+ timeout: int = 10,
611
+ ) -> ProcessSession:
612
+ """
613
+ Spawn a background process through a non-local environment backend.
614
+
615
+ For Docker/Singularity/Modal/Daytona/SSH: runs the command inside the sandbox
616
+ using the environment's execute() interface. We wrap the command to
617
+ capture the in-sandbox PID and redirect output to a log file inside
618
+ the sandbox, then poll the log via subsequent execute() calls.
619
+
620
+ This is less capable than local spawn (no live stdout pipe, no stdin),
621
+ but it ensures the command runs in the correct sandbox context.
622
+ """
623
+ session = ProcessSession(
624
+ id=f"proc_{uuid.uuid4().hex[:12]}",
625
+ command=command,
626
+ task_id=task_id,
627
+ session_key=session_key,
628
+ cwd=cwd,
629
+ started_at=time.time(),
630
+ env_ref=env,
631
+ pid_scope="sandbox",
632
+ )
633
+
634
+ # Run the command in the sandbox with output capture
635
+ temp_dir = self._env_temp_dir(env)
636
+ log_path = f"{temp_dir}/hermes_bg_{session.id}.log"
637
+ pid_path = f"{temp_dir}/hermes_bg_{session.id}.pid"
638
+ exit_path = f"{temp_dir}/hermes_bg_{session.id}.exit"
639
+ quoted_command = shlex.quote(command)
640
+ quoted_temp_dir = shlex.quote(temp_dir)
641
+ quoted_log_path = shlex.quote(log_path)
642
+ quoted_pid_path = shlex.quote(pid_path)
643
+ quoted_exit_path = shlex.quote(exit_path)
644
+ bg_command = (
645
+ f"mkdir -p {quoted_temp_dir} && "
646
+ f"( nohup bash -lc {quoted_command} > {quoted_log_path} 2>&1; "
647
+ f"rc=$?; printf '%s\\n' \"$rc\" > {quoted_exit_path} ) & "
648
+ f"echo $! > {quoted_pid_path} && cat {quoted_pid_path}"
649
+ )
650
+
651
+ try:
652
+ result = env.execute(bg_command, timeout=timeout)
653
+ output = result.get("output", "").strip()
654
+ # Try to extract the PID from the output
655
+ for line in output.splitlines():
656
+ line = line.strip()
657
+ if line.isdigit():
658
+ session.pid = int(line)
659
+ break
660
+ except Exception as e:
661
+ session.exited = True
662
+ session.exit_code = -1
663
+ session.output_buffer = f"Failed to start: {e}"
664
+
665
+ if not session.exited:
666
+ # Start a poller thread that periodically reads the log file
667
+ reader = threading.Thread(
668
+ target=self._env_poller_loop,
669
+ args=(session, env, log_path, pid_path, exit_path),
670
+ daemon=True,
671
+ name=f"proc-poller-{session.id}",
672
+ )
673
+ session._reader_thread = reader
674
+ reader.start()
675
+
676
+ with self._lock:
677
+ self._prune_if_needed()
678
+ self._running[session.id] = session
679
+
680
+ self._write_checkpoint()
681
+ return session
682
+
683
+ # ----- Reader / Poller Threads -----
684
+
685
+ def _reader_loop(self, session: ProcessSession):
686
+ """Background thread: read stdout from a local Popen process."""
687
+ first_chunk = True
688
+ try:
689
+ while True:
690
+ chunk = session.process.stdout.read(4096)
691
+ if not chunk:
692
+ break
693
+ if first_chunk:
694
+ chunk = self._clean_shell_noise(chunk)
695
+ first_chunk = False
696
+ with session._lock:
697
+ session.output_buffer += chunk
698
+ if len(session.output_buffer) > session.max_output_chars:
699
+ session.output_buffer = session.output_buffer[-session.max_output_chars:]
700
+ self._check_watch_patterns(session, chunk)
701
+ except Exception as e:
702
+ logger.debug("Process stdout reader ended: %s", e)
703
+ finally:
704
+ # Always reap the child to prevent zombie processes.
705
+ try:
706
+ session.process.wait(timeout=5)
707
+ except Exception as e:
708
+ logger.debug("Process wait timed out or failed: %s", e)
709
+ session.exited = True
710
+ session.exit_code = session.process.returncode
711
+ self._move_to_finished(session)
712
+
713
+ def _env_poller_loop(
714
+ self, session: ProcessSession, env: Any, log_path: str, pid_path: str, exit_path: str
715
+ ):
716
+ """Background thread: poll a sandbox log file for non-local backends."""
717
+ quoted_log_path = shlex.quote(log_path)
718
+ quoted_pid_path = shlex.quote(pid_path)
719
+ quoted_exit_path = shlex.quote(exit_path)
720
+ prev_output_len = 0 # track delta for watch pattern scanning
721
+ while not session.exited:
722
+ time.sleep(2) # Poll every 2 seconds
723
+ try:
724
+ # Read new output from the log file
725
+ result = env.execute(f"cat {quoted_log_path} 2>/dev/null", timeout=10)
726
+ new_output = result.get("output", "")
727
+ if new_output:
728
+ # Compute delta for watch pattern scanning
729
+ delta = new_output[prev_output_len:] if len(new_output) > prev_output_len else ""
730
+ prev_output_len = len(new_output)
731
+ with session._lock:
732
+ session.output_buffer = new_output
733
+ if len(session.output_buffer) > session.max_output_chars:
734
+ session.output_buffer = session.output_buffer[-session.max_output_chars:]
735
+ if delta:
736
+ self._check_watch_patterns(session, delta)
737
+
738
+ # Check if process is still running
739
+ check = env.execute(
740
+ f"kill -0 \"$(cat {quoted_pid_path} 2>/dev/null)\" 2>/dev/null; echo $?",
741
+ timeout=5,
742
+ )
743
+ check_output = check.get("output", "").strip()
744
+ if check_output and check_output.splitlines()[-1].strip() != "0":
745
+ # Process has exited -- get exit code captured by the wrapper shell.
746
+ exit_result = env.execute(
747
+ f"cat {quoted_exit_path} 2>/dev/null",
748
+ timeout=5,
749
+ )
750
+ exit_str = exit_result.get("output", "").strip()
751
+ try:
752
+ session.exit_code = int(exit_str.splitlines()[-1].strip())
753
+ except (ValueError, IndexError):
754
+ session.exit_code = -1
755
+ session.exited = True
756
+ self._move_to_finished(session)
757
+ return
758
+
759
+ except Exception:
760
+ # Environment might be gone (sandbox reaped, etc.)
761
+ session.exited = True
762
+ session.exit_code = -1
763
+ self._move_to_finished(session)
764
+ return
765
+
766
+ def _pty_reader_loop(self, session: ProcessSession):
767
+ """Background thread: read output from a PTY process."""
768
+ pty = session._pty
769
+ try:
770
+ while pty.isalive():
771
+ try:
772
+ chunk = pty.read(4096)
773
+ if chunk:
774
+ # ptyprocess returns bytes
775
+ text = chunk if isinstance(chunk, str) else chunk.decode("utf-8", errors="replace")
776
+ with session._lock:
777
+ session.output_buffer += text
778
+ if len(session.output_buffer) > session.max_output_chars:
779
+ session.output_buffer = session.output_buffer[-session.max_output_chars:]
780
+ self._check_watch_patterns(session, text)
781
+ except EOFError:
782
+ break
783
+ except Exception:
784
+ break
785
+ except Exception as e:
786
+ logger.debug("PTY stdout reader ended: %s", e)
787
+
788
+ # Process exited
789
+ try:
790
+ pty.wait()
791
+ except Exception as e:
792
+ logger.debug("PTY wait timed out or failed: %s", e)
793
+ session.exited = True
794
+ session.exit_code = pty.exitstatus if hasattr(pty, 'exitstatus') else -1
795
+ self._move_to_finished(session)
796
+
797
+ def _move_to_finished(self, session: ProcessSession):
798
+ """Move a session from running to finished.
799
+
800
+ Idempotent: if the session was already moved (e.g. kill_process raced
801
+ with the reader thread), the second call is a no-op — no duplicate
802
+ completion notification is enqueued.
803
+ """
804
+ with self._lock:
805
+ was_running = self._running.pop(session.id, None) is not None
806
+ self._finished[session.id] = session
807
+ self._write_checkpoint()
808
+
809
+ # Only enqueue completion notification on the FIRST move. Without
810
+ # this guard, kill_process() and the reader thread can both call
811
+ # _move_to_finished(), producing duplicate [IMPORTANT: ...] messages.
812
+ if was_running and session.notify_on_complete:
813
+ from tools.ansi_strip import strip_ansi
814
+ output_tail = strip_ansi(session.output_buffer[-2000:]) if session.output_buffer else ""
815
+ self.completion_queue.put({
816
+ "type": "completion",
817
+ "session_id": session.id,
818
+ "command": session.command,
819
+ "exit_code": session.exit_code,
820
+ "output": output_tail,
821
+ })
822
+
823
+ # ----- Query Methods -----
824
+
825
+ def is_completion_consumed(self, session_id: str) -> bool:
826
+ """Check if a completion notification was already consumed via wait/poll/log."""
827
+ return session_id in self._completion_consumed
828
+
829
+ def drain_notifications(self) -> "list[tuple[dict, str]]":
830
+ """Pop all pending notification events and return formatted pairs.
831
+
832
+ Returns a list of (raw_event, formatted_text) tuples.
833
+ Skips completion events that were already consumed via wait/poll/log.
834
+ """
835
+ results = []
836
+ while not self.completion_queue.empty():
837
+ try:
838
+ evt = self.completion_queue.get_nowait()
839
+ except Exception:
840
+ break
841
+ _evt_sid = evt.get("session_id", "")
842
+ if evt.get("type") == "completion" and self.is_completion_consumed(_evt_sid):
843
+ continue
844
+ text = format_process_notification(evt)
845
+ if text:
846
+ results.append((evt, text))
847
+ return results
848
+
849
+ def get(self, session_id: str) -> Optional[ProcessSession]:
850
+ """Get a session by ID (running or finished)."""
851
+ with self._lock:
852
+ session = self._running.get(session_id) or self._finished.get(session_id)
853
+ return self._refresh_detached_session(session)
854
+
855
+ def _reconcile_local_exit(self, session: "ProcessSession") -> None:
856
+ """Reconcile session.exited against the real child process state.
857
+
858
+ The reader thread (`_reader_loop`) sets `session.exited = True` only
859
+ in its `finally` block, which runs when `stdout.read()` returns EOF.
860
+ If the direct `Popen` child has exited but a descendant process (e.g.
861
+ a daemon spawned by `hermes update` restarting the gateway) is still
862
+ holding the stdout pipe open, the reader blocks forever and poll()
863
+ keeps returning "running" indefinitely (issue #17327 — 74 polls over
864
+ 7 minutes on Feishu).
865
+
866
+ This helper closes that window: when `session.exited` is still False
867
+ but the direct child's `Popen.poll()` reports an exit code, drain any
868
+ readable bytes non-blocking and flip `session.exited`. The orphaned
869
+ reader thread remains stuck on its blocking `read()` but is a daemon
870
+ thread and will be reaped with the process.
871
+
872
+ Safe no-op on sessions without a local `Popen` (env/PTY), already-
873
+ exited sessions, and detached-recovered sessions.
874
+ """
875
+ if session is None or session.exited:
876
+ return
877
+ proc = getattr(session, "process", None)
878
+ if proc is None:
879
+ return
880
+ try:
881
+ rc = proc.poll()
882
+ except Exception:
883
+ return
884
+ if rc is None:
885
+ return # Direct child still running — reader block is legitimate.
886
+
887
+ # Direct child exited. Try to drain any bytes the reader hasn't
888
+ # consumed yet. This is best-effort: if the pipe is held open by a
889
+ # descendant, the non-blocking read returns what's immediately
890
+ # available and we stop.
891
+ drained = ""
892
+ stdout = getattr(proc, "stdout", None)
893
+ if stdout is not None and not _IS_WINDOWS:
894
+ try:
895
+ import fcntl
896
+ fd = stdout.fileno()
897
+ flags = fcntl.fcntl(fd, fcntl.F_GETFL)
898
+ fcntl.fcntl(fd, fcntl.F_SETFL, flags | os.O_NONBLOCK)
899
+ try:
900
+ chunk = stdout.read()
901
+ if chunk:
902
+ drained = chunk if isinstance(chunk, str) else chunk.decode("utf-8", errors="replace")
903
+ except (BlockingIOError, OSError, ValueError):
904
+ pass
905
+ finally:
906
+ try:
907
+ fcntl.fcntl(fd, fcntl.F_SETFL, flags)
908
+ except Exception:
909
+ pass
910
+ except Exception as e:
911
+ logger.debug("Non-blocking drain failed for %s: %s", session.id, e)
912
+
913
+ with session._lock:
914
+ if drained:
915
+ session.output_buffer += drained
916
+ if len(session.output_buffer) > session.max_output_chars:
917
+ session.output_buffer = session.output_buffer[-session.max_output_chars:]
918
+ session.exited = True
919
+ session.exit_code = rc
920
+ logger.info(
921
+ "Reconciled session %s: direct child exited with code %s but reader "
922
+ "was still blocked (orphaned pipe). Flipped to exited.",
923
+ session.id, rc,
924
+ )
925
+ self._move_to_finished(session)
926
+
927
+ def poll(self, session_id: str) -> dict:
928
+ """Check status and get new output for a background process."""
929
+ from tools.ansi_strip import strip_ansi
930
+
931
+ session = self.get(session_id)
932
+ if session is None:
933
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
934
+
935
+ # Reconcile against real child state before reading session.exited.
936
+ # Guards against orphaned-pipe reader hangs (issue #17327).
937
+ self._reconcile_local_exit(session)
938
+
939
+ with session._lock:
940
+ output_preview = strip_ansi(session.output_buffer[-1000:]) if session.output_buffer else ""
941
+
942
+ result = {
943
+ "session_id": session.id,
944
+ "command": session.command,
945
+ "status": "exited" if session.exited else "running",
946
+ "pid": session.pid,
947
+ "uptime_seconds": int(time.time() - session.started_at),
948
+ "output_preview": output_preview,
949
+ }
950
+ if session.exited:
951
+ result["exit_code"] = session.exit_code
952
+ self._completion_consumed.add(session_id)
953
+ if session.detached:
954
+ result["detached"] = True
955
+ result["note"] = "Process recovered after restart -- output history unavailable"
956
+ return result
957
+
958
+ def read_log(self, session_id: str, offset: int = 0, limit: int = 200) -> dict:
959
+ """Read the full output log with optional pagination by lines."""
960
+ from tools.ansi_strip import strip_ansi
961
+
962
+ session = self.get(session_id)
963
+ if session is None:
964
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
965
+
966
+ with session._lock:
967
+ full_output = strip_ansi(session.output_buffer)
968
+
969
+ lines = full_output.splitlines()
970
+ total_lines = len(lines)
971
+
972
+ # Default: last N lines
973
+ if offset == 0 and limit > 0:
974
+ selected = lines[-limit:]
975
+ else:
976
+ selected = lines[offset:offset + limit]
977
+
978
+ result = {
979
+ "session_id": session.id,
980
+ "status": "exited" if session.exited else "running",
981
+ "output": "\n".join(selected),
982
+ "total_lines": total_lines,
983
+ "showing": f"{len(selected)} lines",
984
+ }
985
+ if session.exited:
986
+ self._completion_consumed.add(session_id)
987
+ return result
988
+
989
+ def wait(self, session_id: str, timeout: int = None) -> dict:
990
+ """
991
+ Block until a process exits, timeout, or interrupt.
992
+
993
+ Args:
994
+ session_id: The process to wait for.
995
+ timeout: Max seconds to block. Falls back to TERMINAL_TIMEOUT config.
996
+
997
+ Returns:
998
+ dict with status ("exited", "timeout", "interrupted", "not_found")
999
+ and output snapshot.
1000
+ """
1001
+ from tools.ansi_strip import strip_ansi
1002
+ from tools.interrupt import is_interrupted as _is_interrupted
1003
+
1004
+ try:
1005
+ default_timeout = int(os.getenv("TERMINAL_TIMEOUT", "180"))
1006
+ except (ValueError, TypeError):
1007
+ default_timeout = 180
1008
+ max_timeout = default_timeout
1009
+ requested_timeout = timeout
1010
+ timeout_note = None
1011
+
1012
+ if requested_timeout and requested_timeout > max_timeout:
1013
+ effective_timeout = max_timeout
1014
+ timeout_note = (
1015
+ f"Requested wait of {requested_timeout}s was clamped "
1016
+ f"to configured limit of {max_timeout}s"
1017
+ )
1018
+ else:
1019
+ effective_timeout = requested_timeout or max_timeout
1020
+
1021
+ session = self.get(session_id)
1022
+ if session is None:
1023
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
1024
+
1025
+ deadline = time.monotonic() + effective_timeout
1026
+
1027
+ while time.monotonic() < deadline:
1028
+ session = self._refresh_detached_session(session)
1029
+ # Reconcile against real child state — guards against orphaned-
1030
+ # pipe reader hangs where the reader is blocked but the direct
1031
+ # child has already exited (issue #17327).
1032
+ self._reconcile_local_exit(session)
1033
+ if session.exited:
1034
+ self._completion_consumed.add(session_id)
1035
+ result = {
1036
+ "status": "exited",
1037
+ "exit_code": session.exit_code,
1038
+ "output": strip_ansi(session.output_buffer[-2000:]),
1039
+ }
1040
+ if timeout_note:
1041
+ result["timeout_note"] = timeout_note
1042
+ return result
1043
+
1044
+ if _is_interrupted():
1045
+ result = {
1046
+ "status": "interrupted",
1047
+ "output": strip_ansi(session.output_buffer[-1000:]),
1048
+ "note": "User sent a new message -- wait interrupted",
1049
+ }
1050
+ if timeout_note:
1051
+ result["timeout_note"] = timeout_note
1052
+ return result
1053
+
1054
+ time.sleep(1)
1055
+
1056
+ result = {
1057
+ "status": "timeout",
1058
+ "output": strip_ansi(session.output_buffer[-1000:]),
1059
+ }
1060
+ if timeout_note:
1061
+ result["timeout_note"] = timeout_note
1062
+ else:
1063
+ result["timeout_note"] = f"Waited {effective_timeout}s, process still running"
1064
+ return result
1065
+
1066
+ def kill_process(self, session_id: str) -> dict:
1067
+ """Kill a background process."""
1068
+ session = self.get(session_id)
1069
+ if session is None:
1070
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
1071
+
1072
+ if session.exited:
1073
+ return {
1074
+ "status": "already_exited",
1075
+ "exit_code": session.exit_code,
1076
+ }
1077
+
1078
+ # Kill via PTY, Popen (local), or env execute (non-local)
1079
+ try:
1080
+ if session._pty:
1081
+ # PTY process -- terminate via ptyprocess
1082
+ try:
1083
+ session._pty.terminate(force=True)
1084
+ except Exception:
1085
+ if session.pid:
1086
+ os.kill(session.pid, signal.SIGTERM)
1087
+ elif session.process:
1088
+ # Local process -- kill the process tree
1089
+ try:
1090
+ if _IS_WINDOWS:
1091
+ session.process.terminate()
1092
+ else:
1093
+ import psutil
1094
+ try:
1095
+ parent = psutil.Process(session.process.pid)
1096
+ for child in parent.children(recursive=True):
1097
+ try:
1098
+ child.terminate()
1099
+ except psutil.NoSuchProcess:
1100
+ pass
1101
+ parent.terminate()
1102
+ except psutil.NoSuchProcess:
1103
+ pass
1104
+ except (ProcessLookupError, PermissionError):
1105
+ session.process.kill()
1106
+ elif session.env_ref and session.pid:
1107
+ # Non-local -- kill inside sandbox
1108
+ session.env_ref.execute(f"kill {session.pid} 2>/dev/null", timeout=5)
1109
+ elif session.detached and session.pid_scope == "host" and session.pid:
1110
+ if not self._is_host_pid_alive(session.pid):
1111
+ with session._lock:
1112
+ session.exited = True
1113
+ session.exit_code = None
1114
+ self._move_to_finished(session)
1115
+ return {
1116
+ "status": "already_exited",
1117
+ "exit_code": session.exit_code,
1118
+ }
1119
+ self._terminate_host_pid(session.pid)
1120
+ else:
1121
+ return {
1122
+ "status": "error",
1123
+ "error": (
1124
+ "Recovered process cannot be killed after restart because "
1125
+ "its original runtime handle is no longer available"
1126
+ ),
1127
+ }
1128
+ session.exited = True
1129
+ session.exit_code = -15 # SIGTERM
1130
+ self._move_to_finished(session)
1131
+ self._write_checkpoint()
1132
+ return {"status": "killed", "session_id": session.id}
1133
+ except Exception as e:
1134
+ return {"status": "error", "error": str(e)}
1135
+
1136
+ def write_stdin(self, session_id: str, data: str) -> dict:
1137
+ """Send raw data to a running process's stdin (no newline appended)."""
1138
+ session = self.get(session_id)
1139
+ if session is None:
1140
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
1141
+ if session.exited:
1142
+ return {"status": "already_exited", "error": "Process has already finished"}
1143
+
1144
+ # PTY mode -- write through pty handle (expects bytes)
1145
+ if hasattr(session, '_pty') and session._pty:
1146
+ try:
1147
+ pty_data = data.encode("utf-8") if isinstance(data, str) else data
1148
+ session._pty.write(pty_data)
1149
+ return {"status": "ok", "bytes_written": len(data)}
1150
+ except Exception as e:
1151
+ return {"status": "error", "error": str(e)}
1152
+
1153
+ # Popen mode -- write through stdin pipe
1154
+ if not session.process or not session.process.stdin:
1155
+ return {"status": "error", "error": "Process stdin not available (non-local backend or stdin closed)"}
1156
+ try:
1157
+ session.process.stdin.write(data)
1158
+ session.process.stdin.flush()
1159
+ return {"status": "ok", "bytes_written": len(data)}
1160
+ except Exception as e:
1161
+ return {"status": "error", "error": str(e)}
1162
+
1163
+ def submit_stdin(self, session_id: str, data: str = "") -> dict:
1164
+ """Send data + newline to a running process's stdin (like pressing Enter)."""
1165
+ return self.write_stdin(session_id, data + "\n")
1166
+
1167
+ def close_stdin(self, session_id: str) -> dict:
1168
+ """Close a running process's stdin / send EOF without killing the process."""
1169
+ session = self.get(session_id)
1170
+ if session is None:
1171
+ return {"status": "not_found", "error": f"No process with ID {session_id}"}
1172
+ if session.exited:
1173
+ return {"status": "already_exited", "error": "Process has already finished"}
1174
+
1175
+ if hasattr(session, '_pty') and session._pty:
1176
+ try:
1177
+ session._pty.sendeof()
1178
+ return {"status": "ok", "message": "EOF sent"}
1179
+ except Exception as e:
1180
+ return {"status": "error", "error": str(e)}
1181
+
1182
+ if not session.process or not session.process.stdin:
1183
+ return {"status": "error", "error": "Process stdin not available (non-local backend or stdin closed)"}
1184
+ try:
1185
+ session.process.stdin.close()
1186
+ return {"status": "ok", "message": "stdin closed"}
1187
+ except Exception as e:
1188
+ return {"status": "error", "error": str(e)}
1189
+
1190
+ def list_sessions(self, task_id: str = None) -> list:
1191
+ """List all running and recently-finished processes."""
1192
+ with self._lock:
1193
+ all_sessions = list(self._running.values()) + list(self._finished.values())
1194
+
1195
+ all_sessions = [self._refresh_detached_session(s) for s in all_sessions]
1196
+
1197
+ if task_id:
1198
+ all_sessions = [s for s in all_sessions if s.task_id == task_id]
1199
+
1200
+ result = []
1201
+ for s in all_sessions:
1202
+ entry = {
1203
+ "session_id": s.id,
1204
+ "command": s.command[:200],
1205
+ "cwd": s.cwd,
1206
+ "pid": s.pid,
1207
+ "started_at": time.strftime("%Y-%m-%dT%H:%M:%S", time.localtime(s.started_at)),
1208
+ "uptime_seconds": int(time.time() - s.started_at),
1209
+ "status": "exited" if s.exited else "running",
1210
+ "output_preview": s.output_buffer[-200:] if s.output_buffer else "",
1211
+ }
1212
+ if s.exited:
1213
+ entry["exit_code"] = s.exit_code
1214
+ if s.detached:
1215
+ entry["detached"] = True
1216
+ result.append(entry)
1217
+ return result
1218
+
1219
+ # ----- Session/Task Queries (for gateway integration) -----
1220
+
1221
+ def has_active_processes(self, task_id: str) -> bool:
1222
+ """Check if there are active (running) processes for a task_id."""
1223
+ with self._lock:
1224
+ sessions = list(self._running.values())
1225
+
1226
+ for session in sessions:
1227
+ self._refresh_detached_session(session)
1228
+
1229
+ with self._lock:
1230
+ return any(
1231
+ s.task_id == task_id and not s.exited
1232
+ for s in self._running.values()
1233
+ )
1234
+
1235
+ def has_active_for_session(self, session_key: str) -> bool:
1236
+ """Check if there are active processes for a gateway session key."""
1237
+ with self._lock:
1238
+ sessions = list(self._running.values())
1239
+
1240
+ for session in sessions:
1241
+ self._refresh_detached_session(session)
1242
+
1243
+ with self._lock:
1244
+ return any(
1245
+ s.session_key == session_key and not s.exited
1246
+ for s in self._running.values()
1247
+ )
1248
+
1249
+ def kill_all(self, task_id: str = None) -> int:
1250
+ """Kill all running processes, optionally filtered by task_id. Returns count killed."""
1251
+ with self._lock:
1252
+ targets = [
1253
+ s for s in self._running.values()
1254
+ if (task_id is None or s.task_id == task_id) and not s.exited
1255
+ ]
1256
+
1257
+ killed = 0
1258
+ for session in targets:
1259
+ result = self.kill_process(session.id)
1260
+ if result.get("status") in {"killed", "already_exited"}:
1261
+ killed += 1
1262
+ return killed
1263
+
1264
+ # ----- Cleanup / Pruning -----
1265
+
1266
+ def _prune_if_needed(self):
1267
+ """Remove oldest finished sessions if over MAX_PROCESSES. Must hold _lock."""
1268
+ # First prune expired finished sessions
1269
+ now = time.time()
1270
+ expired = [
1271
+ sid for sid, s in self._finished.items()
1272
+ if (now - s.started_at) > FINISHED_TTL_SECONDS
1273
+ ]
1274
+ for sid in expired:
1275
+ del self._finished[sid]
1276
+ self._completion_consumed.discard(sid)
1277
+
1278
+ # If still over limit, remove oldest finished
1279
+ total = len(self._running) + len(self._finished)
1280
+ if total >= MAX_PROCESSES and self._finished:
1281
+ oldest_id = min(self._finished, key=lambda sid: self._finished[sid].started_at)
1282
+ del self._finished[oldest_id]
1283
+ self._completion_consumed.discard(oldest_id)
1284
+
1285
+ # Drop any _completion_consumed entries whose sessions are no longer
1286
+ # tracked at all — belt-and-suspenders against module-lifetime growth
1287
+ # on process-registry lookup paths that don't reach the dict prunes.
1288
+ tracked = self._running.keys() | self._finished.keys()
1289
+ stale = self._completion_consumed - tracked
1290
+ if stale:
1291
+ self._completion_consumed -= stale
1292
+
1293
+ # ----- Checkpoint (crash recovery) -----
1294
+
1295
+ def _write_checkpoint(self):
1296
+ """Write running process metadata to checkpoint file atomically."""
1297
+ try:
1298
+ with self._lock:
1299
+ entries = []
1300
+ for s in self._running.values():
1301
+ if not s.exited:
1302
+ entries.append({
1303
+ "session_id": s.id,
1304
+ "command": s.command,
1305
+ "pid": s.pid,
1306
+ "pid_scope": s.pid_scope,
1307
+ "cwd": s.cwd,
1308
+ "started_at": s.started_at,
1309
+ "task_id": s.task_id,
1310
+ "session_key": s.session_key,
1311
+ "watcher_platform": s.watcher_platform,
1312
+ "watcher_chat_id": s.watcher_chat_id,
1313
+ "watcher_user_id": s.watcher_user_id,
1314
+ "watcher_user_name": s.watcher_user_name,
1315
+ "watcher_thread_id": s.watcher_thread_id,
1316
+ "watcher_interval": s.watcher_interval,
1317
+ "notify_on_complete": s.notify_on_complete,
1318
+ "watch_patterns": s.watch_patterns,
1319
+ })
1320
+
1321
+ # Atomic write to avoid corruption on crash
1322
+ from utils import atomic_json_write
1323
+ atomic_json_write(CHECKPOINT_PATH, entries)
1324
+ except Exception as e:
1325
+ logger.debug("Failed to write checkpoint file: %s", e, exc_info=True)
1326
+
1327
+ def recover_from_checkpoint(self) -> int:
1328
+ """
1329
+ On gateway startup, probe PIDs from checkpoint file.
1330
+
1331
+ Returns the number of processes recovered as detached.
1332
+ """
1333
+ if not CHECKPOINT_PATH.exists():
1334
+ return 0
1335
+
1336
+ try:
1337
+ entries = json.loads(CHECKPOINT_PATH.read_text(encoding="utf-8"))
1338
+ except Exception:
1339
+ return 0
1340
+
1341
+ recovered = 0
1342
+ for entry in entries:
1343
+ pid = entry.get("pid")
1344
+ if not pid:
1345
+ continue
1346
+
1347
+ pid_scope = entry.get("pid_scope", "host")
1348
+ if pid_scope != "host":
1349
+ # Sandbox-backed processes keep only in-sandbox PIDs in the
1350
+ # checkpoint, which are not meaningful to the restarted host
1351
+ # process once the original environment handle is gone.
1352
+ logger.info(
1353
+ "Skipping recovery for non-host process: %s (pid=%s, scope=%s)",
1354
+ entry.get("command", "unknown")[:60],
1355
+ pid,
1356
+ pid_scope,
1357
+ )
1358
+ continue
1359
+
1360
+ # Check if PID is still alive
1361
+ alive = self._is_host_pid_alive(pid)
1362
+
1363
+ if alive:
1364
+ session = ProcessSession(
1365
+ id=entry["session_id"],
1366
+ command=entry.get("command", "unknown"),
1367
+ task_id=entry.get("task_id", ""),
1368
+ session_key=entry.get("session_key", ""),
1369
+ pid=pid,
1370
+ pid_scope=pid_scope,
1371
+ cwd=entry.get("cwd"),
1372
+ started_at=entry.get("started_at", time.time()),
1373
+ detached=True, # Can't read output, but can report status + kill
1374
+ watcher_platform=entry.get("watcher_platform", ""),
1375
+ watcher_chat_id=entry.get("watcher_chat_id", ""),
1376
+ watcher_user_id=entry.get("watcher_user_id", ""),
1377
+ watcher_user_name=entry.get("watcher_user_name", ""),
1378
+ watcher_thread_id=entry.get("watcher_thread_id", ""),
1379
+ watcher_interval=entry.get("watcher_interval", 0),
1380
+ notify_on_complete=entry.get("notify_on_complete", False),
1381
+ watch_patterns=entry.get("watch_patterns", []),
1382
+ )
1383
+ with self._lock:
1384
+ self._running[session.id] = session
1385
+ recovered += 1
1386
+ logger.info("Recovered detached process: %s (pid=%d)", session.command[:60], pid)
1387
+
1388
+ # Re-enqueue watcher so gateway can resume notifications
1389
+ if session.watcher_interval > 0:
1390
+ self.pending_watchers.append({
1391
+ "session_id": session.id,
1392
+ "check_interval": session.watcher_interval,
1393
+ "session_key": session.session_key,
1394
+ "platform": session.watcher_platform,
1395
+ "chat_id": session.watcher_chat_id,
1396
+ "user_id": session.watcher_user_id,
1397
+ "user_name": session.watcher_user_name,
1398
+ "thread_id": session.watcher_thread_id,
1399
+ "notify_on_complete": session.notify_on_complete,
1400
+ })
1401
+
1402
+ self._write_checkpoint()
1403
+
1404
+ return recovered
1405
+
1406
+
1407
+ # Module-level singleton
1408
+ process_registry = ProcessRegistry()
1409
+
1410
+
1411
+ def format_process_notification(evt: dict) -> "str | None":
1412
+ """Format a process notification event into a [IMPORTANT: ...] message.
1413
+
1414
+ Handles completion events (notify_on_complete), watch pattern matches,
1415
+ and watch disabled events from the unified completion_queue.
1416
+ """
1417
+ evt_type = evt.get("type", "completion")
1418
+ _sid = evt.get("session_id", "unknown")
1419
+ _cmd = evt.get("command", "unknown")
1420
+
1421
+ if evt_type == "watch_disabled":
1422
+ return f"[IMPORTANT: {evt.get('message', '')}]"
1423
+
1424
+ if evt_type == "watch_match":
1425
+ _pat = evt.get("pattern", "?")
1426
+ _out = evt.get("output", "")
1427
+ _sup = evt.get("suppressed", 0)
1428
+ text = (
1429
+ f"[IMPORTANT: Background process {_sid} matched "
1430
+ f"watch pattern \"{_pat}\".\n"
1431
+ f"Command: {_cmd}\n"
1432
+ f"Matched output:\n{_out}"
1433
+ )
1434
+ if _sup:
1435
+ text += f"\n({_sup} earlier matches were suppressed by rate limit)"
1436
+ text += "]"
1437
+ return text
1438
+
1439
+ _exit = evt.get("exit_code", "?")
1440
+ _out = evt.get("output", "")
1441
+ return (
1442
+ f"[IMPORTANT: Background process {_sid} completed "
1443
+ f"(exit code {_exit}).\n"
1444
+ f"Command: {_cmd}\n"
1445
+ f"Output:\n{_out}]"
1446
+ )
1447
+
1448
+
1449
+ # ---------------------------------------------------------------------------
1450
+ # Registry -- the "process" tool schema + handler
1451
+ # ---------------------------------------------------------------------------
1452
+ from tools.registry import registry, tool_error
1453
+
1454
+ PROCESS_SCHEMA = {
1455
+ "name": "process",
1456
+ "description": (
1457
+ "Manage background processes started with terminal(background=true). "
1458
+ "Actions: 'list' (show all), 'poll' (check status + new output), "
1459
+ "'log' (full output with pagination), 'wait' (block until done or timeout), "
1460
+ "'kill' (terminate), 'write' (send raw stdin data without newline), "
1461
+ "'submit' (send data + Enter, for answering prompts), 'close' (close stdin/send EOF)."
1462
+ ),
1463
+ "parameters": {
1464
+ "type": "object",
1465
+ "properties": {
1466
+ "action": {
1467
+ "type": "string",
1468
+ "enum": ["list", "poll", "log", "wait", "kill", "write", "submit", "close"],
1469
+ "description": "Action to perform on background processes"
1470
+ },
1471
+ "session_id": {
1472
+ "type": "string",
1473
+ "description": "Process session ID (from terminal background output). Required for all actions except 'list'."
1474
+ },
1475
+ "data": {
1476
+ "type": "string",
1477
+ "description": "Text to send to process stdin (for 'write' and 'submit' actions)"
1478
+ },
1479
+ "timeout": {
1480
+ "type": "integer",
1481
+ "description": "Max seconds to block for 'wait' action. Returns partial output on timeout.",
1482
+ "minimum": 1
1483
+ },
1484
+ "offset": {
1485
+ "type": "integer",
1486
+ "description": "Line offset for 'log' action (default: last 200 lines)"
1487
+ },
1488
+ "limit": {
1489
+ "type": "integer",
1490
+ "description": "Max lines to return for 'log' action",
1491
+ "minimum": 1
1492
+ }
1493
+ },
1494
+ "required": ["action"]
1495
+ }
1496
+ }
1497
+
1498
+
1499
+ def _handle_process(args, **kw):
1500
+ task_id = kw.get("task_id")
1501
+ action = args.get("action", "")
1502
+ # Coerce to string — some models send session_id as an integer
1503
+ session_id = str(args.get("session_id", "")) if args.get("session_id") is not None else ""
1504
+
1505
+ if action == "list":
1506
+ return json.dumps({"processes": process_registry.list_sessions(task_id=task_id)}, ensure_ascii=False)
1507
+ elif action in {"poll", "log", "wait", "kill", "write", "submit", "close"}:
1508
+ if not session_id:
1509
+ return tool_error(f"session_id is required for {action}")
1510
+ if action == "poll":
1511
+ return json.dumps(process_registry.poll(session_id), ensure_ascii=False)
1512
+ elif action == "log":
1513
+ return json.dumps(process_registry.read_log(
1514
+ session_id, offset=args.get("offset", 0), limit=args.get("limit", 200)), ensure_ascii=False)
1515
+ elif action == "wait":
1516
+ return json.dumps(process_registry.wait(session_id, timeout=args.get("timeout")), ensure_ascii=False)
1517
+ elif action == "kill":
1518
+ return json.dumps(process_registry.kill_process(session_id), ensure_ascii=False)
1519
+ elif action == "write":
1520
+ return json.dumps(process_registry.write_stdin(session_id, str(args.get("data", ""))), ensure_ascii=False)
1521
+ elif action == "submit":
1522
+ return json.dumps(process_registry.submit_stdin(session_id, str(args.get("data", ""))), ensure_ascii=False)
1523
+ elif action == "close":
1524
+ return json.dumps(process_registry.close_stdin(session_id), ensure_ascii=False)
1525
+ return tool_error(f"Unknown process action: {action}. Use: list, poll, log, wait, kill, write, submit, close")
1526
+
1527
+
1528
+ registry.register(
1529
+ name="process",
1530
+ toolset="terminal",
1531
+ schema=PROCESS_SCHEMA,
1532
+ handler=_handle_process,
1533
+ emoji="⚙️",
1534
+ )