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,1610 @@
1
+ """
2
+ WeCom (Enterprise WeChat) platform adapter.
3
+
4
+ Uses the WeCom AI Bot WebSocket gateway for inbound and outbound messages.
5
+ The adapter focuses on the core gateway path:
6
+
7
+ - authenticate via ``aibot_subscribe``
8
+ - receive inbound ``aibot_msg_callback`` events
9
+ - send outbound markdown messages via ``aibot_send_msg``
10
+ - upload outbound media via ``aibot_upload_media_*`` and send native attachments
11
+ - best-effort download of inbound image/file attachments for agent context
12
+
13
+ Configuration in config.yaml:
14
+ platforms:
15
+ wecom:
16
+ enabled: true
17
+ extra:
18
+ bot_id: "your-bot-id" # or WECOM_BOT_ID env var
19
+ secret: "your-secret" # or WECOM_SECRET env var
20
+ websocket_url: "wss://openws.work.weixin.qq.com"
21
+ dm_policy: "open" # open | allowlist | disabled | pairing
22
+ allow_from: ["user_id_1"]
23
+ group_policy: "open" # open | allowlist | disabled
24
+ group_allow_from: ["group_id_1"]
25
+ groups:
26
+ group_id_1:
27
+ allow_from: ["user_id_1"]
28
+ """
29
+
30
+ from __future__ import annotations
31
+
32
+ import asyncio
33
+ import base64
34
+ import hashlib
35
+ import json
36
+ import logging
37
+ import mimetypes
38
+ import os
39
+ import re
40
+ import time
41
+ import uuid
42
+ from datetime import datetime, timezone
43
+ from pathlib import Path
44
+ from typing import Any, Dict, List, Optional, Tuple
45
+ from urllib.parse import unquote, urlparse
46
+
47
+ try:
48
+ import aiohttp
49
+ AIOHTTP_AVAILABLE = True
50
+ except ImportError:
51
+ AIOHTTP_AVAILABLE = False
52
+ aiohttp = None # type: ignore[assignment]
53
+
54
+ try:
55
+ import httpx
56
+ HTTPX_AVAILABLE = True
57
+ except ImportError:
58
+ HTTPX_AVAILABLE = False
59
+ httpx = None # type: ignore[assignment]
60
+
61
+ from gateway.config import Platform, PlatformConfig
62
+ from gateway.platforms.helpers import MessageDeduplicator
63
+ from gateway.platforms.base import (
64
+ BasePlatformAdapter,
65
+ MessageEvent,
66
+ MessageType,
67
+ SendResult,
68
+ cache_document_from_bytes,
69
+ cache_image_from_bytes,
70
+ )
71
+
72
+ logger = logging.getLogger(__name__)
73
+
74
+ DEFAULT_WS_URL = "wss://openws.work.weixin.qq.com"
75
+
76
+ APP_CMD_SUBSCRIBE = "aibot_subscribe"
77
+ APP_CMD_CALLBACK = "aibot_msg_callback"
78
+ APP_CMD_LEGACY_CALLBACK = "aibot_callback"
79
+ APP_CMD_EVENT_CALLBACK = "aibot_event_callback"
80
+ APP_CMD_SEND = "aibot_send_msg"
81
+ APP_CMD_RESPONSE = "aibot_respond_msg"
82
+ APP_CMD_PING = "ping"
83
+ APP_CMD_UPLOAD_MEDIA_INIT = "aibot_upload_media_init"
84
+ APP_CMD_UPLOAD_MEDIA_CHUNK = "aibot_upload_media_chunk"
85
+ APP_CMD_UPLOAD_MEDIA_FINISH = "aibot_upload_media_finish"
86
+
87
+ CALLBACK_COMMANDS = {APP_CMD_CALLBACK, APP_CMD_LEGACY_CALLBACK}
88
+ NON_RESPONSE_COMMANDS = CALLBACK_COMMANDS | {APP_CMD_EVENT_CALLBACK}
89
+
90
+ MAX_MESSAGE_LENGTH = 4000
91
+ CONNECT_TIMEOUT_SECONDS = 20.0
92
+ REQUEST_TIMEOUT_SECONDS = 15.0
93
+ HEARTBEAT_INTERVAL_SECONDS = 30.0
94
+ RECONNECT_BACKOFF = [2, 5, 10, 30, 60]
95
+
96
+ DEDUP_MAX_SIZE = 1000
97
+
98
+ IMAGE_MAX_BYTES = 10 * 1024 * 1024
99
+ VIDEO_MAX_BYTES = 10 * 1024 * 1024
100
+ VOICE_MAX_BYTES = 2 * 1024 * 1024
101
+ FILE_MAX_BYTES = 20 * 1024 * 1024
102
+ ABSOLUTE_MAX_BYTES = FILE_MAX_BYTES
103
+ UPLOAD_CHUNK_SIZE = 512 * 1024
104
+ MAX_UPLOAD_CHUNKS = 100
105
+ VOICE_SUPPORTED_MIMES = {"audio/amr"}
106
+
107
+
108
+ def check_wecom_requirements() -> bool:
109
+ """Check if WeCom runtime dependencies are available."""
110
+ return AIOHTTP_AVAILABLE and HTTPX_AVAILABLE
111
+
112
+
113
+ def _coerce_list(value: Any) -> List[str]:
114
+ """Coerce config values into a trimmed string list."""
115
+ if value is None:
116
+ return []
117
+ if isinstance(value, str):
118
+ return [item.strip() for item in value.split(",") if item.strip()]
119
+ if isinstance(value, (list, tuple, set)):
120
+ return [str(item).strip() for item in value if str(item).strip()]
121
+ return [str(value).strip()] if str(value).strip() else []
122
+
123
+
124
+ def _normalize_entry(raw: str) -> str:
125
+ """Normalize allowlist entries such as ``wecom:user:foo``."""
126
+ value = str(raw).strip()
127
+ value = re.sub(r"^wecom:", "", value, flags=re.IGNORECASE)
128
+ value = re.sub(r"^(user|group):", "", value, flags=re.IGNORECASE)
129
+ return value.strip()
130
+
131
+
132
+ def _entry_matches(entries: List[str], target: str) -> bool:
133
+ """Case-insensitive allowlist match with ``*`` support."""
134
+ normalized_target = str(target).strip().lower()
135
+ for entry in entries:
136
+ normalized = _normalize_entry(entry).lower()
137
+ if normalized == "*" or normalized == normalized_target:
138
+ return True
139
+ return False
140
+
141
+
142
+ class WeComAdapter(BasePlatformAdapter):
143
+ """WeCom AI Bot adapter backed by a persistent WebSocket connection."""
144
+
145
+ MAX_MESSAGE_LENGTH = MAX_MESSAGE_LENGTH
146
+ SUPPORTS_MESSAGE_EDITING = False
147
+ # Threshold for detecting WeCom client-side message splits.
148
+ # When a chunk is near the 4000-char limit, a continuation is almost certain.
149
+ _SPLIT_THRESHOLD = 3900
150
+
151
+ def __init__(self, config: PlatformConfig):
152
+ super().__init__(config, Platform.WECOM)
153
+
154
+ extra = config.extra or {}
155
+ self._bot_id = str(extra.get("bot_id") or os.getenv("WECOM_BOT_ID", "")).strip()
156
+ self._secret = str(extra.get("secret") or os.getenv("WECOM_SECRET", "")).strip()
157
+ self._ws_url = str(
158
+ extra.get("websocket_url")
159
+ or extra.get("websocketUrl")
160
+ or os.getenv("WECOM_WEBSOCKET_URL", DEFAULT_WS_URL)
161
+ ).strip() or DEFAULT_WS_URL
162
+
163
+ self._dm_policy = str(extra.get("dm_policy") or os.getenv("WECOM_DM_POLICY", "open")).strip().lower()
164
+ self._allow_from = _coerce_list(extra.get("allow_from") or extra.get("allowFrom"))
165
+
166
+ self._group_policy = str(extra.get("group_policy") or os.getenv("WECOM_GROUP_POLICY", "open")).strip().lower()
167
+ self._group_allow_from = _coerce_list(extra.get("group_allow_from") or extra.get("groupAllowFrom"))
168
+ self._groups = extra.get("groups") if isinstance(extra.get("groups"), dict) else {}
169
+
170
+ self._session: Optional["aiohttp.ClientSession"] = None
171
+ self._ws: Optional["aiohttp.ClientWebSocketResponse"] = None
172
+ self._http_client: Optional["httpx.AsyncClient"] = None
173
+ self._listen_task: Optional[asyncio.Task] = None
174
+ self._heartbeat_task: Optional[asyncio.Task] = None
175
+ self._pending_responses: Dict[str, asyncio.Future] = {}
176
+ self._dedup = MessageDeduplicator(max_size=DEDUP_MAX_SIZE)
177
+ self._reply_req_ids: Dict[str, str] = {}
178
+
179
+ # Text batching: merge rapid successive messages (Telegram-style).
180
+ # WeCom clients split long messages around 4000 chars.
181
+ self._text_batch_delay_seconds = float(os.getenv("HERMES_WECOM_TEXT_BATCH_DELAY_SECONDS", "0.6"))
182
+ self._text_batch_split_delay_seconds = float(os.getenv("HERMES_WECOM_TEXT_BATCH_SPLIT_DELAY_SECONDS", "2.0"))
183
+ self._pending_text_batches: Dict[str, MessageEvent] = {}
184
+ self._pending_text_batch_tasks: Dict[str, asyncio.Task] = {}
185
+ self._device_id = uuid.uuid4().hex
186
+ self._last_chat_req_ids: Dict[str, str] = {}
187
+
188
+ # ------------------------------------------------------------------
189
+ # Connection lifecycle
190
+ # ------------------------------------------------------------------
191
+
192
+ async def connect(self) -> bool:
193
+ """Connect to the WeCom AI Bot gateway."""
194
+ if not AIOHTTP_AVAILABLE:
195
+ message = "WeCom startup failed: aiohttp not installed"
196
+ self._set_fatal_error("wecom_missing_dependency", message, retryable=True)
197
+ logger.warning("[%s] %s. Run: pip install aiohttp", self.name, message)
198
+ return False
199
+ if not HTTPX_AVAILABLE:
200
+ message = "WeCom startup failed: httpx not installed"
201
+ self._set_fatal_error("wecom_missing_dependency", message, retryable=True)
202
+ logger.warning("[%s] %s. Run: pip install httpx", self.name, message)
203
+ return False
204
+ if not self._bot_id or not self._secret:
205
+ message = "WeCom startup failed: WECOM_BOT_ID and WECOM_SECRET are required"
206
+ self._set_fatal_error("wecom_missing_credentials", message, retryable=True)
207
+ logger.warning("[%s] %s", self.name, message)
208
+ return False
209
+
210
+ try:
211
+ # Tighter keepalive so idle CLOSE_WAIT drains promptly (#18451).
212
+ from gateway.platforms._http_client_limits import platform_httpx_limits
213
+ self._http_client = httpx.AsyncClient(
214
+ timeout=30.0, follow_redirects=True, limits=platform_httpx_limits(),
215
+ )
216
+ await self._open_connection()
217
+ self._mark_connected()
218
+ self._listen_task = asyncio.create_task(self._listen_loop())
219
+ self._heartbeat_task = asyncio.create_task(self._heartbeat_loop())
220
+ logger.info("[%s] Connected to %s", self.name, self._ws_url)
221
+ return True
222
+ except Exception as exc:
223
+ message = f"WeCom startup failed: {exc}"
224
+ self._set_fatal_error("wecom_connect_error", message, retryable=True)
225
+ logger.error("[%s] Failed to connect: %s", self.name, exc, exc_info=True)
226
+ await self._cleanup_ws()
227
+ if self._http_client:
228
+ await self._http_client.aclose()
229
+ self._http_client = None
230
+ return False
231
+
232
+ async def disconnect(self) -> None:
233
+ """Disconnect from WeCom."""
234
+ self._running = False
235
+ self._mark_disconnected()
236
+
237
+ if self._listen_task:
238
+ self._listen_task.cancel()
239
+ try:
240
+ await self._listen_task
241
+ except asyncio.CancelledError:
242
+ pass
243
+ self._listen_task = None
244
+
245
+ if self._heartbeat_task:
246
+ self._heartbeat_task.cancel()
247
+ try:
248
+ await self._heartbeat_task
249
+ except asyncio.CancelledError:
250
+ pass
251
+ self._heartbeat_task = None
252
+
253
+ self._fail_pending_responses(RuntimeError("WeCom adapter disconnected"))
254
+ await self._cleanup_ws()
255
+
256
+ if self._http_client:
257
+ await self._http_client.aclose()
258
+ self._http_client = None
259
+
260
+ self._dedup.clear()
261
+ logger.info("[%s] Disconnected", self.name)
262
+
263
+ async def _cleanup_ws(self) -> None:
264
+ """Close the live websocket/session, if any."""
265
+ if self._ws and not self._ws.closed:
266
+ await self._ws.close()
267
+ self._ws = None
268
+
269
+ if self._session and not self._session.closed:
270
+ await self._session.close()
271
+ self._session = None
272
+
273
+ async def _open_connection(self) -> None:
274
+ """Open and authenticate a websocket connection."""
275
+ await self._cleanup_ws()
276
+ self._session = aiohttp.ClientSession(trust_env=True)
277
+ self._ws = await self._session.ws_connect(
278
+ self._ws_url,
279
+ heartbeat=HEARTBEAT_INTERVAL_SECONDS * 2,
280
+ timeout=CONNECT_TIMEOUT_SECONDS,
281
+ )
282
+
283
+ req_id = self._new_req_id("subscribe")
284
+ await self._send_json(
285
+ {
286
+ "cmd": APP_CMD_SUBSCRIBE,
287
+ "headers": {"req_id": req_id},
288
+ "body": {
289
+ "bot_id": self._bot_id,
290
+ "secret": self._secret,
291
+ "device_id": self._device_id,
292
+ },
293
+ }
294
+ )
295
+
296
+ auth_payload = await self._wait_for_handshake(req_id)
297
+ errcode = auth_payload.get("errcode", 0)
298
+ if errcode not in {0, None}:
299
+ errmsg = auth_payload.get("errmsg", "authentication failed")
300
+ raise RuntimeError(f"{errmsg} (errcode={errcode})")
301
+
302
+ async def _wait_for_handshake(self, req_id: str) -> Dict[str, Any]:
303
+ """Wait for the subscribe acknowledgement."""
304
+ if not self._ws:
305
+ raise RuntimeError("WebSocket not initialized")
306
+
307
+ deadline = asyncio.get_running_loop().time() + CONNECT_TIMEOUT_SECONDS
308
+ while True:
309
+ remaining = deadline - asyncio.get_running_loop().time()
310
+ if remaining <= 0:
311
+ raise TimeoutError("Timed out waiting for WeCom subscribe acknowledgement")
312
+
313
+ msg = await asyncio.wait_for(self._ws.receive(), timeout=remaining)
314
+ if msg.type == aiohttp.WSMsgType.TEXT:
315
+ payload = self._parse_json(msg.data)
316
+ if not payload:
317
+ continue
318
+ if payload.get("cmd") == APP_CMD_PING:
319
+ continue
320
+ if self._payload_req_id(payload) == req_id:
321
+ return payload
322
+ logger.debug("[%s] Ignoring pre-auth payload: %s", self.name, payload.get("cmd"))
323
+ elif msg.type in {aiohttp.WSMsgType.CLOSED, aiohttp.WSMsgType.CLOSE, aiohttp.WSMsgType.ERROR}:
324
+ raise RuntimeError("WeCom websocket closed during authentication")
325
+
326
+ async def _listen_loop(self) -> None:
327
+ """Read websocket events forever, reconnecting on errors."""
328
+ backoff_idx = 0
329
+ while self._running:
330
+ try:
331
+ await self._read_events()
332
+ backoff_idx = 0
333
+ except asyncio.CancelledError:
334
+ return
335
+ except Exception as exc:
336
+ if not self._running:
337
+ return
338
+ logger.warning("[%s] WebSocket error: %s", self.name, exc)
339
+ self._fail_pending_responses(RuntimeError("WeCom connection interrupted"))
340
+
341
+ delay = RECONNECT_BACKOFF[min(backoff_idx, len(RECONNECT_BACKOFF) - 1)]
342
+ backoff_idx += 1
343
+ await asyncio.sleep(delay)
344
+
345
+ try:
346
+ await self._open_connection()
347
+ backoff_idx = 0
348
+ self._mark_connected()
349
+ logger.info("[%s] Reconnected", self.name)
350
+ except Exception as reconnect_exc:
351
+ logger.warning("[%s] Reconnect failed: %s", self.name, reconnect_exc)
352
+
353
+ async def _read_events(self) -> None:
354
+ """Read websocket frames until the connection closes."""
355
+ if not self._ws:
356
+ raise RuntimeError("WebSocket not connected")
357
+
358
+ while self._running and self._ws and not self._ws.closed:
359
+ msg = await self._ws.receive()
360
+ if msg.type == aiohttp.WSMsgType.TEXT:
361
+ payload = self._parse_json(msg.data)
362
+ if payload:
363
+ await self._dispatch_payload(payload)
364
+ elif msg.type in {aiohttp.WSMsgType.CLOSE, aiohttp.WSMsgType.CLOSED, aiohttp.WSMsgType.ERROR}:
365
+ raise RuntimeError("WeCom websocket closed")
366
+
367
+ async def _heartbeat_loop(self) -> None:
368
+ """Send lightweight application-level pings."""
369
+ try:
370
+ while self._running:
371
+ await asyncio.sleep(HEARTBEAT_INTERVAL_SECONDS)
372
+ if not self._ws or self._ws.closed:
373
+ continue
374
+ try:
375
+ await self._send_json(
376
+ {
377
+ "cmd": APP_CMD_PING,
378
+ "headers": {"req_id": self._new_req_id("ping")},
379
+ "body": {},
380
+ }
381
+ )
382
+ except Exception as exc:
383
+ logger.debug("[%s] Heartbeat send failed: %s", self.name, exc)
384
+ except asyncio.CancelledError:
385
+ pass
386
+
387
+ async def _dispatch_payload(self, payload: Dict[str, Any]) -> None:
388
+ """Route inbound websocket payloads."""
389
+ req_id = self._payload_req_id(payload)
390
+ cmd = str(payload.get("cmd") or "")
391
+
392
+ if req_id and req_id in self._pending_responses and cmd not in NON_RESPONSE_COMMANDS:
393
+ future = self._pending_responses.get(req_id)
394
+ if future and not future.done():
395
+ future.set_result(payload)
396
+ return
397
+
398
+ if cmd in CALLBACK_COMMANDS:
399
+ await self._on_message(payload)
400
+ return
401
+ if cmd in {APP_CMD_PING, APP_CMD_EVENT_CALLBACK}:
402
+ return
403
+
404
+ logger.debug("[%s] Ignoring websocket payload: %s", self.name, cmd or payload)
405
+
406
+ def _fail_pending_responses(self, exc: Exception) -> None:
407
+ """Fail all outstanding request futures."""
408
+ for req_id, future in list(self._pending_responses.items()):
409
+ if not future.done():
410
+ future.set_exception(exc)
411
+ self._pending_responses.pop(req_id, None)
412
+
413
+ async def _send_json(self, payload: Dict[str, Any]) -> None:
414
+ """Send a raw JSON frame over the active websocket."""
415
+ if not self._ws or self._ws.closed:
416
+ raise RuntimeError("WeCom websocket is not connected")
417
+ await self._ws.send_json(payload)
418
+
419
+ async def _send_request(self, cmd: str, body: Dict[str, Any], timeout: float = REQUEST_TIMEOUT_SECONDS) -> Dict[str, Any]:
420
+ """Send a JSON request and await the correlated response."""
421
+ if not self._ws or self._ws.closed:
422
+ raise RuntimeError("WeCom websocket is not connected")
423
+
424
+ req_id = self._new_req_id(cmd)
425
+ future = asyncio.get_running_loop().create_future()
426
+ self._pending_responses[req_id] = future
427
+ try:
428
+ await self._send_json({"cmd": cmd, "headers": {"req_id": req_id}, "body": body})
429
+ response = await asyncio.wait_for(future, timeout=timeout)
430
+ return response
431
+ finally:
432
+ self._pending_responses.pop(req_id, None)
433
+
434
+ async def _send_reply_request(
435
+ self,
436
+ reply_req_id: str,
437
+ body: Dict[str, Any],
438
+ cmd: str = APP_CMD_RESPONSE,
439
+ timeout: float = REQUEST_TIMEOUT_SECONDS,
440
+ ) -> Dict[str, Any]:
441
+ """Send a reply frame correlated to an inbound callback req_id."""
442
+ if not self._ws or self._ws.closed:
443
+ raise RuntimeError("WeCom websocket is not connected")
444
+
445
+ normalized_req_id = str(reply_req_id or "").strip()
446
+ if not normalized_req_id:
447
+ raise ValueError("reply_req_id is required")
448
+
449
+ future = asyncio.get_running_loop().create_future()
450
+ self._pending_responses[normalized_req_id] = future
451
+ try:
452
+ await self._send_json(
453
+ {"cmd": cmd, "headers": {"req_id": normalized_req_id}, "body": body}
454
+ )
455
+ response = await asyncio.wait_for(future, timeout=timeout)
456
+ return response
457
+ finally:
458
+ self._pending_responses.pop(normalized_req_id, None)
459
+
460
+ @staticmethod
461
+ def _new_req_id(prefix: str) -> str:
462
+ return f"{prefix}-{uuid.uuid4().hex}"
463
+
464
+ @staticmethod
465
+ def _payload_req_id(payload: Dict[str, Any]) -> str:
466
+ headers = payload.get("headers")
467
+ if isinstance(headers, dict):
468
+ return str(headers.get("req_id") or "")
469
+ return ""
470
+
471
+ @staticmethod
472
+ def _parse_json(raw: Any) -> Optional[Dict[str, Any]]:
473
+ try:
474
+ payload = json.loads(raw)
475
+ except Exception:
476
+ logger.debug("Failed to parse WeCom payload: %r", raw)
477
+ return None
478
+ return payload if isinstance(payload, dict) else None
479
+
480
+ # ------------------------------------------------------------------
481
+ # Inbound message parsing
482
+ # ------------------------------------------------------------------
483
+
484
+ async def _on_message(self, payload: Dict[str, Any]) -> None:
485
+ """Process an inbound WeCom message callback event."""
486
+ body = payload.get("body")
487
+ if not isinstance(body, dict):
488
+ return
489
+
490
+ msg_id = str(body.get("msgid") or self._payload_req_id(payload) or uuid.uuid4().hex)
491
+ if self._dedup.is_duplicate(msg_id):
492
+ logger.debug("[%s] Duplicate message %s ignored", self.name, msg_id)
493
+ return
494
+ self._remember_reply_req_id(msg_id, self._payload_req_id(payload))
495
+
496
+ sender = body.get("from") if isinstance(body.get("from"), dict) else {}
497
+ sender_id = str(sender.get("userid") or "").strip()
498
+ chat_id = str(body.get("chatid") or sender_id).strip()
499
+ if not chat_id:
500
+ logger.debug("[%s] Missing chat id, skipping message", self.name)
501
+ return
502
+
503
+ is_group = str(body.get("chattype") or "").lower() == "group"
504
+ if is_group:
505
+ if not self._is_group_allowed(chat_id, sender_id):
506
+ logger.debug("[%s] Group %s / sender %s blocked by policy", self.name, chat_id, sender_id)
507
+ return
508
+ elif not self._is_dm_allowed(sender_id):
509
+ logger.debug("[%s] DM sender %s blocked by policy", self.name, sender_id)
510
+ return
511
+
512
+ # Cache the inbound req_id after policy checks so proactive sends to
513
+ # this chat can fall back to APP_CMD_RESPONSE (required for groups —
514
+ # WeCom AI Bots cannot initiate APP_CMD_SEND in group chats).
515
+ self._remember_chat_req_id(chat_id, self._payload_req_id(payload))
516
+
517
+ text, reply_text = self._extract_text(body)
518
+ # Strip leading @mention in group chats so slash commands like
519
+ # "@BotName /approve" are correctly recognized as "/approve".
520
+ # Mirrors what the Telegram adapter does (re.sub @botname).
521
+ if is_group and text:
522
+ text = re.sub(r"^@\S+\s*", "", text).strip()
523
+ media_urls, media_types = await self._extract_media(body)
524
+ message_type = self._derive_message_type(body, text, media_types)
525
+ has_reply_context = bool(reply_text and (text or media_urls))
526
+
527
+ if not text and reply_text and not media_urls:
528
+ text = reply_text
529
+
530
+ if not text and not media_urls:
531
+ logger.debug("[%s] Empty WeCom message skipped", self.name)
532
+ return
533
+
534
+ source = self.build_source(
535
+ chat_id=chat_id,
536
+ chat_type="group" if is_group else "dm",
537
+ user_id=sender_id or None,
538
+ user_name=sender_id or None,
539
+ )
540
+
541
+ event = MessageEvent(
542
+ text=text,
543
+ message_type=message_type,
544
+ source=source,
545
+ raw_message=payload,
546
+ message_id=msg_id,
547
+ media_urls=media_urls,
548
+ media_types=media_types,
549
+ reply_to_message_id=f"quote:{msg_id}" if has_reply_context else None,
550
+ reply_to_text=reply_text if has_reply_context else None,
551
+ timestamp=datetime.now(tz=timezone.utc),
552
+ )
553
+
554
+ # Only batch plain text messages — commands, media, etc. dispatch
555
+ # immediately since they won't be split by the WeCom client.
556
+ if message_type == MessageType.TEXT and self._text_batch_delay_seconds > 0:
557
+ self._enqueue_text_event(event)
558
+ else:
559
+ await self.handle_message(event)
560
+
561
+ # ------------------------------------------------------------------
562
+ # Text message aggregation (handles WeCom client-side splits)
563
+ # ------------------------------------------------------------------
564
+
565
+ def _text_batch_key(self, event: MessageEvent) -> str:
566
+ """Session-scoped key for text message batching."""
567
+ from gateway.session import build_session_key
568
+ return build_session_key(
569
+ event.source,
570
+ group_sessions_per_user=self.config.extra.get("group_sessions_per_user", True),
571
+ thread_sessions_per_user=self.config.extra.get("thread_sessions_per_user", False),
572
+ )
573
+
574
+ def _enqueue_text_event(self, event: MessageEvent) -> None:
575
+ """Buffer a text event and reset the flush timer.
576
+
577
+ When WeCom splits a long user message at 4000 chars, the chunks
578
+ arrive within a few hundred milliseconds. This merges them into
579
+ a single event before dispatching.
580
+ """
581
+ key = self._text_batch_key(event)
582
+ existing = self._pending_text_batches.get(key)
583
+ chunk_len = len(event.text or "")
584
+ if existing is None:
585
+ event._last_chunk_len = chunk_len # type: ignore[attr-defined]
586
+ self._pending_text_batches[key] = event
587
+ else:
588
+ if event.text:
589
+ existing.text = f"{existing.text}\n{event.text}" if existing.text else event.text
590
+ existing._last_chunk_len = chunk_len # type: ignore[attr-defined]
591
+ # Merge any media that might be attached
592
+ if event.media_urls:
593
+ existing.media_urls.extend(event.media_urls)
594
+ existing.media_types.extend(event.media_types)
595
+
596
+ # Cancel any pending flush and restart the timer
597
+ prior_task = self._pending_text_batch_tasks.get(key)
598
+ if prior_task and not prior_task.done():
599
+ prior_task.cancel()
600
+ self._pending_text_batch_tasks[key] = asyncio.create_task(
601
+ self._flush_text_batch(key)
602
+ )
603
+
604
+ async def _flush_text_batch(self, key: str) -> None:
605
+ """Wait for the quiet period then dispatch the aggregated text.
606
+
607
+ Uses a longer delay when the latest chunk is near WeCom's 4000-char
608
+ split point, since a continuation chunk is almost certain.
609
+ """
610
+ current_task = asyncio.current_task()
611
+ try:
612
+ pending = self._pending_text_batches.get(key)
613
+ last_len = getattr(pending, "_last_chunk_len", 0) if pending else 0
614
+ if last_len >= self._SPLIT_THRESHOLD:
615
+ delay = self._text_batch_split_delay_seconds
616
+ else:
617
+ delay = self._text_batch_delay_seconds
618
+ await asyncio.sleep(delay)
619
+ event = self._pending_text_batches.pop(key, None)
620
+ if not event:
621
+ return
622
+ logger.info(
623
+ "[WeCom] Flushing text batch %s (%d chars)",
624
+ key, len(event.text or ""),
625
+ )
626
+ await self.handle_message(event)
627
+ finally:
628
+ if self._pending_text_batch_tasks.get(key) is current_task:
629
+ self._pending_text_batch_tasks.pop(key, None)
630
+
631
+ @staticmethod
632
+ def _extract_text(body: Dict[str, Any]) -> Tuple[str, Optional[str]]:
633
+ """Extract plain text and quoted text from a callback payload."""
634
+ text_parts: List[str] = []
635
+ reply_text: Optional[str] = None
636
+ msgtype = str(body.get("msgtype") or "").lower()
637
+
638
+ if msgtype == "mixed":
639
+ _raw_mixed = body.get("mixed")
640
+ mixed = _raw_mixed if isinstance(_raw_mixed, dict) else {}
641
+ _raw_items = mixed.get("msg_item")
642
+ items = _raw_items if isinstance(_raw_items, list) else []
643
+ for item in items:
644
+ if not isinstance(item, dict):
645
+ continue
646
+ if str(item.get("msgtype") or "").lower() == "text":
647
+ _raw_text = item.get("text")
648
+ text_block = _raw_text if isinstance(_raw_text, dict) else {}
649
+ content = str(text_block.get("content") or "").strip()
650
+ if content:
651
+ text_parts.append(content)
652
+ else:
653
+ text_block = body.get("text") if isinstance(body.get("text"), dict) else {}
654
+ content = str(text_block.get("content") or "").strip()
655
+ if content:
656
+ text_parts.append(content)
657
+
658
+ if msgtype == "voice":
659
+ voice_block = body.get("voice") if isinstance(body.get("voice"), dict) else {}
660
+ voice_text = str(voice_block.get("content") or "").strip()
661
+ if voice_text:
662
+ text_parts.append(voice_text)
663
+
664
+ # Extract appmsg title (filename) for WeCom AI Bot attachments
665
+ if msgtype == "appmsg":
666
+ appmsg = body.get("appmsg") if isinstance(body.get("appmsg"), dict) else {}
667
+ title = str(appmsg.get("title") or "").strip()
668
+ if title:
669
+ text_parts.append(title)
670
+
671
+ quote = body.get("quote") if isinstance(body.get("quote"), dict) else {}
672
+ quote_type = str(quote.get("msgtype") or "").lower()
673
+ if quote_type == "text":
674
+ quote_text = quote.get("text") if isinstance(quote.get("text"), dict) else {}
675
+ reply_text = str(quote_text.get("content") or "").strip() or None
676
+ elif quote_type == "voice":
677
+ quote_voice = quote.get("voice") if isinstance(quote.get("voice"), dict) else {}
678
+ reply_text = str(quote_voice.get("content") or "").strip() or None
679
+
680
+ return "\n".join(part for part in text_parts if part).strip(), reply_text
681
+
682
+ async def _extract_media(self, body: Dict[str, Any]) -> Tuple[List[str], List[str]]:
683
+ """Best-effort extraction of inbound media to local cache paths."""
684
+ media_paths: List[str] = []
685
+ media_types: List[str] = []
686
+ refs: List[Tuple[str, Dict[str, Any]]] = []
687
+ msgtype = str(body.get("msgtype") or "").lower()
688
+
689
+ if msgtype == "mixed":
690
+ _raw_mixed = body.get("mixed")
691
+ mixed = _raw_mixed if isinstance(_raw_mixed, dict) else {}
692
+ _raw_items = mixed.get("msg_item")
693
+ items = _raw_items if isinstance(_raw_items, list) else []
694
+ for item in items:
695
+ if not isinstance(item, dict):
696
+ continue
697
+ item_type = str(item.get("msgtype") or "").lower()
698
+ if item_type == "image" and isinstance(item.get("image"), dict):
699
+ refs.append(("image", item["image"]))
700
+ else:
701
+ if isinstance(body.get("image"), dict):
702
+ refs.append(("image", body["image"]))
703
+ if msgtype == "file" and isinstance(body.get("file"), dict):
704
+ refs.append(("file", body["file"]))
705
+ # Handle appmsg (WeCom AI Bot attachments with PDF/Word/Excel)
706
+ if msgtype == "appmsg" and isinstance(body.get("appmsg"), dict):
707
+ appmsg = body["appmsg"]
708
+ if isinstance(appmsg.get("file"), dict):
709
+ refs.append(("file", appmsg["file"]))
710
+ elif isinstance(appmsg.get("image"), dict):
711
+ refs.append(("image", appmsg["image"]))
712
+
713
+ quote = body.get("quote") if isinstance(body.get("quote"), dict) else {}
714
+ quote_type = str(quote.get("msgtype") or "").lower()
715
+ if quote_type == "image" and isinstance(quote.get("image"), dict):
716
+ refs.append(("image", quote["image"]))
717
+ elif quote_type == "file" and isinstance(quote.get("file"), dict):
718
+ refs.append(("file", quote["file"]))
719
+
720
+ for kind, ref in refs:
721
+ cached = await self._cache_media(kind, ref)
722
+ if cached:
723
+ path, content_type = cached
724
+ media_paths.append(path)
725
+ media_types.append(content_type)
726
+
727
+ return media_paths, media_types
728
+
729
+ async def _cache_media(self, kind: str, media: Dict[str, Any]) -> Optional[Tuple[str, str]]:
730
+ """Cache an inbound image/file/media reference to local storage."""
731
+ if "base64" in media and media.get("base64"):
732
+ try:
733
+ raw = self._decode_base64(media["base64"])
734
+ except Exception as exc:
735
+ logger.debug("[%s] Failed to decode %s base64 media: %s", self.name, kind, exc)
736
+ return None
737
+
738
+ if kind == "image":
739
+ ext = self._detect_image_ext(raw)
740
+ try:
741
+ return cache_image_from_bytes(raw, ext), self._mime_for_ext(ext, fallback="image/jpeg")
742
+ except ValueError as exc:
743
+ logger.warning("[%s] Rejected non-image bytes: %s", self.name, exc)
744
+ return None
745
+
746
+ filename = str(media.get("filename") or media.get("name") or "wecom_file")
747
+ return cache_document_from_bytes(raw, filename), mimetypes.guess_type(filename)[0] or "application/octet-stream"
748
+
749
+ url = str(media.get("url") or "").strip()
750
+ if not url:
751
+ return None
752
+
753
+ try:
754
+ raw, headers = await self._download_remote_bytes(url, max_bytes=ABSOLUTE_MAX_BYTES)
755
+ except Exception as exc:
756
+ logger.debug("[%s] Failed to download %s from %s: %s", self.name, kind, url, exc)
757
+ return None
758
+
759
+ aes_key = str(media.get("aeskey") or "").strip()
760
+ if aes_key:
761
+ try:
762
+ raw = self._decrypt_file_bytes(raw, aes_key)
763
+ except Exception as exc:
764
+ logger.debug("[%s] Failed to decrypt %s from %s: %s", self.name, kind, url, exc)
765
+ return None
766
+
767
+ content_type = str(headers.get("content-type") or "").split(";", 1)[0].strip() or "application/octet-stream"
768
+ if kind == "image":
769
+ ext = self._guess_extension(url, content_type, fallback=self._detect_image_ext(raw))
770
+ try:
771
+ return cache_image_from_bytes(raw, ext), content_type or self._mime_for_ext(ext, fallback="image/jpeg")
772
+ except ValueError as exc:
773
+ logger.warning("[%s] Rejected non-image bytes from %s: %s", self.name, url, exc)
774
+ return None
775
+
776
+ filename = self._guess_filename(url, headers.get("content-disposition"), content_type)
777
+ return cache_document_from_bytes(raw, filename), content_type
778
+
779
+ @staticmethod
780
+ def _decode_base64(data: str) -> bytes:
781
+ payload = data.split(",", 1)[-1].strip()
782
+ return base64.b64decode(payload)
783
+
784
+ @staticmethod
785
+ def _detect_image_ext(data: bytes) -> str:
786
+ if data.startswith(b"\x89PNG\r\n\x1a\n"):
787
+ return ".png"
788
+ if data.startswith(b"\xff\xd8\xff"):
789
+ return ".jpg"
790
+ if data.startswith((b"GIF87a", b"GIF89a")):
791
+ return ".gif"
792
+ if data.startswith(b"RIFF") and data[8:12] == b"WEBP":
793
+ return ".webp"
794
+ return ".jpg"
795
+
796
+ @staticmethod
797
+ def _mime_for_ext(ext: str, fallback: str = "application/octet-stream") -> str:
798
+ return mimetypes.types_map.get(ext.lower(), fallback)
799
+
800
+ @staticmethod
801
+ def _guess_extension(url: str, content_type: str, fallback: str) -> str:
802
+ ext = mimetypes.guess_extension(content_type) if content_type else None
803
+ if ext:
804
+ return ext
805
+ path_ext = Path(urlparse(url).path).suffix
806
+ if path_ext:
807
+ return path_ext
808
+ return fallback
809
+
810
+ @staticmethod
811
+ def _guess_filename(url: str, content_disposition: Optional[str], content_type: str) -> str:
812
+ if content_disposition:
813
+ match = re.search(r'filename="?([^";]+)"?', content_disposition)
814
+ if match:
815
+ return match.group(1)
816
+
817
+ name = Path(urlparse(url).path).name or "document"
818
+ if "." not in name:
819
+ ext = mimetypes.guess_extension(content_type) or ".bin"
820
+ name = f"{name}{ext}"
821
+ return name
822
+
823
+ @staticmethod
824
+ def _derive_message_type(body: Dict[str, Any], text: str, media_types: List[str]) -> MessageType:
825
+ """Choose the normalized inbound message type."""
826
+ if any(mtype.startswith(("application/", "text/")) for mtype in media_types):
827
+ return MessageType.DOCUMENT
828
+ if any(mtype.startswith("image/") for mtype in media_types):
829
+ return MessageType.TEXT if text else MessageType.PHOTO
830
+ if str(body.get("msgtype") or "").lower() == "voice":
831
+ return MessageType.VOICE
832
+ return MessageType.TEXT
833
+
834
+ # ------------------------------------------------------------------
835
+ # Policy helpers
836
+ # ------------------------------------------------------------------
837
+
838
+ def _is_dm_allowed(self, sender_id: str) -> bool:
839
+ if self._dm_policy == "disabled":
840
+ return False
841
+ if self._dm_policy == "allowlist":
842
+ return _entry_matches(self._allow_from, sender_id)
843
+ return True
844
+
845
+ def _is_group_allowed(self, chat_id: str, sender_id: str) -> bool:
846
+ if self._group_policy == "disabled":
847
+ return False
848
+ if self._group_policy == "allowlist" and not _entry_matches(self._group_allow_from, chat_id):
849
+ return False
850
+
851
+ group_cfg = self._resolve_group_cfg(chat_id)
852
+ sender_allow = _coerce_list(group_cfg.get("allow_from") or group_cfg.get("allowFrom"))
853
+ if sender_allow:
854
+ return _entry_matches(sender_allow, sender_id)
855
+ return True
856
+
857
+ def _resolve_group_cfg(self, chat_id: str) -> Dict[str, Any]:
858
+ if not isinstance(self._groups, dict):
859
+ return {}
860
+ if chat_id in self._groups and isinstance(self._groups[chat_id], dict):
861
+ return self._groups[chat_id]
862
+ lowered = chat_id.lower()
863
+ for key, value in self._groups.items():
864
+ if isinstance(key, str) and key.lower() == lowered and isinstance(value, dict):
865
+ return value
866
+ wildcard = self._groups.get("*")
867
+ return wildcard if isinstance(wildcard, dict) else {}
868
+
869
+ def _remember_reply_req_id(self, message_id: str, req_id: str) -> None:
870
+ normalized_message_id = str(message_id or "").strip()
871
+ normalized_req_id = str(req_id or "").strip()
872
+ if not normalized_message_id or not normalized_req_id:
873
+ return
874
+ self._reply_req_ids[normalized_message_id] = normalized_req_id
875
+ while len(self._reply_req_ids) > DEDUP_MAX_SIZE:
876
+ self._reply_req_ids.pop(next(iter(self._reply_req_ids)))
877
+
878
+ def _remember_chat_req_id(self, chat_id: str, req_id: str) -> None:
879
+ """Cache the most recent inbound req_id per chat.
880
+
881
+ Used as a fallback reply target when we need to send into a group
882
+ without an explicit ``reply_to`` — WeCom AI Bots are blocked from
883
+ APP_CMD_SEND in groups and must use APP_CMD_RESPONSE bound to some
884
+ prior req_id. Bounded like _reply_req_ids so long-running gateways
885
+ don't leak memory across many chats.
886
+ """
887
+ normalized_chat_id = str(chat_id or "").strip()
888
+ normalized_req_id = str(req_id or "").strip()
889
+ if not normalized_chat_id or not normalized_req_id:
890
+ return
891
+ self._last_chat_req_ids[normalized_chat_id] = normalized_req_id
892
+ while len(self._last_chat_req_ids) > DEDUP_MAX_SIZE:
893
+ self._last_chat_req_ids.pop(next(iter(self._last_chat_req_ids)))
894
+
895
+ def _reply_req_id_for_message(self, reply_to: Optional[str]) -> Optional[str]:
896
+ normalized = str(reply_to or "").strip()
897
+ if not normalized or normalized.startswith("quote:"):
898
+ return None
899
+ return self._reply_req_ids.get(normalized)
900
+
901
+ # ------------------------------------------------------------------
902
+ # Outbound messaging
903
+ # ------------------------------------------------------------------
904
+
905
+ @staticmethod
906
+ def _guess_mime_type(filename: str) -> str:
907
+ mime_type = mimetypes.guess_type(filename)[0]
908
+ if mime_type:
909
+ return mime_type
910
+ if Path(filename).suffix.lower() == ".amr":
911
+ return "audio/amr"
912
+ return "application/octet-stream"
913
+
914
+ @staticmethod
915
+ def _normalize_content_type(content_type: str, filename: str) -> str:
916
+ normalized = str(content_type or "").split(";", 1)[0].strip().lower()
917
+ guessed = WeComAdapter._guess_mime_type(filename)
918
+ if not normalized:
919
+ return guessed
920
+ if normalized in {"application/octet-stream", "text/plain"}:
921
+ return guessed
922
+ return normalized
923
+
924
+ @staticmethod
925
+ def _detect_wecom_media_type(content_type: str) -> str:
926
+ mime_type = str(content_type or "").strip().lower()
927
+ if mime_type.startswith("image/"):
928
+ return "image"
929
+ if mime_type.startswith("video/"):
930
+ return "video"
931
+ if mime_type.startswith("audio/") or mime_type == "application/ogg":
932
+ return "voice"
933
+ return "file"
934
+
935
+ @staticmethod
936
+ def _apply_file_size_limits(file_size: int, detected_type: str, content_type: Optional[str] = None) -> Dict[str, Any]:
937
+ file_size_mb = file_size / (1024 * 1024)
938
+ normalized_type = str(detected_type or "file").lower()
939
+ normalized_content_type = str(content_type or "").strip().lower()
940
+
941
+ if file_size > ABSOLUTE_MAX_BYTES:
942
+ return {
943
+ "final_type": normalized_type,
944
+ "rejected": True,
945
+ "reject_reason": (
946
+ f"文件大小 {file_size_mb:.2f}MB 超过了企业微信允许的最大限制 20MB,无法发送。"
947
+ "请尝试压缩文件或减小文件大小。"
948
+ ),
949
+ "downgraded": False,
950
+ "downgrade_note": None,
951
+ }
952
+
953
+ if normalized_type == "image" and file_size > IMAGE_MAX_BYTES:
954
+ return {
955
+ "final_type": "file",
956
+ "rejected": False,
957
+ "reject_reason": None,
958
+ "downgraded": True,
959
+ "downgrade_note": f"图片大小 {file_size_mb:.2f}MB 超过 10MB 限制,已转为文件格式发送",
960
+ }
961
+
962
+ if normalized_type == "video" and file_size > VIDEO_MAX_BYTES:
963
+ return {
964
+ "final_type": "file",
965
+ "rejected": False,
966
+ "reject_reason": None,
967
+ "downgraded": True,
968
+ "downgrade_note": f"视频大小 {file_size_mb:.2f}MB 超过 10MB 限制,已转为文件格式发送",
969
+ }
970
+
971
+ if normalized_type == "voice":
972
+ if normalized_content_type and normalized_content_type not in VOICE_SUPPORTED_MIMES:
973
+ return {
974
+ "final_type": "file",
975
+ "rejected": False,
976
+ "reject_reason": None,
977
+ "downgraded": True,
978
+ "downgrade_note": (
979
+ f"语音格式 {normalized_content_type} 不支持,企微仅支持 AMR 格式,已转为文件格式发送"
980
+ ),
981
+ }
982
+ if file_size > VOICE_MAX_BYTES:
983
+ return {
984
+ "final_type": "file",
985
+ "rejected": False,
986
+ "reject_reason": None,
987
+ "downgraded": True,
988
+ "downgrade_note": f"语音大小 {file_size_mb:.2f}MB 超过 2MB 限制,已转为文件格式发送",
989
+ }
990
+
991
+ return {
992
+ "final_type": normalized_type,
993
+ "rejected": False,
994
+ "reject_reason": None,
995
+ "downgraded": False,
996
+ "downgrade_note": None,
997
+ }
998
+
999
+ @staticmethod
1000
+ def _response_error(response: Dict[str, Any]) -> Optional[str]:
1001
+ errcode = response.get("errcode", 0)
1002
+ if errcode in {0, None}:
1003
+ return None
1004
+ errmsg = str(response.get("errmsg") or "unknown error")
1005
+ return f"WeCom errcode {errcode}: {errmsg}"
1006
+
1007
+ @classmethod
1008
+ def _raise_for_wecom_error(cls, response: Dict[str, Any], operation: str) -> None:
1009
+ error = cls._response_error(response)
1010
+ if error:
1011
+ raise RuntimeError(f"{operation} failed: {error}")
1012
+
1013
+ @staticmethod
1014
+ def _decrypt_file_bytes(encrypted_data: bytes, aes_key: str) -> bytes:
1015
+ if not encrypted_data:
1016
+ raise ValueError("encrypted_data is empty")
1017
+ if not aes_key:
1018
+ raise ValueError("aes_key is required")
1019
+
1020
+ # WeCom doesn't pad base64 keys; add padding if needed
1021
+ aes_key = aes_key + '=' * ((4 - len(aes_key) % 4) % 4)
1022
+ key = base64.b64decode(aes_key)
1023
+ if len(key) != 32:
1024
+ raise ValueError(f"Invalid WeCom AES key length: expected 32 bytes, got {len(key)}")
1025
+
1026
+ try:
1027
+ from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes
1028
+ except ImportError as exc: # pragma: no cover - dependency is environment-specific
1029
+ raise RuntimeError("cryptography is required for WeCom media decryption") from exc
1030
+
1031
+ cipher = Cipher(algorithms.AES(key), modes.CBC(key[:16]))
1032
+ decryptor = cipher.decryptor()
1033
+ decrypted = decryptor.update(encrypted_data) + decryptor.finalize()
1034
+
1035
+ pad_len = decrypted[-1]
1036
+ if pad_len < 1 or pad_len > 32 or pad_len > len(decrypted):
1037
+ raise ValueError(f"Invalid PKCS#7 padding value: {pad_len}")
1038
+ if any(byte != pad_len for byte in decrypted[-pad_len:]):
1039
+ raise ValueError("Invalid PKCS#7 padding: padding bytes mismatch")
1040
+
1041
+ return decrypted[:-pad_len]
1042
+
1043
+ async def _download_remote_bytes(
1044
+ self,
1045
+ url: str,
1046
+ max_bytes: int,
1047
+ ) -> Tuple[bytes, Dict[str, str]]:
1048
+ from tools.url_safety import is_safe_url
1049
+ if not is_safe_url(url):
1050
+ raise ValueError(f"Blocked unsafe URL (SSRF protection): {url[:80]}")
1051
+
1052
+ if not HTTPX_AVAILABLE:
1053
+ raise RuntimeError("httpx is required for WeCom media download")
1054
+
1055
+ client = self._http_client or httpx.AsyncClient(timeout=30.0, follow_redirects=True)
1056
+ created_client = client is not self._http_client
1057
+ try:
1058
+ async with client.stream(
1059
+ "GET",
1060
+ url,
1061
+ headers={
1062
+ "User-Agent": "HermesAgent/1.0",
1063
+ "Accept": "*/*",
1064
+ },
1065
+ ) as response:
1066
+ response.raise_for_status()
1067
+ headers = {key.lower(): value for key, value in response.headers.items()}
1068
+ content_length = headers.get("content-length")
1069
+ if content_length and content_length.isdigit() and int(content_length) > max_bytes:
1070
+ raise ValueError(
1071
+ f"Remote media exceeds WeCom limit: {int(content_length)} bytes > {max_bytes} bytes"
1072
+ )
1073
+
1074
+ data = bytearray()
1075
+ async for chunk in response.aiter_bytes():
1076
+ data.extend(chunk)
1077
+ if len(data) > max_bytes:
1078
+ raise ValueError(
1079
+ f"Remote media exceeds WeCom limit while downloading: {len(data)} bytes > {max_bytes} bytes"
1080
+ )
1081
+
1082
+ return bytes(data), headers
1083
+ finally:
1084
+ if created_client:
1085
+ await client.aclose()
1086
+
1087
+ @staticmethod
1088
+ def _looks_like_url(media_source: str) -> bool:
1089
+ parsed = urlparse(str(media_source or ""))
1090
+ return parsed.scheme in {"http", "https"}
1091
+
1092
+ async def _load_outbound_media(
1093
+ self,
1094
+ media_source: str,
1095
+ file_name: Optional[str] = None,
1096
+ ) -> Tuple[bytes, str, str]:
1097
+ source = str(media_source or "").strip()
1098
+ if not source:
1099
+ raise ValueError("media source is required")
1100
+ if re.fullmatch(r"<[^>\n]+>", source):
1101
+ raise ValueError(f"Media placeholder was not replaced with a real file path: {source}")
1102
+
1103
+ parsed = urlparse(source)
1104
+ if parsed.scheme in {"http", "https"}:
1105
+ data, headers = await self._download_remote_bytes(source, max_bytes=ABSOLUTE_MAX_BYTES)
1106
+ content_disposition = headers.get("content-disposition")
1107
+ resolved_name = file_name or self._guess_filename(source, content_disposition, headers.get("content-type", ""))
1108
+ content_type = self._normalize_content_type(headers.get("content-type", ""), resolved_name)
1109
+ return data, content_type, resolved_name
1110
+
1111
+ if parsed.scheme == "file":
1112
+ local_path = Path(unquote(parsed.path)).expanduser()
1113
+ else:
1114
+ local_path = Path(source).expanduser()
1115
+
1116
+ if not local_path.is_absolute():
1117
+ local_path = (Path.cwd() / local_path).resolve()
1118
+
1119
+ if not local_path.exists() or not local_path.is_file():
1120
+ raise FileNotFoundError(f"Media file not found: {local_path}")
1121
+
1122
+ data = local_path.read_bytes()
1123
+ resolved_name = file_name or local_path.name
1124
+ content_type = self._normalize_content_type("", resolved_name)
1125
+ return data, content_type, resolved_name
1126
+
1127
+ async def _prepare_outbound_media(
1128
+ self,
1129
+ media_source: str,
1130
+ file_name: Optional[str] = None,
1131
+ ) -> Dict[str, Any]:
1132
+ data, content_type, resolved_name = await self._load_outbound_media(media_source, file_name=file_name)
1133
+ detected_type = self._detect_wecom_media_type(content_type)
1134
+ size_check = self._apply_file_size_limits(len(data), detected_type, content_type)
1135
+ return {
1136
+ "data": data,
1137
+ "content_type": content_type,
1138
+ "file_name": resolved_name,
1139
+ "detected_type": detected_type,
1140
+ **size_check,
1141
+ }
1142
+
1143
+ async def _upload_media_bytes(self, data: bytes, media_type: str, filename: str) -> Dict[str, Any]:
1144
+ if not data:
1145
+ raise ValueError("Cannot upload empty media")
1146
+
1147
+ total_size = len(data)
1148
+ total_chunks = (total_size + UPLOAD_CHUNK_SIZE - 1) // UPLOAD_CHUNK_SIZE
1149
+ if total_chunks > MAX_UPLOAD_CHUNKS:
1150
+ raise ValueError(
1151
+ f"File too large: {total_chunks} chunks exceeds maximum of {MAX_UPLOAD_CHUNKS} chunks"
1152
+ )
1153
+
1154
+ init_response = await self._send_request(
1155
+ APP_CMD_UPLOAD_MEDIA_INIT,
1156
+ {
1157
+ "type": media_type,
1158
+ "filename": filename,
1159
+ "total_size": total_size,
1160
+ "total_chunks": total_chunks,
1161
+ "md5": hashlib.md5(data).hexdigest(),
1162
+ },
1163
+ )
1164
+ self._raise_for_wecom_error(init_response, "media upload init")
1165
+
1166
+ init_body = init_response.get("body") if isinstance(init_response.get("body"), dict) else {}
1167
+ upload_id = str(init_body.get("upload_id") or "").strip()
1168
+ if not upload_id:
1169
+ raise RuntimeError(f"media upload init failed: missing upload_id in response {init_response}")
1170
+
1171
+ for chunk_index, start in enumerate(range(0, total_size, UPLOAD_CHUNK_SIZE)):
1172
+ chunk = data[start : start + UPLOAD_CHUNK_SIZE]
1173
+ chunk_response = await self._send_request(
1174
+ APP_CMD_UPLOAD_MEDIA_CHUNK,
1175
+ {
1176
+ "upload_id": upload_id,
1177
+ # Match the official SDK implementation, which currently uses 0-based chunk indexes.
1178
+ "chunk_index": chunk_index,
1179
+ "base64_data": base64.b64encode(chunk).decode("ascii"),
1180
+ },
1181
+ )
1182
+ self._raise_for_wecom_error(chunk_response, f"media upload chunk {chunk_index}")
1183
+
1184
+ finish_response = await self._send_request(
1185
+ APP_CMD_UPLOAD_MEDIA_FINISH,
1186
+ {"upload_id": upload_id},
1187
+ )
1188
+ self._raise_for_wecom_error(finish_response, "media upload finish")
1189
+
1190
+ finish_body = finish_response.get("body") if isinstance(finish_response.get("body"), dict) else {}
1191
+ media_id = str(finish_body.get("media_id") or "").strip()
1192
+ if not media_id:
1193
+ raise RuntimeError(f"media upload finish failed: missing media_id in response {finish_response}")
1194
+
1195
+ return {
1196
+ "type": str(finish_body.get("type") or media_type),
1197
+ "media_id": media_id,
1198
+ "created_at": finish_body.get("created_at"),
1199
+ }
1200
+
1201
+ async def _send_media_message(self, chat_id: str, media_type: str, media_id: str) -> Dict[str, Any]:
1202
+ response = await self._send_request(
1203
+ APP_CMD_SEND,
1204
+ {
1205
+ "chatid": chat_id,
1206
+ "msgtype": media_type,
1207
+ media_type: {"media_id": media_id},
1208
+ },
1209
+ )
1210
+ self._raise_for_wecom_error(response, "send media message")
1211
+ return response
1212
+
1213
+ async def _send_reply_markdown(self, reply_req_id: str, content: str) -> Dict[str, Any]:
1214
+ response = await self._send_reply_request(
1215
+ reply_req_id,
1216
+ {
1217
+ "msgtype": "markdown",
1218
+ "markdown": {"content": content[:self.MAX_MESSAGE_LENGTH]},
1219
+ },
1220
+ )
1221
+ self._raise_for_wecom_error(response, "send reply markdown")
1222
+ return response
1223
+
1224
+ async def _send_reply_media_message(
1225
+ self,
1226
+ reply_req_id: str,
1227
+ media_type: str,
1228
+ media_id: str,
1229
+ ) -> Dict[str, Any]:
1230
+ response = await self._send_reply_request(
1231
+ reply_req_id,
1232
+ {
1233
+ "msgtype": media_type,
1234
+ media_type: {"media_id": media_id},
1235
+ },
1236
+ )
1237
+ self._raise_for_wecom_error(response, "send reply media message")
1238
+ return response
1239
+
1240
+ async def _send_followup_markdown(
1241
+ self,
1242
+ chat_id: str,
1243
+ content: str,
1244
+ reply_to: Optional[str] = None,
1245
+ ) -> Optional[SendResult]:
1246
+ if not content:
1247
+ return None
1248
+ result = await self.send(chat_id=chat_id, content=content, reply_to=reply_to)
1249
+ if not result.success:
1250
+ logger.warning("[%s] Follow-up markdown send failed: %s", self.name, result.error)
1251
+ return result
1252
+
1253
+ async def _send_media_source(
1254
+ self,
1255
+ chat_id: str,
1256
+ media_source: str,
1257
+ caption: Optional[str] = None,
1258
+ file_name: Optional[str] = None,
1259
+ reply_to: Optional[str] = None,
1260
+ ) -> SendResult:
1261
+ if not chat_id:
1262
+ return SendResult(success=False, error="chat_id is required")
1263
+
1264
+ try:
1265
+ prepared = await self._prepare_outbound_media(media_source, file_name=file_name)
1266
+ except FileNotFoundError as exc:
1267
+ return SendResult(success=False, error=str(exc))
1268
+ except Exception as exc:
1269
+ logger.error("[%s] Failed to prepare outbound media %s: %s", self.name, media_source, exc)
1270
+ return SendResult(success=False, error=str(exc))
1271
+
1272
+ if prepared["rejected"]:
1273
+ await self._send_followup_markdown(
1274
+ chat_id,
1275
+ f"⚠️ {prepared['reject_reason']}",
1276
+ reply_to=reply_to,
1277
+ )
1278
+ return SendResult(success=False, error=prepared["reject_reason"])
1279
+
1280
+ reply_req_id = self._reply_req_id_for_message(reply_to)
1281
+ if not reply_req_id and chat_id in self._last_chat_req_ids:
1282
+ reply_req_id = self._last_chat_req_ids[chat_id]
1283
+
1284
+ try:
1285
+ upload_result = await self._upload_media_bytes(
1286
+ prepared["data"],
1287
+ prepared["final_type"],
1288
+ prepared["file_name"],
1289
+ )
1290
+ if reply_req_id:
1291
+ media_response = await self._send_reply_media_message(
1292
+ reply_req_id,
1293
+ prepared["final_type"],
1294
+ upload_result["media_id"],
1295
+ )
1296
+ else:
1297
+ media_response = await self._send_media_message(
1298
+ chat_id,
1299
+ prepared["final_type"],
1300
+ upload_result["media_id"],
1301
+ )
1302
+ except asyncio.TimeoutError:
1303
+ return SendResult(success=False, error="Timeout sending media to WeCom")
1304
+ except Exception as exc:
1305
+ logger.error("[%s] Failed to send media %s: %s", self.name, media_source, exc)
1306
+ return SendResult(success=False, error=str(exc))
1307
+
1308
+ caption_result = None
1309
+ downgrade_result = None
1310
+ if caption:
1311
+ caption_result = await self._send_followup_markdown(
1312
+ chat_id,
1313
+ caption,
1314
+ reply_to=reply_to,
1315
+ )
1316
+ if prepared["downgraded"] and prepared["downgrade_note"]:
1317
+ downgrade_result = await self._send_followup_markdown(
1318
+ chat_id,
1319
+ f"ℹ️ {prepared['downgrade_note']}",
1320
+ reply_to=reply_to,
1321
+ )
1322
+
1323
+ return SendResult(
1324
+ success=True,
1325
+ message_id=self._payload_req_id(media_response) or uuid.uuid4().hex[:12],
1326
+ raw_response={
1327
+ "upload": upload_result,
1328
+ "media": media_response,
1329
+ "caption": caption_result.raw_response if caption_result else None,
1330
+ "caption_error": caption_result.error if caption_result and not caption_result.success else None,
1331
+ "downgrade": downgrade_result.raw_response if downgrade_result else None,
1332
+ "downgrade_error": downgrade_result.error if downgrade_result and not downgrade_result.success else None,
1333
+ },
1334
+ )
1335
+
1336
+ async def send(
1337
+ self,
1338
+ chat_id: str,
1339
+ content: str,
1340
+ reply_to: Optional[str] = None,
1341
+ metadata: Optional[Dict[str, Any]] = None,
1342
+ ) -> SendResult:
1343
+ """Send markdown to a WeCom chat via proactive ``aibot_send_msg``."""
1344
+ del metadata
1345
+
1346
+ if not chat_id:
1347
+ return SendResult(success=False, error="chat_id is required")
1348
+
1349
+ try:
1350
+ reply_req_id = self._reply_req_id_for_message(reply_to)
1351
+
1352
+ if not reply_req_id and chat_id in self._last_chat_req_ids:
1353
+ reply_req_id = self._last_chat_req_ids[chat_id]
1354
+
1355
+ if reply_req_id:
1356
+ response = await self._send_reply_markdown(reply_req_id, content)
1357
+ else:
1358
+ response = await self._send_request(
1359
+ APP_CMD_SEND,
1360
+ {
1361
+ "chatid": chat_id,
1362
+ "msgtype": "markdown",
1363
+ "markdown": {"content": content[:self.MAX_MESSAGE_LENGTH]},
1364
+ },
1365
+ )
1366
+ except asyncio.TimeoutError:
1367
+ return SendResult(success=False, error="Timeout sending message to WeCom")
1368
+ except Exception as exc:
1369
+ logger.error("[%s] Send failed: %s", self.name, exc)
1370
+ return SendResult(success=False, error=str(exc))
1371
+
1372
+ error = self._response_error(response)
1373
+ if error:
1374
+ return SendResult(success=False, error=error)
1375
+
1376
+ return SendResult(
1377
+ success=True,
1378
+ message_id=self._payload_req_id(response) or uuid.uuid4().hex[:12],
1379
+ raw_response=response,
1380
+ )
1381
+
1382
+ async def send_image(
1383
+ self,
1384
+ chat_id: str,
1385
+ image_url: str,
1386
+ caption: Optional[str] = None,
1387
+ reply_to: Optional[str] = None,
1388
+ metadata: Optional[Dict[str, Any]] = None,
1389
+ ) -> SendResult:
1390
+ del metadata
1391
+
1392
+ result = await self._send_media_source(
1393
+ chat_id=chat_id,
1394
+ media_source=image_url,
1395
+ caption=caption,
1396
+ reply_to=reply_to,
1397
+ )
1398
+ if result.success or not self._looks_like_url(image_url):
1399
+ return result
1400
+
1401
+ logger.warning("[%s] Falling back to text send for image URL %s: %s", self.name, image_url, result.error)
1402
+ fallback_text = f"{caption}\n{image_url}" if caption else image_url
1403
+ return await self.send(chat_id=chat_id, content=fallback_text, reply_to=reply_to)
1404
+
1405
+ async def send_image_file(
1406
+ self,
1407
+ chat_id: str,
1408
+ image_path: str,
1409
+ caption: Optional[str] = None,
1410
+ reply_to: Optional[str] = None,
1411
+ **kwargs,
1412
+ ) -> SendResult:
1413
+ del kwargs
1414
+ return await self._send_media_source(
1415
+ chat_id=chat_id,
1416
+ media_source=image_path,
1417
+ caption=caption,
1418
+ reply_to=reply_to,
1419
+ )
1420
+
1421
+ async def send_document(
1422
+ self,
1423
+ chat_id: str,
1424
+ file_path: str,
1425
+ caption: Optional[str] = None,
1426
+ file_name: Optional[str] = None,
1427
+ reply_to: Optional[str] = None,
1428
+ **kwargs,
1429
+ ) -> SendResult:
1430
+ del kwargs
1431
+ return await self._send_media_source(
1432
+ chat_id=chat_id,
1433
+ media_source=file_path,
1434
+ caption=caption,
1435
+ file_name=file_name,
1436
+ reply_to=reply_to,
1437
+ )
1438
+
1439
+ async def send_voice(
1440
+ self,
1441
+ chat_id: str,
1442
+ audio_path: str,
1443
+ caption: Optional[str] = None,
1444
+ reply_to: Optional[str] = None,
1445
+ **kwargs,
1446
+ ) -> SendResult:
1447
+ del kwargs
1448
+ return await self._send_media_source(
1449
+ chat_id=chat_id,
1450
+ media_source=audio_path,
1451
+ caption=caption,
1452
+ reply_to=reply_to,
1453
+ )
1454
+
1455
+ async def send_video(
1456
+ self,
1457
+ chat_id: str,
1458
+ video_path: str,
1459
+ caption: Optional[str] = None,
1460
+ reply_to: Optional[str] = None,
1461
+ **kwargs,
1462
+ ) -> SendResult:
1463
+ del kwargs
1464
+ return await self._send_media_source(
1465
+ chat_id=chat_id,
1466
+ media_source=video_path,
1467
+ caption=caption,
1468
+ reply_to=reply_to,
1469
+ )
1470
+
1471
+ async def send_typing(self, chat_id: str, metadata=None) -> None:
1472
+ """WeCom does not expose typing indicators in this adapter."""
1473
+ del chat_id, metadata
1474
+
1475
+ async def get_chat_info(self, chat_id: str) -> Dict[str, Any]:
1476
+ """Return minimal chat info."""
1477
+ return {
1478
+ "name": chat_id,
1479
+ "type": "group" if chat_id and chat_id.lower().startswith("group") else "dm",
1480
+ }
1481
+
1482
+
1483
+ # ------------------------------------------------------------------
1484
+ # QR code scan flow for obtaining bot credentials
1485
+ # ------------------------------------------------------------------
1486
+
1487
+ _QR_GENERATE_URL = "https://work.weixin.qq.com/ai/qc/generate"
1488
+ _QR_QUERY_URL = "https://work.weixin.qq.com/ai/qc/query_result"
1489
+ _QR_CODE_PAGE = "https://work.weixin.qq.com/ai/qc/gen?source=hermes&scode="
1490
+ _QR_POLL_INTERVAL = 3 # seconds
1491
+ _QR_POLL_TIMEOUT = 300 # 5 minutes
1492
+
1493
+
1494
+ def qr_scan_for_bot_info(
1495
+ *,
1496
+ timeout_seconds: int = _QR_POLL_TIMEOUT,
1497
+ ) -> Optional[Dict[str, str]]:
1498
+ """Run the WeCom QR scan flow to obtain bot_id and secret.
1499
+
1500
+ Fetches a QR code from WeCom, renders it in the terminal, and polls
1501
+ until the user scans it or the timeout expires.
1502
+
1503
+ Returns ``{"bot_id": ..., "secret": ...}`` on success, ``None`` on
1504
+ failure or timeout.
1505
+
1506
+ Note: the ``work.weixin.qq.com/ai/qc/{generate,query_result}`` endpoints
1507
+ used here are not part of WeCom's public developer API — they back the
1508
+ admin-console web UI's bot-creation flow and may change without notice.
1509
+ The same pattern is used by the feishu/dingtalk QR setup wizards.
1510
+ """
1511
+ try:
1512
+ import urllib.request
1513
+ import urllib.parse
1514
+ except ImportError: # pragma: no cover
1515
+ logger.error("urllib is required for WeCom QR scan")
1516
+ return None
1517
+
1518
+ generate_url = f"{_QR_GENERATE_URL}?source=hermes"
1519
+
1520
+ # ── Step 1: Fetch QR code ──
1521
+ print(" Connecting to WeCom...", end="", flush=True)
1522
+ try:
1523
+ req = urllib.request.Request(generate_url, headers={"User-Agent": "HermesAgent/1.0"})
1524
+ with urllib.request.urlopen(req, timeout=15) as resp:
1525
+ raw = json.loads(resp.read().decode("utf-8"))
1526
+ except Exception as exc:
1527
+ logger.error("WeCom QR: failed to fetch QR code: %s", exc)
1528
+ print(f" failed: {exc}")
1529
+ return None
1530
+
1531
+ data = raw.get("data") or {}
1532
+ scode = str(data.get("scode") or "").strip()
1533
+ auth_url = str(data.get("auth_url") or "").strip()
1534
+
1535
+ if not scode or not auth_url:
1536
+ logger.error("WeCom QR: unexpected response format: %s", raw)
1537
+ print(" failed: unexpected response format")
1538
+ return None
1539
+
1540
+ print(" done.")
1541
+
1542
+ # ── Step 2: Render QR code in terminal ──
1543
+ print()
1544
+ qr_rendered = False
1545
+ try:
1546
+ import qrcode as _qrcode
1547
+ qr = _qrcode.QRCode()
1548
+ qr.add_data(auth_url)
1549
+ qr.make(fit=True)
1550
+ qr.print_ascii(invert=True)
1551
+ qr_rendered = True
1552
+ except ImportError:
1553
+ pass
1554
+ except Exception:
1555
+ pass
1556
+
1557
+ page_url = f"{_QR_CODE_PAGE}{urllib.parse.quote(scode)}"
1558
+ if qr_rendered:
1559
+ print(f"\n Scan the QR code above, or open this URL directly:\n {page_url}")
1560
+ else:
1561
+ print(f" Open this URL in WeCom on your phone:\n\n {page_url}\n")
1562
+ print(" Tip: pip install qrcode to display a scannable QR code here next time")
1563
+ print()
1564
+ print(" Fetching configuration results...", end="", flush=True)
1565
+
1566
+ # ── Step 3: Poll for result ──
1567
+ deadline = time.monotonic() + timeout_seconds
1568
+ query_url = f"{_QR_QUERY_URL}?scode={urllib.parse.quote(scode)}"
1569
+ poll_count = 0
1570
+
1571
+ while time.monotonic() < deadline:
1572
+ try:
1573
+ req = urllib.request.Request(query_url, headers={"User-Agent": "HermesAgent/1.0"})
1574
+ with urllib.request.urlopen(req, timeout=10) as resp:
1575
+ result = json.loads(resp.read().decode("utf-8"))
1576
+ except Exception as exc:
1577
+ logger.debug("WeCom QR poll error: %s", exc)
1578
+ time.sleep(_QR_POLL_INTERVAL)
1579
+ continue
1580
+
1581
+ poll_count += 1
1582
+ # Print a dot on every poll so progress is visible within 3s.
1583
+ print(".", end="", flush=True)
1584
+
1585
+ result_data = result.get("data") or {}
1586
+ status = str(result_data.get("status") or "").lower()
1587
+
1588
+ if status == "success":
1589
+ print() # newline after "Fetching configuration results..." dots
1590
+ bot_info = result_data.get("bot_info") or {}
1591
+ bot_id = str(bot_info.get("botid") or bot_info.get("bot_id") or "").strip()
1592
+ secret = str(bot_info.get("secret") or "").strip()
1593
+ if bot_id and secret:
1594
+ return {"bot_id": bot_id, "secret": secret}
1595
+ logger.warning(
1596
+ "WeCom QR: scan reported success but bot_info missing or incomplete: %s",
1597
+ result_data,
1598
+ )
1599
+ print(
1600
+ " QR scan reported success but no bot credentials were returned.\n"
1601
+ " This usually means the bot was not actually created on the WeCom side.\n"
1602
+ " Falling back to manual credential entry."
1603
+ )
1604
+ return None
1605
+
1606
+ time.sleep(_QR_POLL_INTERVAL)
1607
+
1608
+ print() # newline after dots
1609
+ print(f" QR scan timed out ({timeout_seconds // 60} minutes). Please try again.")
1610
+ return None