@clawpump/claw-agent 0.1.4 → 0.1.6

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 (1214) hide show
  1. package/agent/.dockerignore +67 -0
  2. package/agent/.envrc +1 -1
  3. package/agent/.gitattributes +8 -0
  4. package/agent/AGENTS.md +216 -4
  5. package/agent/CONTRIBUTING.md +46 -8
  6. package/agent/Dockerfile +78 -35
  7. package/agent/MANIFEST.in +2 -0
  8. package/agent/README.md +12 -5
  9. package/agent/README.ur-pk.md +261 -0
  10. package/agent/README.zh-CN.md +11 -8
  11. package/agent/SECURITY.md +5 -4
  12. package/agent/acp_adapter/provenance.py +127 -0
  13. package/agent/acp_adapter/server.py +112 -5
  14. package/agent/acp_adapter/session.py +1 -6
  15. package/agent/acp_registry/agent.json +2 -2
  16. package/agent/agent/account_usage.py +313 -1
  17. package/agent/agent/agent_init.py +140 -37
  18. package/agent/agent/agent_runtime_helpers.py +342 -83
  19. package/agent/agent/anthropic_adapter.py +320 -33
  20. package/agent/agent/auxiliary_client.py +525 -105
  21. package/agent/agent/background_review.py +157 -19
  22. package/agent/agent/bedrock_adapter.py +71 -6
  23. package/agent/agent/billing_view.py +295 -0
  24. package/agent/agent/chat_completion_helpers.py +229 -4
  25. package/agent/agent/codex_responses_adapter.py +86 -10
  26. package/agent/agent/codex_runtime.py +153 -1
  27. package/agent/agent/coding_context.py +738 -0
  28. package/agent/agent/context_compressor.py +392 -44
  29. package/agent/agent/context_references.py +34 -1
  30. package/agent/agent/conversation_compression.py +159 -22
  31. package/agent/agent/conversation_loop.py +643 -908
  32. package/agent/agent/copilot_acp_client.py +4 -11
  33. package/agent/agent/credential_pool.py +5 -3
  34. package/agent/agent/credits_tracker.py +794 -0
  35. package/agent/agent/curator.py +91 -18
  36. package/agent/agent/curator_backup.py +26 -10
  37. package/agent/agent/display.py +42 -1
  38. package/agent/agent/error_classifier.py +52 -3
  39. package/agent/agent/errors.py +3 -0
  40. package/agent/agent/file_safety.py +0 -17
  41. package/agent/agent/gemini_native_adapter.py +31 -1
  42. package/agent/agent/i18n.py +48 -4
  43. package/agent/agent/image_gen_provider.py +74 -5
  44. package/agent/agent/image_routing.py +29 -0
  45. package/agent/agent/insights.py +8 -17
  46. package/agent/agent/lsp/install.py +3 -0
  47. package/agent/agent/memory_manager.py +326 -31
  48. package/agent/agent/message_content.py +50 -0
  49. package/agent/agent/model_metadata.py +214 -3
  50. package/agent/agent/moonshot_schema.py +8 -1
  51. package/agent/agent/onboarding.py +60 -0
  52. package/agent/agent/prompt_builder.py +327 -37
  53. package/agent/agent/redact.py +1 -0
  54. package/agent/agent/runtime_cwd.py +34 -5
  55. package/agent/agent/secret_scope.py +205 -0
  56. package/agent/agent/secret_sources/bitwarden.py +34 -2
  57. package/agent/agent/skill_commands.py +90 -1
  58. package/agent/agent/skill_preprocessing.py +1 -0
  59. package/agent/agent/skill_utils.py +209 -36
  60. package/agent/agent/ssl_guard.py +94 -0
  61. package/agent/agent/system_prompt.py +133 -5
  62. package/agent/agent/tool_executor.py +496 -70
  63. package/agent/agent/transports/anthropic.py +83 -21
  64. package/agent/agent/transports/chat_completions.py +94 -5
  65. package/agent/agent/transports/codex.py +67 -2
  66. package/agent/agent/transports/codex_app_server.py +1 -0
  67. package/agent/agent/transports/codex_app_server_session.py +30 -0
  68. package/agent/agent/transports/types.py +12 -0
  69. package/agent/agent/turn_context.py +408 -0
  70. package/agent/agent/turn_finalizer.py +428 -0
  71. package/agent/agent/turn_retry_state.py +68 -0
  72. package/agent/agent/usage_pricing.py +3 -0
  73. package/agent/apps/bootstrap-installer/package.json +6 -5
  74. package/agent/apps/bootstrap-installer/src/routes/failure.tsx +12 -5
  75. package/agent/apps/bootstrap-installer/src/routes/progress.tsx +1 -3
  76. package/agent/apps/bootstrap-installer/src/store.ts +3 -2
  77. package/agent/apps/bootstrap-installer/src-tauri/src/bootstrap.rs +172 -7
  78. package/agent/apps/bootstrap-installer/src-tauri/src/events.rs +14 -1
  79. package/agent/apps/bootstrap-installer/src-tauri/src/paths.rs +29 -0
  80. package/agent/apps/bootstrap-installer/src-tauri/src/powershell.rs +93 -3
  81. package/agent/apps/bootstrap-installer/src-tauri/src/update.rs +695 -39
  82. package/agent/apps/bootstrap-installer/tsconfig.json +3 -4
  83. package/agent/apps/desktop/DESIGN.md +167 -0
  84. package/agent/apps/desktop/README.md +20 -16
  85. package/agent/apps/desktop/assets/icon.icns +0 -0
  86. package/agent/apps/desktop/assets/icon.ico +0 -0
  87. package/agent/apps/desktop/assets/icon.png +0 -0
  88. package/agent/apps/desktop/electron/backend-env.cjs +112 -0
  89. package/agent/apps/desktop/electron/backend-env.test.cjs +111 -0
  90. package/agent/apps/desktop/electron/backend-probes.test.cjs +3 -1
  91. package/agent/apps/desktop/electron/backend-ready.cjs +66 -0
  92. package/agent/apps/desktop/electron/bootstrap-platform.cjs +52 -0
  93. package/agent/apps/desktop/electron/bootstrap-platform.test.cjs +59 -1
  94. package/agent/apps/desktop/electron/bootstrap-runner.cjs +176 -38
  95. package/agent/apps/desktop/electron/bootstrap-runner.test.cjs +112 -1
  96. package/agent/apps/desktop/electron/connection-config.cjs +288 -0
  97. package/agent/apps/desktop/electron/connection-config.test.cjs +396 -0
  98. package/agent/apps/desktop/electron/dashboard-token.cjs +99 -0
  99. package/agent/apps/desktop/electron/dashboard-token.test.cjs +142 -0
  100. package/agent/apps/desktop/electron/desktop-uninstall.cjs +232 -0
  101. package/agent/apps/desktop/electron/desktop-uninstall.test.cjs +246 -0
  102. package/agent/apps/desktop/electron/entitlements.mac.inherit.plist +2 -0
  103. package/agent/apps/desktop/electron/fs-read-dir.cjs +109 -0
  104. package/agent/apps/desktop/electron/fs-read-dir.test.cjs +364 -0
  105. package/agent/apps/desktop/electron/gateway-ws-probe.cjs +188 -0
  106. package/agent/apps/desktop/electron/gateway-ws-probe.test.cjs +122 -0
  107. package/agent/apps/desktop/electron/git-root.cjs +54 -0
  108. package/agent/apps/desktop/electron/git-root.test.cjs +40 -0
  109. package/agent/apps/desktop/electron/git-worktrees.cjs +174 -0
  110. package/agent/apps/desktop/electron/hardening.cjs +123 -28
  111. package/agent/apps/desktop/electron/hardening.test.cjs +163 -0
  112. package/agent/apps/desktop/electron/main.cjs +3121 -331
  113. package/agent/apps/desktop/electron/oauth-net-request.cjs +20 -0
  114. package/agent/apps/desktop/electron/oauth-net-request.test.cjs +34 -0
  115. package/agent/apps/desktop/electron/preload.cjs +52 -2
  116. package/agent/apps/desktop/electron/session-windows.cjs +124 -0
  117. package/agent/apps/desktop/electron/session-windows.test.cjs +199 -0
  118. package/agent/apps/desktop/electron/update-rebuild.cjs +29 -0
  119. package/agent/apps/desktop/electron/update-rebuild.test.cjs +55 -0
  120. package/agent/apps/desktop/electron/update-remote.cjs +56 -0
  121. package/agent/apps/desktop/electron/update-remote.test.cjs +78 -0
  122. package/agent/apps/desktop/electron/vscode-marketplace.cjs +331 -0
  123. package/agent/apps/desktop/electron/vscode-marketplace.test.cjs +113 -0
  124. package/agent/apps/desktop/electron/windows-child-process.test.cjs +57 -0
  125. package/agent/apps/desktop/electron/windows-user-env.cjs +76 -0
  126. package/agent/apps/desktop/electron/windows-user-env.test.cjs +90 -0
  127. package/agent/apps/desktop/electron/workspace-cwd.cjs +38 -0
  128. package/agent/apps/desktop/electron/workspace-cwd.test.cjs +45 -0
  129. package/agent/apps/desktop/eslint.config.mjs +0 -3
  130. package/agent/apps/desktop/index.html +27 -2
  131. package/agent/apps/desktop/package.json +31 -11
  132. package/agent/apps/desktop/pr-assets/session-source-folders.png +0 -0
  133. package/agent/apps/desktop/public/apple-touch-icon.png +0 -0
  134. package/agent/apps/desktop/public/nous-girl.jpg +0 -0
  135. package/agent/apps/desktop/scripts/assert-dist-built.cjs +70 -0
  136. package/agent/apps/desktop/scripts/assert-dist-built.test.cjs +84 -0
  137. package/agent/apps/desktop/scripts/before-pack.cjs +78 -0
  138. package/agent/apps/desktop/scripts/before-pack.test.cjs +53 -0
  139. package/agent/apps/desktop/scripts/diag-scroll-reset.mjs +229 -0
  140. package/agent/apps/desktop/scripts/patch-electron-builder-mac-binary.cjs +64 -0
  141. package/agent/apps/desktop/scripts/run-electron-builder.cjs +57 -0
  142. package/agent/apps/desktop/src/app/agents/index.tsx +53 -45
  143. package/agent/apps/desktop/src/app/artifacts/index.tsx +102 -83
  144. package/agent/apps/desktop/src/app/chat/chat-drop-overlay.tsx +29 -8
  145. package/agent/apps/desktop/src/app/chat/chat-swap-overlay.tsx +47 -0
  146. package/agent/apps/desktop/src/app/chat/composer/attachments.tsx +81 -45
  147. package/agent/apps/desktop/src/app/chat/composer/completion-drawer.tsx +13 -24
  148. package/agent/apps/desktop/src/app/chat/composer/context-menu.tsx +138 -88
  149. package/agent/apps/desktop/src/app/chat/composer/controls.tsx +138 -90
  150. package/agent/apps/desktop/src/app/chat/composer/enter-submit-dom-race.test.tsx +218 -0
  151. package/agent/apps/desktop/src/app/chat/composer/focus.ts +32 -0
  152. package/agent/apps/desktop/src/app/chat/composer/help-hint.tsx +38 -25
  153. package/agent/apps/desktop/src/app/chat/composer/hooks/use-live-completion-adapter.ts +7 -0
  154. package/agent/apps/desktop/src/app/chat/composer/hooks/use-mic-recorder.ts +22 -12
  155. package/agent/apps/desktop/src/app/chat/composer/hooks/use-slash-completions.ts +142 -14
  156. package/agent/apps/desktop/src/app/chat/composer/hooks/use-voice-conversation.ts +14 -11
  157. package/agent/apps/desktop/src/app/chat/composer/hooks/use-voice-recorder.ts +9 -6
  158. package/agent/apps/desktop/src/app/chat/composer/ime-composition-dom-repro.test.tsx +108 -0
  159. package/agent/apps/desktop/src/app/chat/composer/index.tsx +930 -180
  160. package/agent/apps/desktop/src/app/chat/composer/inline-refs.ts +136 -32
  161. package/agent/apps/desktop/src/app/chat/composer/model-pill.tsx +86 -0
  162. package/agent/apps/desktop/src/app/chat/composer/queue-panel.tsx +54 -75
  163. package/agent/apps/desktop/src/app/chat/composer/rich-editor.test.ts +117 -1
  164. package/agent/apps/desktop/src/app/chat/composer/rich-editor.ts +117 -6
  165. package/agent/apps/desktop/src/app/chat/composer/slash-nav-dom-repro.test.tsx +186 -0
  166. package/agent/apps/desktop/src/app/chat/composer/status-stack/index.tsx +202 -0
  167. package/agent/apps/desktop/src/app/chat/composer/status-stack/status-row.tsx +155 -0
  168. package/agent/apps/desktop/src/app/chat/composer/text-utils.test.ts +104 -0
  169. package/agent/apps/desktop/src/app/chat/composer/text-utils.ts +37 -9
  170. package/agent/apps/desktop/src/app/chat/composer/trigger-popover.test.tsx +50 -0
  171. package/agent/apps/desktop/src/app/chat/composer/trigger-popover.tsx +105 -40
  172. package/agent/apps/desktop/src/app/chat/composer/types.ts +5 -0
  173. package/agent/apps/desktop/src/app/chat/composer/url-dialog.tsx +11 -15
  174. package/agent/apps/desktop/src/app/chat/composer/voice-activity.tsx +8 -4
  175. package/agent/apps/desktop/src/app/chat/hooks/use-composer-actions.test.ts +57 -0
  176. package/agent/apps/desktop/src/app/chat/hooks/use-composer-actions.ts +70 -16
  177. package/agent/apps/desktop/src/app/chat/hooks/use-file-drop-zone.ts +52 -16
  178. package/agent/apps/desktop/src/app/chat/index.tsx +234 -81
  179. package/agent/apps/desktop/src/app/chat/perf-probe.tsx +69 -21
  180. package/agent/apps/desktop/src/app/chat/right-rail/preview-console.tsx +44 -40
  181. package/agent/apps/desktop/src/app/chat/right-rail/preview-file.tsx +71 -25
  182. package/agent/apps/desktop/src/app/chat/right-rail/preview-pane.test.tsx +40 -1
  183. package/agent/apps/desktop/src/app/chat/right-rail/preview-pane.tsx +55 -53
  184. package/agent/apps/desktop/src/app/chat/right-rail/preview.tsx +35 -17
  185. package/agent/apps/desktop/src/app/chat/scroll-to-bottom-button.test.tsx +67 -0
  186. package/agent/apps/desktop/src/app/chat/scroll-to-bottom-button.tsx +74 -0
  187. package/agent/apps/desktop/src/app/chat/sidebar/cron-jobs-section.tsx +356 -0
  188. package/agent/apps/desktop/src/app/chat/sidebar/index.tsx +1189 -364
  189. package/agent/apps/desktop/src/app/chat/sidebar/load-more-row.tsx +30 -0
  190. package/agent/apps/desktop/src/app/chat/sidebar/order.test.ts +21 -0
  191. package/agent/apps/desktop/src/app/chat/sidebar/order.ts +17 -0
  192. package/agent/apps/desktop/src/app/chat/sidebar/profile-switcher.tsx +524 -0
  193. package/agent/apps/desktop/src/app/chat/sidebar/session-actions-menu.tsx +80 -45
  194. package/agent/apps/desktop/src/app/chat/sidebar/session-row.tsx +120 -25
  195. package/agent/apps/desktop/src/app/chat/sidebar/virtual-session-list.tsx +7 -13
  196. package/agent/apps/desktop/src/app/chat/sidebar/workspace-groups.test.ts +149 -0
  197. package/agent/apps/desktop/src/app/chat/sidebar/workspace-groups.ts +326 -0
  198. package/agent/apps/desktop/src/app/chat/thread-loading.ts +7 -2
  199. package/agent/apps/desktop/src/app/command-center/index.tsx +320 -581
  200. package/agent/apps/desktop/src/app/command-palette/index.tsx +681 -0
  201. package/agent/apps/desktop/src/app/command-palette/marketplace-theme-page.tsx +157 -0
  202. package/agent/apps/desktop/src/app/cron/index.tsx +392 -324
  203. package/agent/apps/desktop/src/app/cron/job-state.ts +29 -0
  204. package/agent/apps/desktop/src/app/desktop-controller.tsx +618 -123
  205. package/agent/apps/desktop/src/app/floating-hud.ts +22 -0
  206. package/agent/apps/desktop/src/app/gateway/hooks/use-gateway-boot.test.tsx +265 -0
  207. package/agent/apps/desktop/src/app/gateway/hooks/use-gateway-boot.ts +260 -14
  208. package/agent/apps/desktop/src/app/gateway/hooks/use-gateway-request.ts +48 -4
  209. package/agent/apps/desktop/src/app/hooks/use-keybinds.ts +270 -0
  210. package/agent/apps/desktop/src/app/hooks/use-refresh-hotkey.ts +45 -0
  211. package/agent/apps/desktop/src/app/layout-constants.ts +19 -0
  212. package/agent/apps/desktop/src/app/messaging/index.tsx +136 -241
  213. package/agent/apps/desktop/src/app/messaging/platform-icon.tsx +95 -0
  214. package/agent/apps/desktop/src/app/model-visibility-overlay.tsx +31 -0
  215. package/agent/apps/desktop/src/app/overlays/overlay-search-input.tsx +18 -62
  216. package/agent/apps/desktop/src/app/overlays/overlay-split-layout.tsx +59 -7
  217. package/agent/apps/desktop/src/app/overlays/overlay-view.tsx +9 -5
  218. package/agent/apps/desktop/src/app/page-search-shell.tsx +42 -20
  219. package/agent/apps/desktop/src/app/profiles/create-profile-dialog.tsx +165 -0
  220. package/agent/apps/desktop/src/app/profiles/delete-profile-dialog.tsx +65 -0
  221. package/agent/apps/desktop/src/app/profiles/index.tsx +174 -199
  222. package/agent/apps/desktop/src/app/profiles/rename-profile-dialog.tsx +125 -0
  223. package/agent/apps/desktop/src/app/right-sidebar/files/dnd-manager.ts +27 -0
  224. package/agent/apps/desktop/src/app/right-sidebar/files/ipc.test.ts +100 -0
  225. package/agent/apps/desktop/src/app/right-sidebar/files/ipc.ts +12 -18
  226. package/agent/apps/desktop/src/app/right-sidebar/files/remote-picker.tsx +177 -0
  227. package/agent/apps/desktop/src/app/right-sidebar/files/tree.tsx +35 -21
  228. package/agent/apps/desktop/src/app/right-sidebar/files/use-project-tree.test.ts +75 -3
  229. package/agent/apps/desktop/src/app/right-sidebar/files/use-project-tree.ts +152 -5
  230. package/agent/apps/desktop/src/app/right-sidebar/index.test.tsx +75 -0
  231. package/agent/apps/desktop/src/app/right-sidebar/index.tsx +166 -129
  232. package/agent/apps/desktop/src/app/right-sidebar/store.ts +19 -4
  233. package/agent/apps/desktop/src/app/right-sidebar/terminal/buffer.ts +65 -0
  234. package/agent/apps/desktop/src/app/right-sidebar/terminal/index.tsx +29 -34
  235. package/agent/apps/desktop/src/app/right-sidebar/terminal/persistent.tsx +18 -6
  236. package/agent/apps/desktop/src/app/right-sidebar/terminal/selection.ts +93 -32
  237. package/agent/apps/desktop/src/app/right-sidebar/terminal/use-terminal-session.ts +381 -119
  238. package/agent/apps/desktop/src/app/routes.ts +9 -0
  239. package/agent/apps/desktop/src/app/session/hooks/use-cwd-actions.ts +17 -7
  240. package/agent/apps/desktop/src/app/session/hooks/use-message-stream.ts +365 -47
  241. package/agent/apps/desktop/src/app/session/hooks/use-model-controls.test.tsx +198 -0
  242. package/agent/apps/desktop/src/app/session/hooks/use-model-controls.ts +70 -34
  243. package/agent/apps/desktop/src/app/session/hooks/use-prompt-actions.test.tsx +1061 -0
  244. package/agent/apps/desktop/src/app/session/hooks/use-prompt-actions.ts +1143 -165
  245. package/agent/apps/desktop/src/app/session/hooks/use-route-resume.test.tsx +341 -2
  246. package/agent/apps/desktop/src/app/session/hooks/use-route-resume.ts +176 -5
  247. package/agent/apps/desktop/src/app/session/hooks/use-session-actions.test.tsx +259 -0
  248. package/agent/apps/desktop/src/app/session/hooks/use-session-actions.ts +452 -149
  249. package/agent/apps/desktop/src/app/session/hooks/use-session-state-cache.test.tsx +327 -0
  250. package/agent/apps/desktop/src/app/session/hooks/use-session-state-cache.ts +133 -4
  251. package/agent/apps/desktop/src/app/session-picker-overlay.tsx +32 -0
  252. package/agent/apps/desktop/src/app/session-switcher.tsx +107 -0
  253. package/agent/apps/desktop/src/app/settings/about-settings.tsx +45 -36
  254. package/agent/apps/desktop/src/app/settings/appearance-settings.tsx +243 -162
  255. package/agent/apps/desktop/src/app/settings/config-settings.tsx +86 -66
  256. package/agent/apps/desktop/src/app/settings/constants.ts +459 -122
  257. package/agent/apps/desktop/src/app/settings/credential-key-ui.tsx +373 -0
  258. package/agent/apps/desktop/src/app/settings/env-credentials.tsx +198 -0
  259. package/agent/apps/desktop/src/app/settings/env-var-actions-menu.tsx +136 -0
  260. package/agent/apps/desktop/src/app/settings/field-copy.ts +56 -0
  261. package/agent/apps/desktop/src/app/settings/gateway-settings.tsx +385 -72
  262. package/agent/apps/desktop/src/app/settings/helpers.test.ts +156 -1
  263. package/agent/apps/desktop/src/app/settings/helpers.ts +30 -2
  264. package/agent/apps/desktop/src/app/settings/index.tsx +118 -84
  265. package/agent/apps/desktop/src/app/settings/keys-settings.tsx +62 -419
  266. package/agent/apps/desktop/src/app/settings/mcp-settings.tsx +65 -60
  267. package/agent/apps/desktop/src/app/settings/model-settings.test.tsx +129 -5
  268. package/agent/apps/desktop/src/app/settings/model-settings.tsx +370 -65
  269. package/agent/apps/desktop/src/app/settings/notifications-settings.tsx +150 -0
  270. package/agent/apps/desktop/src/app/settings/primitives.tsx +5 -11
  271. package/agent/apps/desktop/src/app/settings/provider-config-panel.test.tsx +142 -0
  272. package/agent/apps/desktop/src/app/settings/provider-config-panel.tsx +182 -0
  273. package/agent/apps/desktop/src/app/settings/providers-settings.test.tsx +171 -0
  274. package/agent/apps/desktop/src/app/settings/providers-settings.tsx +471 -0
  275. package/agent/apps/desktop/src/app/settings/sessions-settings.tsx +183 -71
  276. package/agent/apps/desktop/src/app/settings/toolset-config-panel.test.tsx +135 -1
  277. package/agent/apps/desktop/src/app/settings/toolset-config-panel.tsx +180 -57
  278. package/agent/apps/desktop/src/app/settings/types.ts +9 -6
  279. package/agent/apps/desktop/src/app/settings/uninstall-section.tsx +185 -0
  280. package/agent/apps/desktop/src/app/settings/use-deep-link-highlight.ts +60 -0
  281. package/agent/apps/desktop/src/app/shell/app-shell.tsx +59 -13
  282. package/agent/apps/desktop/src/app/shell/gateway-menu-panel.tsx +37 -32
  283. package/agent/apps/desktop/src/app/shell/hooks/use-overlay-routing.ts +6 -3
  284. package/agent/apps/desktop/src/app/shell/hooks/use-statusbar-items.tsx +212 -53
  285. package/agent/apps/desktop/src/app/shell/keybind-panel.tsx +215 -0
  286. package/agent/apps/desktop/src/app/shell/model-edit-submenu.test.tsx +84 -0
  287. package/agent/apps/desktop/src/app/shell/model-edit-submenu.tsx +244 -0
  288. package/agent/apps/desktop/src/app/shell/model-menu-panel.tsx +392 -0
  289. package/agent/apps/desktop/src/app/shell/statusbar-controls.tsx +23 -33
  290. package/agent/apps/desktop/src/app/shell/titlebar-controls.tsx +79 -95
  291. package/agent/apps/desktop/src/app/shell/titlebar.ts +8 -2
  292. package/agent/apps/desktop/src/app/skills/index.test.tsx +11 -0
  293. package/agent/apps/desktop/src/app/skills/index.tsx +79 -64
  294. package/agent/apps/desktop/src/app/types.ts +85 -0
  295. package/agent/apps/desktop/src/app/updates-overlay.tsx +110 -105
  296. package/agent/apps/desktop/src/components/assistant-ui/ansi-text.tsx +34 -0
  297. package/agent/apps/desktop/src/components/assistant-ui/block-direction.test.tsx +129 -0
  298. package/agent/apps/desktop/src/components/assistant-ui/clarify-tool.tsx +102 -81
  299. package/agent/apps/desktop/src/components/assistant-ui/directive-text.tsx +92 -15
  300. package/agent/apps/desktop/src/components/assistant-ui/markdown-text.test.ts +38 -0
  301. package/agent/apps/desktop/src/components/assistant-ui/markdown-text.tsx +304 -45
  302. package/agent/apps/desktop/src/components/assistant-ui/message-render-boundary.test.tsx +80 -0
  303. package/agent/apps/desktop/src/components/assistant-ui/message-render-boundary.tsx +48 -0
  304. package/agent/apps/desktop/src/components/assistant-ui/streaming.test.tsx +142 -90
  305. package/agent/apps/desktop/src/components/assistant-ui/thread-list.tsx +337 -0
  306. package/agent/apps/desktop/src/components/assistant-ui/thread.tsx +667 -190
  307. package/agent/apps/desktop/src/components/assistant-ui/tool-approval-group.test.tsx +299 -0
  308. package/agent/apps/desktop/src/components/assistant-ui/tool-approval.test.tsx +133 -0
  309. package/agent/apps/desktop/src/components/assistant-ui/tool-approval.tsx +239 -0
  310. package/agent/apps/desktop/src/components/assistant-ui/tool-fallback-model.test.ts +31 -0
  311. package/agent/apps/desktop/src/components/assistant-ui/tool-fallback-model.ts +152 -134
  312. package/agent/apps/desktop/src/components/assistant-ui/tool-fallback.tsx +142 -150
  313. package/agent/apps/desktop/src/components/assistant-ui/tooltip-icon-button.tsx +14 -12
  314. package/agent/apps/desktop/src/components/assistant-ui/user-message-edit.test.tsx +141 -0
  315. package/agent/apps/desktop/src/components/assistant-ui/user-message-text.tsx +152 -0
  316. package/agent/apps/desktop/src/components/boot-failure-overlay.tsx +150 -33
  317. package/agent/apps/desktop/src/components/boot-failure-reauth.test.ts +100 -0
  318. package/agent/apps/desktop/src/components/boot-failure-reauth.ts +81 -0
  319. package/agent/apps/desktop/src/components/brand-mark.tsx +19 -0
  320. package/agent/apps/desktop/src/components/chat/code-card.tsx +1 -1
  321. package/agent/apps/desktop/src/components/chat/composer-dock.ts +31 -0
  322. package/agent/apps/desktop/src/components/chat/diff-lines.tsx +1 -1
  323. package/agent/apps/desktop/src/components/chat/disclosure-row.tsx +13 -3
  324. package/agent/apps/desktop/src/components/chat/expandable-block.tsx +52 -0
  325. package/agent/apps/desktop/src/components/chat/generated-image-result.tsx +174 -0
  326. package/agent/apps/desktop/src/components/chat/image-generation-placeholder.tsx +70 -37
  327. package/agent/apps/desktop/src/components/chat/intro.tsx +8 -7
  328. package/agent/apps/desktop/src/components/chat/preview-attachment.tsx +4 -2
  329. package/agent/apps/desktop/src/components/chat/shiki-highlighter.test.ts +37 -0
  330. package/agent/apps/desktop/src/components/chat/shiki-highlighter.tsx +96 -22
  331. package/agent/apps/desktop/src/components/chat/status-row.tsx +70 -0
  332. package/agent/apps/desktop/src/components/chat/status-section.tsx +42 -0
  333. package/agent/apps/desktop/src/components/chat/terminal-output.tsx +54 -0
  334. package/agent/apps/desktop/src/components/chat/zoomable-image.tsx +70 -109
  335. package/agent/apps/desktop/src/components/desktop-install-overlay.tsx +154 -84
  336. package/agent/apps/desktop/src/components/desktop-onboarding-overlay.test.tsx +38 -8
  337. package/agent/apps/desktop/src/components/desktop-onboarding-overlay.tsx +789 -233
  338. package/agent/apps/desktop/src/components/error-boundary.tsx +77 -0
  339. package/agent/apps/desktop/src/components/gateway-connecting-overlay.test.tsx +144 -0
  340. package/agent/apps/desktop/src/components/gateway-connecting-overlay.tsx +7 -1
  341. package/agent/apps/desktop/src/components/haptics-provider.tsx +24 -0
  342. package/agent/apps/desktop/src/components/language-switcher.test.tsx +53 -0
  343. package/agent/apps/desktop/src/components/language-switcher.tsx +175 -0
  344. package/agent/apps/desktop/src/components/model-picker.tsx +42 -40
  345. package/agent/apps/desktop/src/components/model-visibility-dialog.tsx +166 -0
  346. package/agent/apps/desktop/src/components/notifications.tsx +48 -27
  347. package/agent/apps/desktop/src/components/pane-shell/index.ts +1 -1
  348. package/agent/apps/desktop/src/components/pane-shell/pane-shell.tsx +146 -9
  349. package/agent/apps/desktop/src/components/prompt-overlays.tsx +234 -0
  350. package/agent/apps/desktop/src/components/session-picker.tsx +108 -0
  351. package/agent/apps/desktop/src/components/ui/action-status.tsx +25 -0
  352. package/agent/apps/desktop/src/components/ui/badge.tsx +35 -0
  353. package/agent/apps/desktop/src/components/ui/button.tsx +37 -13
  354. package/agent/apps/desktop/src/components/ui/confirm-dialog.tsx +109 -0
  355. package/agent/apps/desktop/src/components/ui/control.ts +25 -0
  356. package/agent/apps/desktop/src/components/ui/copy-button.test.tsx +36 -0
  357. package/agent/apps/desktop/src/components/ui/copy-button.tsx +38 -27
  358. package/agent/apps/desktop/src/components/ui/dialog.tsx +39 -11
  359. package/agent/apps/desktop/src/components/ui/dropdown-menu.tsx +98 -24
  360. package/agent/apps/desktop/src/components/ui/error-state.tsx +50 -0
  361. package/agent/apps/desktop/src/components/ui/fade-text.tsx +9 -2
  362. package/agent/apps/desktop/src/components/ui/{braille-spinner.tsx → glyph-spinner.tsx} +15 -13
  363. package/agent/apps/desktop/src/components/ui/input.tsx +5 -2
  364. package/agent/apps/desktop/src/components/ui/kbd.tsx +83 -12
  365. package/agent/apps/desktop/src/components/ui/log-view.tsx +19 -0
  366. package/agent/apps/desktop/src/components/ui/pagination.tsx +12 -5
  367. package/agent/apps/desktop/src/components/ui/popover.tsx +44 -0
  368. package/agent/apps/desktop/src/components/ui/search-field.tsx +80 -0
  369. package/agent/apps/desktop/src/components/ui/segmented-control.tsx +51 -0
  370. package/agent/apps/desktop/src/components/ui/select.tsx +10 -3
  371. package/agent/apps/desktop/src/components/ui/sheet.tsx +8 -2
  372. package/agent/apps/desktop/src/components/ui/sidebar.tsx +18 -25
  373. package/agent/apps/desktop/src/components/ui/switch.tsx +38 -15
  374. package/agent/apps/desktop/src/components/ui/textarea.tsx +4 -11
  375. package/agent/apps/desktop/src/components/ui/tool-icon.tsx +65 -0
  376. package/agent/apps/desktop/src/components/ui/tooltip.tsx +31 -4
  377. package/agent/apps/desktop/src/fonts/JetBrainsMono-Bold.woff2 +0 -0
  378. package/agent/apps/desktop/src/fonts/JetBrainsMono-Italic.woff2 +0 -0
  379. package/agent/apps/desktop/src/fonts/JetBrainsMono-Regular.woff2 +0 -0
  380. package/agent/apps/desktop/src/global.d.ts +181 -4
  381. package/agent/apps/desktop/src/hermes.test.ts +60 -0
  382. package/agent/apps/desktop/src/hermes.ts +190 -13
  383. package/agent/apps/desktop/src/hooks/use-image-download.ts +85 -0
  384. package/agent/apps/desktop/src/hooks/use-resize-observer.ts +13 -4
  385. package/agent/apps/desktop/src/hooks/use-worktree-info.ts +68 -0
  386. package/agent/apps/desktop/src/i18n/catalog.ts +12 -0
  387. package/agent/apps/desktop/src/i18n/context.test.tsx +232 -0
  388. package/agent/apps/desktop/src/i18n/context.tsx +183 -0
  389. package/agent/apps/desktop/src/i18n/define-locale.ts +41 -0
  390. package/agent/apps/desktop/src/i18n/en.ts +1921 -0
  391. package/agent/apps/desktop/src/i18n/index.ts +20 -0
  392. package/agent/apps/desktop/src/i18n/ja.ts +2053 -0
  393. package/agent/apps/desktop/src/i18n/languages.test.ts +43 -0
  394. package/agent/apps/desktop/src/i18n/languages.ts +86 -0
  395. package/agent/apps/desktop/src/i18n/runtime.test.ts +75 -0
  396. package/agent/apps/desktop/src/i18n/runtime.ts +53 -0
  397. package/agent/apps/desktop/src/i18n/types.ts +1559 -0
  398. package/agent/apps/desktop/src/i18n/zh-hant.ts +1992 -0
  399. package/agent/apps/desktop/src/i18n/zh.ts +2099 -0
  400. package/agent/apps/desktop/src/lib/ansi.test.ts +123 -0
  401. package/agent/apps/desktop/src/lib/ansi.ts +186 -0
  402. package/agent/apps/desktop/src/lib/chat-messages.test.ts +79 -0
  403. package/agent/apps/desktop/src/lib/chat-messages.ts +68 -29
  404. package/agent/apps/desktop/src/lib/chat-runtime.test.ts +65 -1
  405. package/agent/apps/desktop/src/lib/chat-runtime.ts +39 -3
  406. package/agent/apps/desktop/src/lib/completion-sound.ts +519 -0
  407. package/agent/apps/desktop/src/lib/desktop-fs.test.ts +116 -0
  408. package/agent/apps/desktop/src/lib/desktop-fs.ts +113 -0
  409. package/agent/apps/desktop/src/lib/desktop-slash-commands.test.ts +89 -6
  410. package/agent/apps/desktop/src/lib/desktop-slash-commands.ts +270 -131
  411. package/agent/apps/desktop/src/lib/external-link.test.tsx +27 -0
  412. package/agent/apps/desktop/src/lib/external-link.tsx +9 -2
  413. package/agent/apps/desktop/src/lib/gateway-events.test.ts +27 -0
  414. package/agent/apps/desktop/src/lib/gateway-events.ts +16 -0
  415. package/agent/apps/desktop/src/lib/gateway-ws-url.test.ts +78 -0
  416. package/agent/apps/desktop/src/lib/gateway-ws-url.ts +91 -0
  417. package/agent/apps/desktop/src/lib/generated-images.test.ts +97 -0
  418. package/agent/apps/desktop/src/lib/generated-images.ts +116 -0
  419. package/agent/apps/desktop/src/lib/haptics.ts +17 -0
  420. package/agent/apps/desktop/src/lib/icons.ts +10 -2
  421. package/agent/apps/desktop/src/lib/keybinds/actions.ts +137 -0
  422. package/agent/apps/desktop/src/lib/keybinds/combo.test.ts +86 -0
  423. package/agent/apps/desktop/src/lib/keybinds/combo.ts +195 -0
  424. package/agent/apps/desktop/src/lib/local-preview.ts +23 -2
  425. package/agent/apps/desktop/src/lib/markdown-preprocess.ts +20 -7
  426. package/agent/apps/desktop/src/lib/media.remote.test.ts +90 -0
  427. package/agent/apps/desktop/src/lib/media.ts +40 -1
  428. package/agent/apps/desktop/src/lib/model-status-label.test.ts +59 -0
  429. package/agent/apps/desktop/src/lib/model-status-label.ts +122 -0
  430. package/agent/apps/desktop/src/lib/mutable-ref.ts +6 -0
  431. package/agent/apps/desktop/src/lib/profile-color.ts +58 -0
  432. package/agent/apps/desktop/src/lib/query-client.ts +13 -0
  433. package/agent/apps/desktop/src/lib/remend-tail.test.ts +105 -0
  434. package/agent/apps/desktop/src/lib/remend-tail.ts +108 -0
  435. package/agent/apps/desktop/src/lib/session-export.ts +6 -3
  436. package/agent/apps/desktop/src/lib/session-ids.test.ts +44 -0
  437. package/agent/apps/desktop/src/lib/session-ids.ts +26 -0
  438. package/agent/apps/desktop/src/lib/session-search.test.ts +66 -0
  439. package/agent/apps/desktop/src/lib/session-search.ts +21 -0
  440. package/agent/apps/desktop/src/lib/session-source.ts +126 -0
  441. package/agent/apps/desktop/src/lib/storage.test.ts +25 -0
  442. package/agent/apps/desktop/src/lib/storage.ts +35 -1
  443. package/agent/apps/desktop/src/lib/todos.test.ts +46 -1
  444. package/agent/apps/desktop/src/lib/todos.ts +37 -0
  445. package/agent/apps/desktop/src/lib/tool-result-summary.ts +5 -1
  446. package/agent/apps/desktop/src/lib/update-copy.test.ts +38 -0
  447. package/agent/apps/desktop/src/lib/update-copy.ts +44 -0
  448. package/agent/apps/desktop/src/lib/use-enter-animation.ts +2 -2
  449. package/agent/apps/desktop/src/lib/yolo-session.ts +50 -0
  450. package/agent/apps/desktop/src/main.tsx +19 -19
  451. package/agent/apps/desktop/src/store/boot.ts +4 -3
  452. package/agent/apps/desktop/src/store/clarify.test.ts +81 -0
  453. package/agent/apps/desktop/src/store/clarify.ts +50 -13
  454. package/agent/apps/desktop/src/store/command-palette.ts +20 -0
  455. package/agent/apps/desktop/src/store/compaction.test.ts +53 -0
  456. package/agent/apps/desktop/src/store/compaction.ts +38 -0
  457. package/agent/apps/desktop/src/store/completion-sound.ts +32 -0
  458. package/agent/apps/desktop/src/store/composer-input-history.test.ts +147 -0
  459. package/agent/apps/desktop/src/store/composer-input-history.ts +158 -0
  460. package/agent/apps/desktop/src/store/composer-queue.test.ts +68 -0
  461. package/agent/apps/desktop/src/store/composer-queue.ts +76 -0
  462. package/agent/apps/desktop/src/store/composer-status.test.ts +99 -0
  463. package/agent/apps/desktop/src/store/composer-status.ts +277 -0
  464. package/agent/apps/desktop/src/store/composer.test.ts +106 -0
  465. package/agent/apps/desktop/src/store/composer.ts +116 -0
  466. package/agent/apps/desktop/src/store/cron.ts +19 -0
  467. package/agent/apps/desktop/src/store/gateway.ts +280 -6
  468. package/agent/apps/desktop/src/store/keybinds.ts +143 -0
  469. package/agent/apps/desktop/src/store/layout.ts +107 -9
  470. package/agent/apps/desktop/src/store/model-presets.test.ts +51 -0
  471. package/agent/apps/desktop/src/store/model-presets.ts +86 -0
  472. package/agent/apps/desktop/src/store/model-visibility.test.ts +99 -0
  473. package/agent/apps/desktop/src/store/model-visibility.ts +161 -0
  474. package/agent/apps/desktop/src/store/native-notifications.test.ts +192 -0
  475. package/agent/apps/desktop/src/store/native-notifications.ts +203 -0
  476. package/agent/apps/desktop/src/store/notifications.ts +10 -7
  477. package/agent/apps/desktop/src/store/onboarding.test.ts +271 -1
  478. package/agent/apps/desktop/src/store/onboarding.ts +268 -38
  479. package/agent/apps/desktop/src/store/preview.ts +10 -1
  480. package/agent/apps/desktop/src/store/profile.test.ts +89 -0
  481. package/agent/apps/desktop/src/store/profile.ts +395 -0
  482. package/agent/apps/desktop/src/store/prompts.test.ts +127 -0
  483. package/agent/apps/desktop/src/store/prompts.ts +117 -0
  484. package/agent/apps/desktop/src/store/session-switcher.test.ts +115 -0
  485. package/agent/apps/desktop/src/store/session-switcher.ts +128 -0
  486. package/agent/apps/desktop/src/store/session-sync.ts +25 -0
  487. package/agent/apps/desktop/src/store/session.test.ts +268 -2
  488. package/agent/apps/desktop/src/store/session.ts +392 -18
  489. package/agent/apps/desktop/src/store/subagents.ts +3 -0
  490. package/agent/apps/desktop/src/store/system-actions.ts +48 -0
  491. package/agent/apps/desktop/src/store/thread-scroll.ts +58 -5
  492. package/agent/apps/desktop/src/store/todos.test.ts +47 -0
  493. package/agent/apps/desktop/src/store/todos.ts +64 -0
  494. package/agent/apps/desktop/src/store/tool-dismiss.ts +45 -0
  495. package/agent/apps/desktop/src/store/translucency.ts +38 -0
  496. package/agent/apps/desktop/src/store/updates.test.ts +187 -2
  497. package/agent/apps/desktop/src/store/updates.ts +268 -18
  498. package/agent/apps/desktop/src/store/windows.test.ts +143 -0
  499. package/agent/apps/desktop/src/store/windows.ts +115 -0
  500. package/agent/apps/desktop/src/styles.css +510 -119
  501. package/agent/apps/desktop/src/themes/color.ts +142 -0
  502. package/agent/apps/desktop/src/themes/context.tsx +128 -75
  503. package/agent/apps/desktop/src/themes/install.test.ts +119 -0
  504. package/agent/apps/desktop/src/themes/install.ts +95 -0
  505. package/agent/apps/desktop/src/themes/presets.test.ts +33 -0
  506. package/agent/apps/desktop/src/themes/presets.ts +13 -4
  507. package/agent/apps/desktop/src/themes/profile-theme.test.ts +41 -0
  508. package/agent/apps/desktop/src/themes/types.ts +35 -0
  509. package/agent/apps/desktop/src/themes/user-themes.test.ts +63 -0
  510. package/agent/apps/desktop/src/themes/user-themes.ts +122 -0
  511. package/agent/apps/desktop/src/themes/vscode.test.ts +171 -0
  512. package/agent/apps/desktop/src/themes/vscode.ts +343 -0
  513. package/agent/apps/desktop/src/types/hermes.ts +138 -1
  514. package/agent/apps/desktop/tsconfig.json +2 -2
  515. package/agent/apps/desktop/vite.config.ts +18 -0
  516. package/agent/apps/shared/package.json +1 -1
  517. package/agent/apps/shared/src/json-rpc-gateway.ts +63 -2
  518. package/agent/apps/shared/tsconfig.json +2 -2
  519. package/agent/cli-config.yaml.example +78 -1
  520. package/agent/cli.py +2294 -3146
  521. package/agent/cron/blueprint_catalog.py +713 -0
  522. package/agent/cron/jobs.py +226 -110
  523. package/agent/cron/scheduler.py +468 -193
  524. package/agent/cron/scheduler_provider.py +177 -0
  525. package/agent/cron/scripts/__init__.py +1 -0
  526. package/agent/cron/scripts/classify_items.py +226 -0
  527. package/agent/cron/suggestion_catalog.py +154 -0
  528. package/agent/cron/suggestions.py +257 -0
  529. package/agent/docs/chronos-managed-cron-contract.md +196 -0
  530. package/agent/docs/design/profile-builder.md +146 -0
  531. package/agent/docs/middleware/README.md +260 -0
  532. package/agent/docs/observability/README.md +316 -0
  533. package/agent/docs/plans/2026-06-09-003-fix-telegram-stream-overflow-continuations-plan.md +240 -0
  534. package/agent/docs/rca-ssl-cacert-post-git-pull.md +54 -0
  535. package/agent/docs/relay-connector-contract.md +285 -0
  536. package/agent/gateway/authz_mixin.py +536 -0
  537. package/agent/gateway/channel_directory.py +65 -3
  538. package/agent/gateway/config.py +222 -12
  539. package/agent/gateway/display_config.py +10 -0
  540. package/agent/gateway/hooks.py +17 -0
  541. package/agent/gateway/kanban_watchers.py +1146 -0
  542. package/agent/gateway/message_timestamps.py +166 -0
  543. package/agent/gateway/platforms/ADDING_A_PLATFORM.md +29 -0
  544. package/agent/gateway/platforms/api_server.py +216 -38
  545. package/agent/gateway/platforms/base.py +210 -58
  546. package/agent/gateway/platforms/email.py +122 -12
  547. package/agent/gateway/platforms/feishu.py +80 -11
  548. package/agent/gateway/platforms/feishu_meeting_invite.py +212 -0
  549. package/agent/gateway/platforms/matrix.py +1498 -297
  550. package/agent/gateway/platforms/qqbot/adapter.py +6 -0
  551. package/agent/gateway/platforms/signal.py +8 -0
  552. package/agent/gateway/platforms/slack.py +308 -12
  553. package/agent/gateway/platforms/telegram.py +831 -24
  554. package/agent/gateway/platforms/webhook.py +109 -21
  555. package/agent/gateway/platforms/weixin.py +113 -2
  556. package/agent/gateway/platforms/whatsapp.py +94 -288
  557. package/agent/gateway/platforms/whatsapp_cloud.py +1956 -0
  558. package/agent/gateway/platforms/whatsapp_common.py +367 -0
  559. package/agent/gateway/platforms/yuanbao.py +608 -191
  560. package/agent/gateway/platforms/yuanbao_proto.py +232 -23
  561. package/agent/gateway/relay/__init__.py +375 -0
  562. package/agent/gateway/relay/adapter.py +222 -0
  563. package/agent/gateway/relay/auth.py +168 -0
  564. package/agent/gateway/relay/descriptor.py +118 -0
  565. package/agent/gateway/relay/transport.py +101 -0
  566. package/agent/gateway/relay/ws_transport.py +327 -0
  567. package/agent/gateway/response_filters.py +53 -0
  568. package/agent/gateway/rich_sent_store.py +80 -0
  569. package/agent/gateway/run.py +2940 -5001
  570. package/agent/gateway/session.py +109 -8
  571. package/agent/gateway/session_context.py +22 -4
  572. package/agent/gateway/slash_commands.py +3854 -0
  573. package/agent/gateway/status.py +141 -21
  574. package/agent/gateway/stream_consumer.py +288 -31
  575. package/agent/hermes-already-has-routines.md +1 -1
  576. package/agent/hermes_cli/__init__.py +62 -17
  577. package/agent/hermes_cli/_parser.py +30 -0
  578. package/agent/hermes_cli/_subprocess_compat.py +61 -0
  579. package/agent/hermes_cli/active_sessions.py +320 -0
  580. package/agent/hermes_cli/auth.py +707 -59
  581. package/agent/hermes_cli/auth_commands.py +39 -22
  582. package/agent/hermes_cli/backup.py +109 -7
  583. package/agent/hermes_cli/banner.py +88 -0
  584. package/agent/hermes_cli/blueprint_cmd.py +318 -0
  585. package/agent/hermes_cli/clawpump_cli.py +3 -3
  586. package/agent/hermes_cli/cli_agent_setup_mixin.py +684 -0
  587. package/agent/hermes_cli/cli_commands_mixin.py +2293 -0
  588. package/agent/hermes_cli/commands.py +216 -91
  589. package/agent/hermes_cli/config.py +967 -130
  590. package/agent/hermes_cli/container_boot.py +76 -11
  591. package/agent/hermes_cli/cron.py +5 -11
  592. package/agent/hermes_cli/curator.py +21 -0
  593. package/agent/hermes_cli/dashboard_auth/__init__.py +2 -0
  594. package/agent/hermes_cli/dashboard_auth/base.py +62 -0
  595. package/agent/hermes_cli/dashboard_auth/cookies.py +32 -19
  596. package/agent/hermes_cli/dashboard_auth/login_page.py +156 -6
  597. package/agent/hermes_cli/dashboard_auth/middleware.py +28 -4
  598. package/agent/hermes_cli/dashboard_auth/prefix.py +46 -2
  599. package/agent/hermes_cli/dashboard_auth/public_paths.py +6 -0
  600. package/agent/hermes_cli/dashboard_auth/routes.py +158 -2
  601. package/agent/hermes_cli/dashboard_auth/ws_tickets.py +85 -11
  602. package/agent/hermes_cli/dashboard_register.py +427 -0
  603. package/agent/hermes_cli/debug.py +155 -50
  604. package/agent/hermes_cli/distribution.py +227 -0
  605. package/agent/hermes_cli/doctor.py +255 -14
  606. package/agent/hermes_cli/dump.py +60 -6
  607. package/agent/hermes_cli/env_loader.py +33 -0
  608. package/agent/hermes_cli/gateway.py +755 -103
  609. package/agent/hermes_cli/gateway_enroll.py +250 -0
  610. package/agent/hermes_cli/gateway_windows.py +254 -11
  611. package/agent/hermes_cli/gui_uninstall.py +285 -0
  612. package/agent/hermes_cli/inventory.py +105 -4
  613. package/agent/hermes_cli/kanban.py +58 -71
  614. package/agent/hermes_cli/kanban_db.py +391 -14
  615. package/agent/hermes_cli/kanban_decompose.py +2 -2
  616. package/agent/hermes_cli/kanban_specify.py +3 -1
  617. package/agent/hermes_cli/logs.py +2 -0
  618. package/agent/hermes_cli/main.py +2889 -5287
  619. package/agent/hermes_cli/managed_scope.py +214 -0
  620. package/agent/hermes_cli/managed_uv.py +254 -0
  621. package/agent/hermes_cli/mcp_catalog.py +6 -3
  622. package/agent/hermes_cli/mcp_config.py +145 -21
  623. package/agent/hermes_cli/mcp_security.py +96 -0
  624. package/agent/hermes_cli/mcp_startup.py +32 -3
  625. package/agent/hermes_cli/memory_providers.py +149 -0
  626. package/agent/hermes_cli/memory_setup.py +97 -42
  627. package/agent/hermes_cli/middleware.py +313 -0
  628. package/agent/hermes_cli/model_catalog.py +31 -0
  629. package/agent/hermes_cli/model_cost_guard.py +134 -0
  630. package/agent/hermes_cli/model_normalize.py +2 -1
  631. package/agent/hermes_cli/model_setup_flows.py +2759 -0
  632. package/agent/hermes_cli/model_switch.py +242 -27
  633. package/agent/hermes_cli/models.py +284 -44
  634. package/agent/hermes_cli/nous_account.py +33 -6
  635. package/agent/hermes_cli/nous_billing.py +406 -0
  636. package/agent/hermes_cli/nous_subscription.py +202 -5
  637. package/agent/hermes_cli/platforms.py +1 -0
  638. package/agent/hermes_cli/plugins.py +218 -18
  639. package/agent/hermes_cli/plugins_cmd.py +249 -105
  640. package/agent/hermes_cli/portal_cli.py +56 -16
  641. package/agent/hermes_cli/profile_distribution.py +6 -1
  642. package/agent/hermes_cli/profiles.py +283 -32
  643. package/agent/hermes_cli/provider_catalog.py +170 -0
  644. package/agent/hermes_cli/providers.py +4 -1
  645. package/agent/hermes_cli/pty_bridge.py +53 -4
  646. package/agent/hermes_cli/runtime_provider.py +216 -34
  647. package/agent/hermes_cli/secret_prompt.py +4 -4
  648. package/agent/hermes_cli/secrets_cli.py +24 -0
  649. package/agent/hermes_cli/send_cmd.py +28 -2
  650. package/agent/hermes_cli/service_manager.py +166 -19
  651. package/agent/hermes_cli/session_listing.py +97 -0
  652. package/agent/hermes_cli/setup.py +158 -94
  653. package/agent/hermes_cli/setup_whatsapp_cloud.py +541 -0
  654. package/agent/hermes_cli/skills_config.py +8 -2
  655. package/agent/hermes_cli/skills_hub.py +149 -7
  656. package/agent/hermes_cli/status.py +2 -2
  657. package/agent/hermes_cli/subcommands/__init__.py +18 -0
  658. package/agent/hermes_cli/subcommands/_shared.py +29 -0
  659. package/agent/hermes_cli/subcommands/acp.py +52 -0
  660. package/agent/hermes_cli/subcommands/auth.py +109 -0
  661. package/agent/hermes_cli/subcommands/backup.py +38 -0
  662. package/agent/hermes_cli/subcommands/claw.py +92 -0
  663. package/agent/hermes_cli/subcommands/config.py +49 -0
  664. package/agent/hermes_cli/subcommands/cron.py +163 -0
  665. package/agent/hermes_cli/subcommands/dashboard.py +143 -0
  666. package/agent/hermes_cli/subcommands/debug.py +77 -0
  667. package/agent/hermes_cli/subcommands/doctor.py +35 -0
  668. package/agent/hermes_cli/subcommands/dump.py +28 -0
  669. package/agent/hermes_cli/subcommands/gateway.py +332 -0
  670. package/agent/hermes_cli/subcommands/gui.py +63 -0
  671. package/agent/hermes_cli/subcommands/hooks.py +77 -0
  672. package/agent/hermes_cli/subcommands/import_cmd.py +31 -0
  673. package/agent/hermes_cli/subcommands/insights.py +25 -0
  674. package/agent/hermes_cli/subcommands/login.py +78 -0
  675. package/agent/hermes_cli/subcommands/logout.py +28 -0
  676. package/agent/hermes_cli/subcommands/logs.py +78 -0
  677. package/agent/hermes_cli/subcommands/mcp.py +108 -0
  678. package/agent/hermes_cli/subcommands/memory.py +53 -0
  679. package/agent/hermes_cli/subcommands/model.py +72 -0
  680. package/agent/hermes_cli/subcommands/pairing.py +36 -0
  681. package/agent/hermes_cli/subcommands/plugins.py +94 -0
  682. package/agent/hermes_cli/subcommands/postinstall.py +23 -0
  683. package/agent/hermes_cli/subcommands/profile.py +203 -0
  684. package/agent/hermes_cli/subcommands/prompt_size.py +36 -0
  685. package/agent/hermes_cli/subcommands/security.py +62 -0
  686. package/agent/hermes_cli/subcommands/setup.py +58 -0
  687. package/agent/hermes_cli/subcommands/skills.py +298 -0
  688. package/agent/hermes_cli/subcommands/slack.py +60 -0
  689. package/agent/hermes_cli/subcommands/status.py +28 -0
  690. package/agent/hermes_cli/subcommands/tools.py +95 -0
  691. package/agent/hermes_cli/subcommands/uninstall.py +41 -0
  692. package/agent/hermes_cli/subcommands/update.py +70 -0
  693. package/agent/hermes_cli/subcommands/version.py +18 -0
  694. package/agent/hermes_cli/subcommands/webhook.py +76 -0
  695. package/agent/hermes_cli/subcommands/whatsapp.py +22 -0
  696. package/agent/hermes_cli/suggestions_cmd.py +153 -0
  697. package/agent/hermes_cli/telegram_managed_bot.py +358 -0
  698. package/agent/hermes_cli/tips.py +3 -4
  699. package/agent/hermes_cli/tools_config.py +155 -28
  700. package/agent/hermes_cli/uninstall.py +231 -35
  701. package/agent/hermes_cli/web_server.py +6188 -975
  702. package/agent/hermes_cli/win_pty_bridge.py +179 -0
  703. package/agent/hermes_cli/write_approval_commands.py +209 -0
  704. package/agent/hermes_constants.py +164 -33
  705. package/agent/hermes_logging.py +74 -2
  706. package/agent/hermes_state.py +919 -106
  707. package/agent/hermes_time.py +20 -0
  708. package/agent/locales/af.yaml +23 -0
  709. package/agent/locales/de.yaml +23 -0
  710. package/agent/locales/en.yaml +20 -0
  711. package/agent/locales/es.yaml +23 -0
  712. package/agent/locales/fr.yaml +23 -0
  713. package/agent/locales/ga.yaml +23 -0
  714. package/agent/locales/hu.yaml +23 -0
  715. package/agent/locales/it.yaml +23 -0
  716. package/agent/locales/ja.yaml +23 -0
  717. package/agent/locales/ko.yaml +23 -0
  718. package/agent/locales/pt.yaml +23 -0
  719. package/agent/locales/ru.yaml +23 -0
  720. package/agent/locales/tr.yaml +23 -0
  721. package/agent/locales/uk.yaml +23 -0
  722. package/agent/locales/zh-hant.yaml +23 -0
  723. package/agent/locales/zh.yaml +23 -0
  724. package/agent/model_tools.py +204 -40
  725. package/agent/optional-mcps/clawpump/manifest.yaml +15 -5
  726. package/agent/optional-mcps/clawpump-stdio/manifest.yaml +14 -4
  727. package/agent/optional-mcps/unreal-engine/manifest.yaml +54 -0
  728. package/agent/optional-skills/blockchain/hyperliquid/SKILL.md +2 -2
  729. package/agent/optional-skills/blockchain/hyperliquid/scripts/hyperliquid_client.py +1 -1
  730. package/agent/optional-skills/creative/kanban-video-orchestrator/SKILL.md +1 -1
  731. package/agent/optional-skills/creative/kanban-video-orchestrator/assets/setup.sh.tmpl +4 -3
  732. package/agent/optional-skills/creative/kanban-video-orchestrator/references/kanban-setup.md +6 -4
  733. package/agent/optional-skills/creative/kanban-video-orchestrator/references/tool-matrix.md +2 -2
  734. package/agent/{skills/software-development → optional-skills/devops}/hermes-s6-container-supervision/SKILL.md +2 -0
  735. package/agent/optional-skills/devops/watchers/SKILL.md +1 -1
  736. package/agent/optional-skills/devops/watchers/scripts/watch_github.py +2 -1
  737. package/agent/optional-skills/payments/mpp-agent/SKILL.md +124 -0
  738. package/agent/optional-skills/payments/stripe-link-cli/SKILL.md +184 -0
  739. package/agent/optional-skills/payments/stripe-projects/SKILL.md +120 -0
  740. package/agent/optional-skills/productivity/canvas/SKILL.md +1 -1
  741. package/agent/optional-skills/productivity/canvas/scripts/canvas_api.py +4 -1
  742. package/agent/optional-skills/productivity/shop/SKILL.md +224 -0
  743. package/agent/optional-skills/productivity/shop/references/catalog-mcp.md +236 -0
  744. package/agent/optional-skills/productivity/shop/references/direct-api.md +278 -0
  745. package/agent/optional-skills/productivity/shop/references/legal.md +3 -0
  746. package/agent/optional-skills/productivity/shop/references/safety.md +36 -0
  747. package/agent/optional-skills/productivity/shopify/SKILL.md +1 -1
  748. package/agent/optional-skills/productivity/siyuan/SKILL.md +1 -1
  749. package/agent/optional-skills/productivity/telephony/SKILL.md +4 -4
  750. package/agent/optional-skills/productivity/telephony/scripts/telephony.py +15 -15
  751. package/agent/optional-skills/security/1password/SKILL.md +1 -1
  752. package/agent/{skills/red-teaming → optional-skills/security}/godmode/SKILL.md +3 -4
  753. package/agent/{skills/red-teaming → optional-skills/security}/godmode/scripts/auto_jailbreak.py +3 -1
  754. package/agent/optional-skills/software-development/rest-graphql-debug/SKILL.md +1 -1
  755. package/agent/{skills → optional-skills}/software-development/subagent-driven-development/SKILL.md +5 -5
  756. package/agent/package-lock.json +4082 -7907
  757. package/agent/package.json +18 -3
  758. package/agent/plugins/browser/firecrawl/provider.py +4 -1
  759. package/agent/plugins/cron/__init__.py +344 -0
  760. package/agent/plugins/cron/chronos/__init__.py +241 -0
  761. package/agent/plugins/cron/chronos/_nas_client.py +123 -0
  762. package/agent/plugins/cron/chronos/plugin.yaml +9 -0
  763. package/agent/plugins/cron/chronos/verify.py +103 -0
  764. package/agent/plugins/dashboard_auth/basic/__init__.py +491 -0
  765. package/agent/plugins/dashboard_auth/basic/plugin.yaml +7 -0
  766. package/agent/plugins/dashboard_auth/nous/__init__.py +12 -14
  767. package/agent/plugins/dashboard_auth/self_hosted/__init__.py +736 -0
  768. package/agent/plugins/dashboard_auth/self_hosted/plugin.yaml +8 -0
  769. package/agent/plugins/disk-cleanup/disk_cleanup.py +100 -20
  770. package/agent/plugins/google_meet/audio_bridge.py +4 -0
  771. package/agent/plugins/google_meet/meet_bot.py +7 -1
  772. package/agent/plugins/hermes-achievements/dashboard/dist/index.js +9 -15
  773. package/agent/plugins/image_gen/fal/__init__.py +35 -6
  774. package/agent/plugins/image_gen/krea/__init__.py +56 -13
  775. package/agent/plugins/image_gen/openai/__init__.py +122 -24
  776. package/agent/plugins/image_gen/openai-codex/__init__.py +28 -2
  777. package/agent/plugins/image_gen/xai/__init__.py +92 -12
  778. package/agent/plugins/kanban/dashboard/dist/index.js +63 -48
  779. package/agent/plugins/kanban/dashboard/plugin_api.py +39 -35
  780. package/agent/plugins/memory/__init__.py +48 -5
  781. package/agent/plugins/memory/byterover/__init__.py +1 -0
  782. package/agent/plugins/memory/hindsight/README.md +1 -1
  783. package/agent/plugins/memory/hindsight/__init__.py +138 -24
  784. package/agent/plugins/memory/hindsight/plugin.yaml +1 -1
  785. package/agent/plugins/memory/honcho/README.md +13 -10
  786. package/agent/plugins/memory/honcho/cli.py +247 -122
  787. package/agent/plugins/memory/honcho/client.py +112 -102
  788. package/agent/plugins/memory/openviking/README.md +12 -1
  789. package/agent/plugins/memory/openviking/__init__.py +2281 -107
  790. package/agent/plugins/memory/openviking/plugin.yaml +1 -2
  791. package/agent/plugins/memory/supermemory/README.md +22 -10
  792. package/agent/plugins/memory/supermemory/__init__.py +142 -37
  793. package/agent/plugins/memory/supermemory/plugin.yaml +1 -1
  794. package/agent/plugins/model-providers/anthropic/__init__.py +1 -0
  795. package/agent/plugins/model-providers/bedrock/__init__.py +1 -0
  796. package/agent/plugins/model-providers/copilot-acp/__init__.py +1 -0
  797. package/agent/plugins/model-providers/custom/__init__.py +8 -2
  798. package/agent/plugins/model-providers/kimi-coding/__init__.py +16 -7
  799. package/agent/plugins/model-providers/minimax/__init__.py +60 -8
  800. package/agent/plugins/model-providers/opencode-zen/__init__.py +12 -3
  801. package/agent/plugins/model-providers/openrouter/__init__.py +75 -4
  802. package/agent/plugins/model-providers/xiaomi/__init__.py +2 -0
  803. package/agent/plugins/model-providers/zai/__init__.py +1 -0
  804. package/agent/plugins/observability/langfuse/__init__.py +147 -14
  805. package/agent/plugins/observability/nemo_relay/README.md +559 -0
  806. package/agent/plugins/observability/nemo_relay/__init__.py +962 -0
  807. package/agent/plugins/observability/nemo_relay/plugin.yaml +20 -0
  808. package/agent/plugins/platforms/discord/adapter.py +932 -61
  809. package/agent/plugins/platforms/discord/voice_mixer.py +379 -0
  810. package/agent/plugins/platforms/google_chat/adapter.py +9 -3
  811. package/agent/plugins/platforms/google_chat/oauth.py +1 -1
  812. package/agent/plugins/platforms/homeassistant/__init__.py +3 -0
  813. package/agent/{gateway/platforms/homeassistant.py → plugins/platforms/homeassistant/adapter.py} +128 -0
  814. package/agent/plugins/platforms/homeassistant/plugin.yaml +22 -0
  815. package/agent/plugins/platforms/irc/adapter.py +4 -1
  816. package/agent/plugins/platforms/line/adapter.py +16 -1
  817. package/agent/plugins/platforms/mattermost/adapter.py +100 -24
  818. package/agent/plugins/platforms/photon/README.md +179 -0
  819. package/agent/plugins/platforms/photon/__init__.py +4 -0
  820. package/agent/plugins/platforms/photon/adapter.py +1586 -0
  821. package/agent/plugins/platforms/photon/auth.py +1046 -0
  822. package/agent/plugins/platforms/photon/cli.py +439 -0
  823. package/agent/plugins/platforms/photon/plugin.yaml +88 -0
  824. package/agent/plugins/platforms/photon/sidecar/README.md +52 -0
  825. package/agent/plugins/platforms/photon/sidecar/index.mjs +720 -0
  826. package/agent/plugins/platforms/photon/sidecar/package-lock.json +1730 -0
  827. package/agent/plugins/platforms/photon/sidecar/package.json +25 -0
  828. package/agent/plugins/platforms/photon/sidecar/patch-spectrum-mixed-attachments.mjs +155 -0
  829. package/agent/plugins/platforms/raft/__init__.py +3 -0
  830. package/agent/plugins/platforms/raft/adapter.py +774 -0
  831. package/agent/plugins/platforms/raft/plugin.yaml +19 -0
  832. package/agent/plugins/platforms/simplex/adapter.py +777 -220
  833. package/agent/plugins/platforms/simplex/plugin.yaml +21 -2
  834. package/agent/plugins/platforms/teams/adapter.py +175 -5
  835. package/agent/plugins/plugin_utils.py +135 -0
  836. package/agent/plugins/video_gen/fal/__init__.py +10 -3
  837. package/agent/plugins/web/searxng/provider.py +15 -2
  838. package/agent/plugins/web/xai/provider.py +2 -2
  839. package/agent/providers/base.py +22 -3
  840. package/agent/pyproject.toml +115 -21
  841. package/agent/run_agent.py +733 -39
  842. package/agent/scripts/build_skills_index.py +51 -19
  843. package/agent/scripts/check_subprocess_stdin.py +177 -0
  844. package/agent/scripts/contributor_audit.py +2 -0
  845. package/agent/scripts/docker_config_migrate.py +67 -0
  846. package/agent/scripts/install.cmd +3 -3
  847. package/agent/scripts/install.ps1 +580 -154
  848. package/agent/scripts/install.sh +402 -185
  849. package/agent/scripts/lib/node-bootstrap.sh +39 -4
  850. package/agent/scripts/release.py +183 -0
  851. package/agent/scripts/run_tests.sh +1 -0
  852. package/agent/scripts/run_tests_parallel.py +18 -23
  853. package/agent/scripts/whatsapp-bridge/bridge.js +25 -4
  854. package/agent/setup.py +59 -0
  855. package/agent/skills/autonomous-ai-agents/codex/SKILL.md +19 -0
  856. package/agent/skills/autonomous-ai-agents/hermes-agent/SKILL.md +10 -3
  857. package/agent/skills/{mcp/native-mcp/SKILL.md → autonomous-ai-agents/hermes-agent/references/native-mcp.md} +0 -13
  858. package/agent/skills/{devops/webhook-subscriptions/SKILL.md → autonomous-ai-agents/hermes-agent/references/webhooks.md} +1 -11
  859. package/agent/skills/clawpump/SKILL.md +53 -5
  860. package/agent/skills/devops/kanban-orchestrator/SKILL.md +1 -0
  861. package/agent/skills/devops/kanban-worker/SKILL.md +1 -0
  862. package/agent/skills/github/github-auth/SKILL.md +2 -2
  863. package/agent/skills/github/github-auth/scripts/gh-env.sh +2 -2
  864. package/agent/skills/github/github-code-review/SKILL.md +2 -2
  865. package/agent/skills/github/github-issues/SKILL.md +2 -2
  866. package/agent/skills/github/github-pr-workflow/SKILL.md +2 -2
  867. package/agent/skills/github/github-repo-management/SKILL.md +2 -2
  868. package/agent/skills/media/gif-search/SKILL.md +1 -1
  869. package/agent/skills/media/youtube-content/SKILL.md +10 -7
  870. package/agent/skills/media/youtube-content/scripts/fetch_transcript.py +3 -3
  871. package/agent/skills/note-taking/obsidian/SKILL.md +1 -1
  872. package/agent/skills/productivity/airtable/SKILL.md +2 -2
  873. package/agent/skills/productivity/google-workspace/scripts/setup.py +33 -7
  874. package/agent/skills/productivity/notion/SKILL.md +2 -2
  875. package/agent/skills/productivity/teams-meeting-pipeline/SKILL.md +1 -1
  876. package/agent/skills/research/llm-wiki/SKILL.md +1 -1
  877. package/agent/skills/social-media/xurl/SKILL.md +9 -0
  878. package/agent/skills/software-development/hermes-agent-skill-authoring/SKILL.md +1 -1
  879. package/agent/skills/software-development/plan/SKILL.md +285 -5
  880. package/agent/skills/software-development/requesting-code-review/SKILL.md +2 -2
  881. package/agent/skills/software-development/simplify-code/SKILL.md +212 -0
  882. package/agent/skills/software-development/spike/SKILL.md +2 -2
  883. package/agent/skills/software-development/systematic-debugging/SKILL.md +1 -1
  884. package/agent/skills/software-development/test-driven-development/SKILL.md +1 -1
  885. package/agent/tools/approval.py +302 -4
  886. package/agent/tools/async_delegation.py +386 -0
  887. package/agent/tools/blueprints.py +325 -0
  888. package/agent/tools/browser_cdp_tool.py +3 -3
  889. package/agent/tools/browser_tool.py +34 -6
  890. package/agent/tools/checkpoint_manager.py +31 -1
  891. package/agent/tools/clarify_tool.py +55 -5
  892. package/agent/tools/code_execution_tool.py +31 -14
  893. package/agent/tools/computer_use/cua_backend.py +81 -3
  894. package/agent/tools/computer_use/tool.py +79 -5
  895. package/agent/tools/computer_use/vision_routing.py +55 -3
  896. package/agent/tools/credential_files.py +31 -12
  897. package/agent/tools/cronjob_tools.py +30 -20
  898. package/agent/tools/delegate_tool.py +356 -31
  899. package/agent/tools/env_probe.py +1 -0
  900. package/agent/tools/environments/docker.py +163 -8
  901. package/agent/tools/environments/file_sync.py +2 -1
  902. package/agent/tools/environments/local.py +74 -23
  903. package/agent/tools/environments/singularity.py +4 -1
  904. package/agent/tools/environments/ssh.py +78 -11
  905. package/agent/tools/file_operations.py +277 -41
  906. package/agent/tools/file_tools.py +166 -28
  907. package/agent/tools/image_generation_tool.py +515 -29
  908. package/agent/tools/kanban_tools.py +99 -0
  909. package/agent/tools/lazy_deps.py +33 -2
  910. package/agent/tools/mcp_oauth.py +5 -5
  911. package/agent/tools/mcp_oauth_manager.py +7 -5
  912. package/agent/tools/mcp_tool.py +840 -33
  913. package/agent/tools/memory_tool.py +335 -38
  914. package/agent/tools/osv_check.py +15 -1
  915. package/agent/tools/process_registry.py +155 -11
  916. package/agent/tools/read_extract.py +248 -0
  917. package/agent/tools/read_terminal_tool.py +93 -0
  918. package/agent/tools/schema_sanitizer.py +38 -0
  919. package/agent/tools/send_message_tool.py +163 -49
  920. package/agent/tools/session_search_tool.py +189 -7
  921. package/agent/tools/skill_manager_tool.py +202 -3
  922. package/agent/tools/skill_usage.py +52 -4
  923. package/agent/tools/skills_hub.py +184 -44
  924. package/agent/tools/skills_sync.py +232 -5
  925. package/agent/tools/skills_tool.py +125 -11
  926. package/agent/tools/terminal_tool.py +148 -26
  927. package/agent/tools/tirith_security.py +2 -0
  928. package/agent/tools/todo_tool.py +32 -1
  929. package/agent/tools/transcription_tools.py +13 -5
  930. package/agent/tools/tts_tool.py +332 -38
  931. package/agent/tools/url_safety.py +52 -1
  932. package/agent/tools/vision_tools.py +124 -39
  933. package/agent/tools/voice_mode.py +4 -3
  934. package/agent/tools/web_tools.py +45 -15
  935. package/agent/tools/write_approval.py +493 -0
  936. package/agent/toolsets.py +34 -10
  937. package/agent/trajectory_compressor.py +81 -10
  938. package/agent/tui_gateway/entry.py +43 -6
  939. package/agent/tui_gateway/server.py +3335 -330
  940. package/agent/tui_gateway/slash_worker.py +61 -0
  941. package/agent/tui_gateway/ws.py +67 -9
  942. package/agent/ui-tui/eslint.config.mjs +0 -4
  943. package/agent/ui-tui/package.json +6 -6
  944. package/agent/ui-tui/packages/hermes-ink/package.json +1 -1
  945. package/agent/ui-tui/packages/hermes-ink/src/ink/app-mouse.test.ts +34 -1
  946. package/agent/ui-tui/packages/hermes-ink/src/ink/app-rawmode-mouse.test.ts +91 -0
  947. package/agent/ui-tui/packages/hermes-ink/src/ink/components/App.tsx +35 -2
  948. package/agent/ui-tui/packages/hermes-ink/src/ink/events/input-event.ts +4 -11
  949. package/agent/ui-tui/packages/hermes-ink/src/ink/parse-keypress.test.ts +23 -57
  950. package/agent/ui-tui/packages/hermes-ink/src/ink/parse-keypress.ts +11 -135
  951. package/agent/ui-tui/packages/hermes-ink/src/ink/termio/tokenize.test.ts +185 -0
  952. package/agent/ui-tui/packages/hermes-ink/src/ink/termio/tokenize.ts +37 -3
  953. package/agent/ui-tui/packages/hermes-ink/src/utils/execFileNoThrow.ts +5 -5
  954. package/agent/ui-tui/src/__tests__/appChromeStatusRule.test.tsx +217 -0
  955. package/agent/ui-tui/src/__tests__/appChromeStatusRuleDevCredits.test.tsx +73 -0
  956. package/agent/ui-tui/src/__tests__/approvalAction.test.ts +11 -0
  957. package/agent/ui-tui/src/__tests__/billingCommand.test.ts +301 -0
  958. package/agent/ui-tui/src/__tests__/blockLayout.test.ts +122 -0
  959. package/agent/ui-tui/src/__tests__/brandingMcpCount.test.ts +111 -0
  960. package/agent/ui-tui/src/__tests__/completionApply.test.ts +51 -0
  961. package/agent/ui-tui/src/__tests__/createGatewayEventHandler.test.ts +487 -2
  962. package/agent/ui-tui/src/__tests__/createSlashHandler.test.ts +54 -0
  963. package/agent/ui-tui/src/__tests__/creditsCommand.test.ts +144 -0
  964. package/agent/ui-tui/src/__tests__/gatewayClient.test.ts +120 -99
  965. package/agent/ui-tui/src/__tests__/gracefulExit.test.ts +11 -0
  966. package/agent/ui-tui/src/__tests__/memoryMonitor.test.ts +102 -0
  967. package/agent/ui-tui/src/__tests__/paths.test.ts +41 -1
  968. package/agent/ui-tui/src/__tests__/terminalModes.test.ts +22 -0
  969. package/agent/ui-tui/src/__tests__/text.test.ts +23 -0
  970. package/agent/ui-tui/src/__tests__/textInputFastEcho.test.ts +37 -0
  971. package/agent/ui-tui/src/__tests__/turnControllerNotice.test.ts +43 -0
  972. package/agent/ui-tui/src/__tests__/useInputHandlers.test.ts +38 -1
  973. package/agent/ui-tui/src/__tests__/virtualHeights.test.ts +8 -0
  974. package/agent/ui-tui/src/app/createGatewayEventHandler.ts +102 -7
  975. package/agent/ui-tui/src/app/interfaces.ts +64 -1
  976. package/agent/ui-tui/src/app/overlayStore.ts +18 -2
  977. package/agent/ui-tui/src/app/slash/commands/billing.ts +332 -0
  978. package/agent/ui-tui/src/app/slash/commands/core.ts +31 -2
  979. package/agent/ui-tui/src/app/slash/commands/credits.ts +57 -0
  980. package/agent/ui-tui/src/app/slash/commands/ops.ts +28 -0
  981. package/agent/ui-tui/src/app/slash/commands/session.ts +32 -4
  982. package/agent/ui-tui/src/app/slash/registry.ts +4 -0
  983. package/agent/ui-tui/src/app/turnController.ts +145 -2
  984. package/agent/ui-tui/src/app/uiStore.ts +2 -0
  985. package/agent/ui-tui/src/app/useInputHandlers.ts +42 -4
  986. package/agent/ui-tui/src/app/useMainApp.ts +54 -8
  987. package/agent/ui-tui/src/app/useSessionLifecycle.ts +40 -31
  988. package/agent/ui-tui/src/app/useSubmission.ts +23 -31
  989. package/agent/ui-tui/src/components/appChrome.tsx +112 -5
  990. package/agent/ui-tui/src/components/appLayout.tsx +9 -0
  991. package/agent/ui-tui/src/components/appOverlays.tsx +25 -1
  992. package/agent/ui-tui/src/components/billingOverlay.tsx +684 -0
  993. package/agent/ui-tui/src/components/branding.tsx +15 -3
  994. package/agent/ui-tui/src/components/messageLine.tsx +25 -3
  995. package/agent/ui-tui/src/components/pluginsHub.tsx +238 -0
  996. package/agent/ui-tui/src/components/prompts.tsx +31 -17
  997. package/agent/ui-tui/src/components/streamingAssistant.tsx +63 -55
  998. package/agent/ui-tui/src/components/textInput.tsx +16 -0
  999. package/agent/ui-tui/src/config/env.ts +12 -0
  1000. package/agent/ui-tui/src/config/limits.ts +13 -0
  1001. package/agent/ui-tui/src/domain/blockLayout.ts +146 -0
  1002. package/agent/ui-tui/src/domain/paths.ts +24 -0
  1003. package/agent/ui-tui/src/domain/slash.ts +40 -0
  1004. package/agent/ui-tui/src/entry.tsx +35 -4
  1005. package/agent/ui-tui/src/gatewayClient.ts +22 -10
  1006. package/agent/ui-tui/src/gatewayTypes.ts +130 -1
  1007. package/agent/ui-tui/src/lib/gracefulExit.ts +24 -4
  1008. package/agent/ui-tui/src/lib/memory.test.ts +162 -0
  1009. package/agent/ui-tui/src/lib/memory.ts +60 -1
  1010. package/agent/ui-tui/src/lib/memoryMonitor.ts +79 -4
  1011. package/agent/ui-tui/src/lib/osc52.ts +1 -1
  1012. package/agent/ui-tui/src/lib/text.test.ts +32 -1
  1013. package/agent/ui-tui/src/lib/text.ts +29 -2
  1014. package/agent/ui-tui/src/lib/virtualHeights.ts +13 -0
  1015. package/agent/ui-tui/src/types.ts +5 -0
  1016. package/agent/ui-tui/tsconfig.build.json +0 -1
  1017. package/agent/ui-tui/tsconfig.json +2 -1
  1018. package/agent/utils.py +66 -2
  1019. package/agent/uv.lock +308 -696
  1020. package/agent/web/index.html +2 -2
  1021. package/agent/web/package.json +11 -6
  1022. package/agent/web/public/claw-bg.webp +0 -0
  1023. package/agent/web/public/claw-logo.webp +0 -0
  1024. package/agent/web/src/App.tsx +138 -48
  1025. package/agent/web/src/components/AutomationBlueprints.tsx +225 -0
  1026. package/agent/web/src/components/Backdrop.tsx +15 -0
  1027. package/agent/web/src/components/ChatSessionList.tsx +260 -0
  1028. package/agent/web/src/components/ChatSidebar.tsx +262 -78
  1029. package/agent/web/src/components/ConfirmDialog.tsx +122 -0
  1030. package/agent/web/src/components/ModelPickerDialog.tsx +111 -16
  1031. package/agent/web/src/components/ModelReloadConfirm.tsx +40 -0
  1032. package/agent/web/src/components/ProfileScopeBanner.tsx +30 -0
  1033. package/agent/web/src/components/ProfileSwitcher.tsx +67 -0
  1034. package/agent/web/src/components/ReasoningPicker.tsx +167 -0
  1035. package/agent/web/src/components/SkillEditorDialog.tsx +215 -0
  1036. package/agent/web/src/components/ThemeSwitcher.tsx +119 -4
  1037. package/agent/web/src/components/ToolsetConfigDrawer.tsx +457 -0
  1038. package/agent/web/src/contexts/PageHeaderProvider.tsx +7 -4
  1039. package/agent/web/src/contexts/ProfileProvider.tsx +137 -0
  1040. package/agent/web/src/contexts/SystemActions.tsx +6 -8
  1041. package/agent/web/src/contexts/profile-context.ts +19 -0
  1042. package/agent/web/src/contexts/useProfileScope.ts +6 -0
  1043. package/agent/web/src/i18n/af.ts +5 -4
  1044. package/agent/web/src/i18n/de.ts +5 -4
  1045. package/agent/web/src/i18n/en.ts +58 -4
  1046. package/agent/web/src/i18n/es.ts +5 -3
  1047. package/agent/web/src/i18n/fr.ts +5 -3
  1048. package/agent/web/src/i18n/ga.ts +5 -4
  1049. package/agent/web/src/i18n/hu.ts +5 -4
  1050. package/agent/web/src/i18n/it.ts +5 -4
  1051. package/agent/web/src/i18n/ja.ts +5 -4
  1052. package/agent/web/src/i18n/ko.ts +5 -4
  1053. package/agent/web/src/i18n/pt.ts +5 -3
  1054. package/agent/web/src/i18n/ru.ts +5 -4
  1055. package/agent/web/src/i18n/tr.ts +5 -4
  1056. package/agent/web/src/i18n/types.ts +59 -1
  1057. package/agent/web/src/i18n/uk.ts +5 -3
  1058. package/agent/web/src/i18n/zh-hant.ts +5 -4
  1059. package/agent/web/src/i18n/zh.ts +5 -4
  1060. package/agent/web/src/index.css +2 -2
  1061. package/agent/web/src/lib/api.ts +819 -52
  1062. package/agent/web/src/lib/dashboard-flags.ts +16 -7
  1063. package/agent/web/src/lib/reasoning-effort.test.ts +48 -0
  1064. package/agent/web/src/lib/reasoning-effort.ts +36 -0
  1065. package/agent/web/src/lib/session-refresh.test.ts +21 -0
  1066. package/agent/web/src/lib/session-refresh.ts +26 -0
  1067. package/agent/web/src/pages/ChannelsPage.tsx +529 -68
  1068. package/agent/web/src/pages/ChatPage.tsx +249 -56
  1069. package/agent/web/src/pages/ConfigPage.tsx +11 -1
  1070. package/agent/web/src/pages/CronPage.tsx +219 -31
  1071. package/agent/web/src/pages/EnvPage.tsx +25 -6
  1072. package/agent/web/src/pages/FilesPage.tsx +525 -0
  1073. package/agent/web/src/pages/McpPage.tsx +80 -3
  1074. package/agent/web/src/pages/ModelsPage.tsx +97 -12
  1075. package/agent/web/src/pages/PluginsPage.tsx +1 -1
  1076. package/agent/web/src/pages/ProfileBuilderPage.tsx +611 -0
  1077. package/agent/web/src/pages/ProfilesPage.tsx +1038 -172
  1078. package/agent/web/src/pages/SessionsPage.tsx +144 -13
  1079. package/agent/web/src/pages/SkillsPage.tsx +851 -70
  1080. package/agent/web/src/pages/SystemPage.tsx +340 -4
  1081. package/agent/web/src/pages/WalletPage.tsx +401 -0
  1082. package/agent/web/src/pages/WebhooksPage.tsx +145 -15
  1083. package/agent/web/src/pages/X402Page.tsx +207 -0
  1084. package/agent/web/src/plugins/registry.ts +28 -11
  1085. package/agent/web/src/plugins/sdk.d.ts +160 -0
  1086. package/agent/web/src/themes/context.tsx +112 -5
  1087. package/agent/web/src/themes/fonts.ts +167 -0
  1088. package/agent/web/src/themes/index.ts +7 -0
  1089. package/agent/web/tsconfig.app.json +0 -1
  1090. package/agent/web/vite.config.ts +1 -8
  1091. package/agent/web/vitest.config.ts +16 -0
  1092. package/package.json +1 -1
  1093. package/agent/apps/desktop/package-lock.json +0 -18363
  1094. package/agent/apps/desktop/src/app/chat/composer/skin-slash-popover.tsx +0 -56
  1095. package/agent/apps/desktop/src/components/assistant-ui/thread-virtualizer.tsx +0 -382
  1096. package/agent/apps/desktop/src/components/assistant-ui/todo-tool.tsx +0 -109
  1097. package/agent/apps/desktop/src/components/chat/generated-image-context.tsx +0 -19
  1098. package/agent/optional-skills/productivity/shop-app/SKILL.md +0 -340
  1099. package/agent/skills/autonomous-ai-agents/kanban-codex-lane/SKILL.md +0 -277
  1100. package/agent/skills/autonomous-ai-agents/kanban-codex-lane/templates/pmb-codex-lane-prompt.md +0 -57
  1101. package/agent/skills/diagramming/DESCRIPTION.md +0 -3
  1102. package/agent/skills/domain/DESCRIPTION.md +0 -24
  1103. package/agent/skills/gifs/DESCRIPTION.md +0 -3
  1104. package/agent/skills/inference-sh/DESCRIPTION.md +0 -19
  1105. package/agent/skills/mcp/DESCRIPTION.md +0 -3
  1106. package/agent/skills/media/spotify/SKILL.md +0 -135
  1107. package/agent/skills/mlops/training/DESCRIPTION.md +0 -3
  1108. package/agent/skills/mlops/vector-databases/DESCRIPTION.md +0 -3
  1109. package/agent/skills/productivity/linear/SKILL.md +0 -380
  1110. package/agent/skills/productivity/linear/scripts/linear_api.py +0 -445
  1111. package/agent/skills/software-development/debugging-hermes-tui-commands/SKILL.md +0 -152
  1112. package/agent/skills/software-development/writing-plans/SKILL.md +0 -297
  1113. package/agent/ui-tui/package-lock.json +0 -7449
  1114. package/agent/ui-tui/packages/hermes-ink/package-lock.json +0 -1289
  1115. package/agent/web/package-lock.json +0 -8887
  1116. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/PORT_NOTES.md +0 -0
  1117. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/SKILL.md +0 -0
  1118. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/prompts/system.md +0 -0
  1119. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/palettes/macaron.md +0 -0
  1120. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/palettes/mono-ink.md +0 -0
  1121. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/palettes/neon.md +0 -0
  1122. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/palettes/warm.md +0 -0
  1123. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/prompt-construction.md +0 -0
  1124. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/style-presets.md +0 -0
  1125. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/blueprint.md +0 -0
  1126. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/chalkboard.md +0 -0
  1127. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/editorial.md +0 -0
  1128. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/elegant.md +0 -0
  1129. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/fantasy-animation.md +0 -0
  1130. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/flat-doodle.md +0 -0
  1131. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/flat.md +0 -0
  1132. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/ink-notes.md +0 -0
  1133. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/intuition-machine.md +0 -0
  1134. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/minimal.md +0 -0
  1135. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/nature.md +0 -0
  1136. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/notion.md +0 -0
  1137. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/pixel-art.md +0 -0
  1138. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/playful.md +0 -0
  1139. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/retro.md +0 -0
  1140. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/scientific.md +0 -0
  1141. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/screen-print.md +0 -0
  1142. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/sketch-notes.md +0 -0
  1143. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/sketch.md +0 -0
  1144. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/vector-illustration.md +0 -0
  1145. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/vintage.md +0 -0
  1146. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/warm.md +0 -0
  1147. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles/watercolor.md +0 -0
  1148. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/styles.md +0 -0
  1149. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/usage.md +0 -0
  1150. /package/agent/{skills → optional-skills}/creative/baoyu-article-illustrator/references/workflow.md +0 -0
  1151. /package/agent/{skills → optional-skills}/creative/baoyu-comic/PORT_NOTES.md +0 -0
  1152. /package/agent/{skills → optional-skills}/creative/baoyu-comic/SKILL.md +0 -0
  1153. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/analysis-framework.md +0 -0
  1154. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/chalk.md +0 -0
  1155. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/ink-brush.md +0 -0
  1156. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/ligne-claire.md +0 -0
  1157. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/manga.md +0 -0
  1158. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/minimalist.md +0 -0
  1159. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/art-styles/realistic.md +0 -0
  1160. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/auto-selection.md +0 -0
  1161. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/base-prompt.md +0 -0
  1162. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/character-template.md +0 -0
  1163. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/cinematic.md +0 -0
  1164. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/dense.md +0 -0
  1165. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/four-panel.md +0 -0
  1166. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/mixed.md +0 -0
  1167. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/splash.md +0 -0
  1168. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/standard.md +0 -0
  1169. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/layouts/webtoon.md +0 -0
  1170. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/ohmsha-guide.md +0 -0
  1171. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/partial-workflows.md +0 -0
  1172. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/presets/concept-story.md +0 -0
  1173. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/presets/four-panel.md +0 -0
  1174. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/presets/ohmsha.md +0 -0
  1175. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/presets/shoujo.md +0 -0
  1176. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/presets/wuxia.md +0 -0
  1177. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/storyboard-template.md +0 -0
  1178. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/action.md +0 -0
  1179. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/dramatic.md +0 -0
  1180. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/energetic.md +0 -0
  1181. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/neutral.md +0 -0
  1182. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/romantic.md +0 -0
  1183. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/vintage.md +0 -0
  1184. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/tones/warm.md +0 -0
  1185. /package/agent/{skills → optional-skills}/creative/baoyu-comic/references/workflow.md +0 -0
  1186. /package/agent/{skills → optional-skills}/creative/creative-ideation/SKILL.md +0 -0
  1187. /package/agent/{skills → optional-skills}/creative/creative-ideation/references/full-prompt-library.md +0 -0
  1188. /package/agent/{skills → optional-skills}/creative/pixel-art/ATTRIBUTION.md +0 -0
  1189. /package/agent/{skills → optional-skills}/creative/pixel-art/SKILL.md +0 -0
  1190. /package/agent/{skills → optional-skills}/creative/pixel-art/references/palettes.md +0 -0
  1191. /package/agent/{skills → optional-skills}/creative/pixel-art/scripts/__init__.py +0 -0
  1192. /package/agent/{skills → optional-skills}/creative/pixel-art/scripts/palettes.py +0 -0
  1193. /package/agent/{skills → optional-skills}/creative/pixel-art/scripts/pixel_art.py +0 -0
  1194. /package/agent/{skills → optional-skills}/creative/pixel-art/scripts/pixel_art_video.py +0 -0
  1195. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/SKILL.md +0 -0
  1196. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/references/analysis-modules.md +0 -0
  1197. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/references/methods-guide.md +0 -0
  1198. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/templates/abliteration-config.yaml +0 -0
  1199. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/templates/analysis-study.yaml +0 -0
  1200. /package/agent/{skills/mlops/inference → optional-skills/mlops}/obliteratus/templates/batch-abliteration.yaml +0 -0
  1201. /package/agent/{skills → optional-skills}/mlops/research/DESCRIPTION.md +0 -0
  1202. /package/agent/{skills → optional-skills}/mlops/research/dspy/SKILL.md +0 -0
  1203. /package/agent/{skills → optional-skills}/mlops/research/dspy/references/examples.md +0 -0
  1204. /package/agent/{skills → optional-skills}/mlops/research/dspy/references/modules.md +0 -0
  1205. /package/agent/{skills → optional-skills}/mlops/research/dspy/references/optimizers.md +0 -0
  1206. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/references/jailbreak-templates.md +0 -0
  1207. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/references/refusal-detection.md +0 -0
  1208. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/scripts/godmode_race.py +0 -0
  1209. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/scripts/load_godmode.py +0 -0
  1210. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/scripts/parseltongue.py +0 -0
  1211. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/templates/prefill-subtle.json +0 -0
  1212. /package/agent/{skills/red-teaming → optional-skills/security}/godmode/templates/prefill.json +0 -0
  1213. /package/agent/{skills → optional-skills}/software-development/subagent-driven-development/references/context-budget-discipline.md +0 -0
  1214. /package/agent/{skills → optional-skills}/software-development/subagent-driven-development/references/gates-taxonomy.md +0 -0
@@ -1,4 +1,4 @@
1
- import { useEffect, useLayoutEffect, useState, useMemo } from "react";
1
+ import { useEffect, useLayoutEffect, useState, useMemo, useCallback } from "react";
2
2
  import {
3
3
  Package,
4
4
  Search,
@@ -7,6 +7,9 @@ import {
7
7
  Cpu,
8
8
  Globe,
9
9
  Shield,
10
+ ShieldCheck,
11
+ ShieldAlert,
12
+ ShieldQuestion,
10
13
  Eye,
11
14
  Paintbrush,
12
15
  Brain,
@@ -16,9 +19,28 @@ import {
16
19
  Filter,
17
20
  Download,
18
21
  RefreshCw,
22
+ FileText,
23
+ ExternalLink,
24
+ CheckCircle2,
25
+ AlertTriangle,
26
+ Sparkles,
27
+ Loader2,
28
+ Pencil,
29
+ Plus,
19
30
  } from "lucide-react";
20
31
  import { api } from "@/lib/api";
21
- import type { SkillInfo, ToolsetInfo, SkillHubResult } from "@/lib/api";
32
+ import type {
33
+ SkillInfo,
34
+ ToolsetInfo,
35
+ SkillHubResult,
36
+ SkillHubSource,
37
+ SkillHubInstalledEntry,
38
+ SkillHubPreview,
39
+ SkillHubScan,
40
+ } from "@/lib/api";
41
+ import { useProfileScope } from "@/contexts/useProfileScope";
42
+ import { ToolsetConfigDrawer } from "@/components/ToolsetConfigDrawer";
43
+ import { SkillEditorDialog } from "@/components/SkillEditorDialog";
22
44
  import { useToast } from "@nous-research/ui/hooks/use-toast";
23
45
  import { Toast } from "@nous-research/ui/ui/components/toast";
24
46
  import { Card, CardContent, CardHeader, CardTitle } from "@nous-research/ui/ui/components/card";
@@ -27,6 +49,13 @@ import { Button } from "@nous-research/ui/ui/components/button";
27
49
  import { ListItem } from "@nous-research/ui/ui/components/list-item";
28
50
  import { Spinner } from "@nous-research/ui/ui/components/spinner";
29
51
  import { Switch } from "@nous-research/ui/ui/components/switch";
52
+ import {
53
+ Dialog,
54
+ DialogContent,
55
+ DialogDescription,
56
+ DialogHeader,
57
+ DialogTitle,
58
+ } from "@nous-research/ui/ui/components/dialog";
30
59
  import { cn } from "@/lib/utils";
31
60
  import { Input } from "@nous-research/ui/ui/components/input";
32
61
  import { useI18n } from "@/i18n";
@@ -103,25 +132,51 @@ export default function SkillsPage() {
103
132
  const [view, setView] = useState<"skills" | "toolsets" | "hub">("skills");
104
133
  const [activeCategory, setActiveCategory] = useState<string | null>(null);
105
134
  const [togglingSkills, setTogglingSkills] = useState<Set<string>>(new Set());
135
+ const [configToolset, setConfigToolset] = useState<ToolsetInfo | null>(null);
136
+ // Skill editor dialog: open + which skill is being edited (null = create).
137
+ const [editorOpen, setEditorOpen] = useState(false);
138
+ const [editorSkill, setEditorSkill] = useState<string | null>(null);
106
139
  const { toast, showToast } = useToast();
107
140
  const { t } = useI18n();
108
141
  const { setAfterTitle, setEnd } = usePageHeader();
109
142
 
143
+ // ── Profile scoping ──
144
+ // The write target comes from the GLOBAL profile switcher (sidebar) via
145
+ // ProfileContext — one selector for the whole dashboard, deep-linkable
146
+ // as ?profile=<name>. This page just consumes it: the fetchJSON layer
147
+ // appends the param automatically; we still pass it explicitly where the
148
+ // call signature supports it (clearer, and robust if a caller bypasses
149
+ // the auto-injection).
150
+ const {
151
+ profile: selectedProfile,
152
+ } = useProfileScope();
153
+
110
154
  useEffect(() => {
111
- Promise.all([api.getSkills(), api.getToolsets()])
155
+ // Promise-chain shape: setState fires only inside async callbacks so the
156
+ // effect body stays lint-clean (react-hooks/set-state-in-effect). On a
157
+ // profile switch the old list stays visible until the new one arrives.
158
+ let cancelled = false;
159
+ Promise.all([
160
+ api.getSkills(selectedProfile || undefined),
161
+ api.getToolsets(selectedProfile || undefined),
162
+ ])
112
163
  .then(([s, tsets]) => {
164
+ if (cancelled) return;
113
165
  setSkills(s);
114
166
  setToolsets(tsets);
115
167
  })
116
- .catch(() => showToast(t.common.loading, "error"))
117
- .finally(() => setLoading(false));
118
- }, []);
168
+ .catch(() => !cancelled && showToast(t.common.loading, "error"))
169
+ .finally(() => !cancelled && setLoading(false));
170
+ return () => {
171
+ cancelled = true;
172
+ };
173
+ }, [selectedProfile]);
119
174
 
120
175
  /* ---- Toggle skill ---- */
121
176
  const handleToggleSkill = async (skill: SkillInfo) => {
122
177
  setTogglingSkills((prev) => new Set(prev).add(skill.name));
123
178
  try {
124
- await api.toggleSkill(skill.name, !skill.enabled);
179
+ await api.toggleSkill(skill.name, !skill.enabled, selectedProfile || undefined);
125
180
  setSkills((prev) =>
126
181
  prev.map((s) =>
127
182
  s.name === skill.name ? { ...s, enabled: !s.enabled } : s,
@@ -142,6 +197,38 @@ export default function SkillsPage() {
142
197
  }
143
198
  };
144
199
 
200
+ /* ---- Refresh toolsets after a config change ---- */
201
+ const refreshToolsets = async () => {
202
+ try {
203
+ const tsets = await api.getToolsets();
204
+ setToolsets(tsets);
205
+ } catch {
206
+ /* non-fatal: the drawer already toasted on the failing write */
207
+ }
208
+ };
209
+
210
+ /* ---- Skill editor (create / edit SKILL.md) ---- */
211
+ const openCreateEditor = useCallback(() => {
212
+ setEditorSkill(null);
213
+ setEditorOpen(true);
214
+ }, []);
215
+ const openEditEditor = useCallback((skillName: string) => {
216
+ setEditorSkill(skillName);
217
+ setEditorOpen(true);
218
+ }, []);
219
+ const handleEditorSaved = useCallback(
220
+ (skillName: string) => {
221
+ showToast(`${skillName} saved ✓`, "success");
222
+ // Reload the list so a newly created skill (or an edited description)
223
+ // shows up immediately.
224
+ api
225
+ .getSkills(selectedProfile || undefined)
226
+ .then(setSkills)
227
+ .catch(() => {});
228
+ },
229
+ [selectedProfile, showToast],
230
+ );
231
+
145
232
  /* ---- Derived data ---- */
146
233
  const lowerSearch = search.toLowerCase();
147
234
  const isSearching = search.trim().length > 0;
@@ -197,7 +284,7 @@ export default function SkillsPage() {
197
284
  return;
198
285
  }
199
286
  setAfterTitle(
200
- <span className="whitespace-nowrap text-xs text-muted-foreground">
287
+ <span className="flex items-center gap-2 whitespace-nowrap text-xs text-muted-foreground">
201
288
  {t.skills.enabledOf
202
289
  .replace("{enabled}", String(enabledCount))
203
290
  .replace("{total}", String(skills.length))}
@@ -229,7 +316,15 @@ export default function SkillsPage() {
229
316
  setAfterTitle(null);
230
317
  setEnd(null);
231
318
  };
232
- }, [enabledCount, loading, search, setAfterTitle, setEnd, skills.length, t]);
319
+ }, [
320
+ enabledCount,
321
+ loading,
322
+ search,
323
+ setAfterTitle,
324
+ setEnd,
325
+ skills.length,
326
+ t,
327
+ ]);
233
328
 
234
329
  const filteredToolsets = useMemo(() => {
235
330
  return toolsets.filter(
@@ -369,6 +464,7 @@ export default function SkillsPage() {
369
464
  skill={skill}
370
465
  toggling={togglingSkills.has(skill.name)}
371
466
  onToggle={() => handleToggleSkill(skill)}
467
+ onEdit={() => openEditEditor(skill.name)}
372
468
  noDescriptionLabel={t.skills.noDescription}
373
469
  />
374
470
  ))}
@@ -390,11 +486,21 @@ export default function SkillsPage() {
390
486
  )
391
487
  : t.skills.all}
392
488
  </CardTitle>
393
- <Badge tone="secondary" className="text-xs">
394
- {t.skills.skillCount
395
- .replace("{count}", String(activeSkills.length))
396
- .replace("{s}", activeSkills.length !== 1 ? "s" : "")}
397
- </Badge>
489
+ <div className="flex items-center gap-2">
490
+ <Badge tone="secondary" className="text-xs">
491
+ {t.skills.skillCount
492
+ .replace("{count}", String(activeSkills.length))
493
+ .replace("{s}", activeSkills.length !== 1 ? "s" : "")}
494
+ </Badge>
495
+ <Button
496
+ size="sm"
497
+ outlined
498
+ onClick={openCreateEditor}
499
+ prefix={<Plus />}
500
+ >
501
+ New skill
502
+ </Button>
503
+ </div>
398
504
  </div>
399
505
  </CardHeader>
400
506
  <CardContent className="px-4 pb-4">
@@ -412,6 +518,7 @@ export default function SkillsPage() {
412
518
  skill={skill}
413
519
  toggling={togglingSkills.has(skill.name)}
414
520
  onToggle={() => handleToggleSkill(skill)}
521
+ onEdit={() => openEditEditor(skill.name)}
415
522
  noDescriptionLabel={t.skills.noDescription}
416
523
  />
417
524
  ))}
@@ -432,9 +539,7 @@ export default function SkillsPage() {
432
539
  <div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-3">
433
540
  {filteredToolsets.map((ts) => {
434
541
  const TsIcon = toolsetIcon(ts.name);
435
- const labelText =
436
- ts.label.replace(/^[\p{Emoji}\s]+/u, "").trim() ||
437
- ts.name;
542
+ const labelText = ts.label.trim() || ts.name;
438
543
 
439
544
  return (
440
545
  <Card key={ts.name} className="relative rounded-none">
@@ -486,6 +591,16 @@ export default function SkillsPage() {
486
591
  : t.skills.disabledForCli}
487
592
  </span>
488
593
  )}
594
+ <div className="mt-3">
595
+ <Button
596
+ size="sm"
597
+ outlined
598
+ onClick={() => setConfigToolset(ts)}
599
+ prefix={<Wrench />}
600
+ >
601
+ Configure
602
+ </Button>
603
+ </div>
489
604
  </div>
490
605
  </div>
491
606
  </CardContent>
@@ -496,10 +611,25 @@ export default function SkillsPage() {
496
611
  )}
497
612
  </>
498
613
  ) : (
499
- <HubBrowser showToast={showToast} />
614
+ <HubBrowser showToast={showToast} profile={selectedProfile || undefined} />
500
615
  )}
501
616
  </div>
502
617
  </div>
618
+ {configToolset && (
619
+ <ToolsetConfigDrawer
620
+ toolset={configToolset}
621
+ profile={selectedProfile || undefined}
622
+ onClose={() => setConfigToolset(null)}
623
+ onChanged={() => void refreshToolsets()}
624
+ />
625
+ )}
626
+ <SkillEditorDialog
627
+ open={editorOpen}
628
+ editName={editorSkill}
629
+ profile={selectedProfile || undefined}
630
+ onClose={() => setEditorOpen(false)}
631
+ onSaved={handleEditorSaved}
632
+ />
503
633
  <PluginSlot name="skills:bottom" />
504
634
  </div>
505
635
  );
@@ -509,6 +639,7 @@ function SkillRow({
509
639
  skill,
510
640
  toggling,
511
641
  onToggle,
642
+ onEdit,
512
643
  noDescriptionLabel,
513
644
  }: SkillRowProps) {
514
645
  return (
@@ -534,6 +665,16 @@ function SkillRow({
534
665
  {skill.description || noDescriptionLabel}
535
666
  </p>
536
667
  </div>
668
+ <Button
669
+ ghost
670
+ size="icon"
671
+ className="shrink-0 text-muted-foreground opacity-0 transition-opacity group-hover:opacity-100 focus-visible:opacity-100 hover:text-foreground"
672
+ title="Edit SKILL.md"
673
+ aria-label={`Edit ${skill.name}`}
674
+ onClick={onEdit}
675
+ >
676
+ <Pencil />
677
+ </Button>
537
678
  </div>
538
679
  );
539
680
  }
@@ -565,45 +706,136 @@ interface PanelItemProps {
565
706
  interface SkillRowProps {
566
707
  noDescriptionLabel: string;
567
708
  onToggle: () => void;
709
+ onEdit: () => void;
568
710
  skill: SkillInfo;
569
711
  toggling: boolean;
570
712
  }
571
713
 
572
714
  /* ------------------------------------------------------------------ */
573
- /* Hub browser — search the skill hub, install by identifier */
715
+ /* Hub browser — search the skill hub, preview, scan, install */
574
716
  /* ------------------------------------------------------------------ */
575
717
 
718
+ /** Map a trust level to a Badge tone + label + icon. */
719
+ function trustVisual(level: string): {
720
+ tone: "success" | "secondary" | "warning" | "outline";
721
+ label: string;
722
+ } {
723
+ switch (level) {
724
+ case "trusted":
725
+ return { tone: "success", label: "trusted" };
726
+ case "builtin":
727
+ return { tone: "secondary", label: "builtin" };
728
+ case "community":
729
+ return { tone: "warning", label: "community" };
730
+ default:
731
+ return { tone: "outline", label: level || "unknown" };
732
+ }
733
+ }
734
+
735
+ /** Map a scan verdict to tone + icon. */
736
+ function verdictVisual(verdict: string): {
737
+ tone: "success" | "warning" | "destructive";
738
+ Icon: React.ComponentType<{ className?: string }>;
739
+ label: string;
740
+ } {
741
+ switch (verdict) {
742
+ case "safe":
743
+ return { tone: "success", Icon: ShieldCheck, label: "Safe" };
744
+ case "caution":
745
+ return { tone: "warning", Icon: ShieldAlert, label: "Caution" };
746
+ case "dangerous":
747
+ return { tone: "destructive", Icon: ShieldAlert, label: "Dangerous" };
748
+ default:
749
+ return { tone: "warning", Icon: ShieldQuestion, label: verdict };
750
+ }
751
+ }
752
+
753
+ const SEVERITY_TONE: Record<string, "destructive" | "warning" | "secondary" | "outline"> = {
754
+ critical: "destructive",
755
+ high: "destructive",
756
+ medium: "warning",
757
+ low: "secondary",
758
+ };
759
+
576
760
  function HubBrowser({
577
761
  showToast,
762
+ profile,
578
763
  }: {
579
764
  showToast: (msg: string, kind: "success" | "error") => void;
765
+ /** Optional profile scoping installs + installed-state badges. */
766
+ profile?: string;
580
767
  }) {
581
768
  const [query, setQuery] = useState("");
582
769
  const [results, setResults] = useState<SkillHubResult[]>([]);
583
770
  const [searching, setSearching] = useState(false);
584
771
  const [searched, setSearched] = useState(false);
585
- // Live action log for the most recent install/update (tailed via action status).
772
+ const [sourceCounts, setSourceCounts] = useState<Record<string, number>>({});
773
+ const [timedOut, setTimedOut] = useState<string[]>([]);
774
+ const [searchMs, setSearchMs] = useState<number | null>(null);
775
+
776
+ // Landing state: which hubs are wired up + featured skills.
777
+ const [sources, setSources] = useState<SkillHubSource[]>([]);
778
+ const [featured, setFeatured] = useState<SkillHubResult[]>([]);
779
+ const [sourcesLoading, setSourcesLoading] = useState(true);
780
+
781
+ // identifier -> installed entry (drives "Installed" badges).
782
+ const [installed, setInstalled] = useState<Record<string, SkillHubInstalledEntry>>({});
783
+
784
+ // Live action log for the most recent install/update.
586
785
  const [action, setAction] = useState<string | null>(null);
587
786
  const [actionLog, setActionLog] = useState<string[]>([]);
588
787
  const [actionRunning, setActionRunning] = useState(false);
589
788
 
590
- const runSearch = async () => {
789
+ // Detail dialog (preview + scan for a single skill).
790
+ const [detail, setDetail] = useState<SkillHubResult | null>(null);
791
+
792
+ /* ---- Load connected hubs + featured skills on mount ---- */
793
+ useEffect(() => {
794
+ let cancelled = false;
795
+ api
796
+ .getSkillHubSources(profile)
797
+ .then((r) => {
798
+ if (cancelled) return;
799
+ setSources(r.sources);
800
+ setFeatured(r.featured);
801
+ setInstalled(r.installed);
802
+ })
803
+ .catch(() => {
804
+ /* leave landing minimal on failure */
805
+ })
806
+ .finally(() => {
807
+ if (!cancelled) setSourcesLoading(false);
808
+ });
809
+ return () => {
810
+ cancelled = true;
811
+ };
812
+ }, [profile]);
813
+
814
+ /* ---- Search ---- */
815
+ const runSearch = useCallback(async () => {
591
816
  const q = query.trim();
592
817
  if (!q) return;
593
818
  setSearching(true);
594
819
  setSearched(true);
820
+ const t0 = performance.now();
595
821
  try {
596
- const r = await api.searchSkillsHub(q);
822
+ const r = await api.searchSkillsHub(q, "all", 20, profile);
597
823
  setResults(r.results);
824
+ setSourceCounts(r.source_counts || {});
825
+ setTimedOut(r.timed_out || []);
826
+ setInstalled((prev) => ({ ...prev, ...(r.installed || {}) }));
598
827
  } catch (e) {
599
828
  showToast(`Hub search failed: ${e}`, "error");
600
829
  setResults([]);
830
+ setSourceCounts({});
831
+ setTimedOut([]);
601
832
  } finally {
833
+ setSearchMs(Math.round(performance.now() - t0));
602
834
  setSearching(false);
603
835
  }
604
- };
836
+ }, [query, showToast, profile]);
605
837
 
606
- // Poll a spawned action's log until it exits.
838
+ /* ---- Poll a spawned action's log until it exits ---- */
607
839
  useEffect(() => {
608
840
  if (!action) return;
609
841
  let cancelled = false;
@@ -614,7 +846,15 @@ function HubBrowser({
614
846
  if (cancelled) return;
615
847
  setActionLog(st.lines);
616
848
  setActionRunning(st.running);
617
- if (st.running) timer = setTimeout(poll, 1200);
849
+ if (st.running) {
850
+ timer = setTimeout(poll, 1200);
851
+ } else {
852
+ // Install finished — refresh installed-state so badges update.
853
+ api
854
+ .getSkillHubSources(profile)
855
+ .then((r) => !cancelled && setInstalled(r.installed))
856
+ .catch(() => {});
857
+ }
618
858
  } catch {
619
859
  if (!cancelled) setActionRunning(false);
620
860
  }
@@ -624,23 +864,27 @@ function HubBrowser({
624
864
  cancelled = true;
625
865
  if (timer) clearTimeout(timer);
626
866
  };
627
- }, [action]);
867
+ }, [action, profile]);
628
868
 
629
- const install = async (identifier: string) => {
630
- try {
631
- const res = await api.installSkillFromHub(identifier);
632
- showToast(`Installing ${identifier}…`, "success");
633
- setActionLog([]);
634
- setActionRunning(true);
635
- setAction(res.name);
636
- } catch (e) {
637
- showToast(`Install failed: ${e}`, "error");
638
- }
639
- };
869
+ const install = useCallback(
870
+ async (identifier: string) => {
871
+ try {
872
+ const res = await api.installSkillFromHub(identifier, profile);
873
+ showToast(`Installing ${identifier}…`, "success");
874
+ setActionLog([]);
875
+ setActionRunning(true);
876
+ setAction(res.name);
877
+ setDetail(null);
878
+ } catch (e) {
879
+ showToast(`Install failed: ${e}`, "error");
880
+ }
881
+ },
882
+ [showToast, profile],
883
+ );
640
884
 
641
- const updateAll = async () => {
885
+ const updateAll = useCallback(async () => {
642
886
  try {
643
- const res = await api.updateSkillsFromHub();
887
+ const res = await api.updateSkillsFromHub(profile);
644
888
  showToast("Updating installed skills…", "success");
645
889
  setActionLog([]);
646
890
  setActionRunning(true);
@@ -648,10 +892,18 @@ function HubBrowser({
648
892
  } catch (e) {
649
893
  showToast(`Update failed: ${e}`, "error");
650
894
  }
651
- };
895
+ }, [showToast, profile]);
896
+
897
+ const isInstalled = useCallback(
898
+ (identifier: string) => Boolean(installed[identifier]),
899
+ [installed],
900
+ );
901
+
902
+ const showLanding = !searched && !searching;
652
903
 
653
904
  return (
654
905
  <div className="flex flex-col gap-3">
906
+ {/* ── Search bar ── */}
655
907
  <Card className="rounded-none">
656
908
  <CardContent className="py-4 flex flex-col gap-3">
657
909
  <div className="flex items-center gap-2">
@@ -684,13 +936,13 @@ function HubBrowser({
684
936
  Update all
685
937
  </Button>
686
938
  </div>
687
- <p className="text-xs text-muted-foreground">
688
- Results come from the same sources as <span className="font-mono">hermes skills search</span>.
689
- Installs run in the background; the log streams below.
690
- </p>
939
+
940
+ {/* Connected hubs strip proves the tab is wired up. */}
941
+ <ConnectedHubs sources={sources} loading={sourcesLoading} />
691
942
  </CardContent>
692
943
  </Card>
693
944
 
945
+ {/* ── Install/update action log ── */}
694
946
  {action && (
695
947
  <Card className="rounded-none">
696
948
  <CardContent className="py-3">
@@ -702,6 +954,17 @@ function HubBrowser({
702
954
  ) : (
703
955
  <Badge tone="success">done</Badge>
704
956
  )}
957
+ {!actionRunning && (
958
+ <Button
959
+ ghost
960
+ size="xs"
961
+ className="ml-auto text-muted-foreground"
962
+ onClick={() => setAction(null)}
963
+ aria-label="Dismiss"
964
+ >
965
+ <X className="h-3.5 w-3.5" />
966
+ </Button>
967
+ )}
705
968
  </div>
706
969
  <pre className="max-h-48 overflow-auto whitespace-pre-wrap break-words bg-background/50 border border-border p-2 text-xs font-mono text-muted-foreground">
707
970
  {actionLog.length ? actionLog.join("\n") : "Starting…"}
@@ -710,46 +973,564 @@ function HubBrowser({
710
973
  </Card>
711
974
  )}
712
975
 
976
+ {/* ── Landing: featured skills (before any search) ── */}
977
+ {showLanding && (
978
+ <>
979
+ {sourcesLoading ? (
980
+ <div className="flex items-center justify-center py-12">
981
+ <Spinner className="text-xl text-primary" />
982
+ </div>
983
+ ) : featured.length > 0 ? (
984
+ <div className="flex flex-col gap-2">
985
+ <div className="flex items-center gap-2 px-1">
986
+ <Sparkles className="h-3.5 w-3.5 text-primary" />
987
+ <span className="font-mondwest text-display text-xs tracking-[0.12em] text-text-secondary uppercase">
988
+ Featured skills
989
+ </span>
990
+ <span className="text-xs text-text-tertiary">
991
+ from the Hermes index — search above for thousands more
992
+ </span>
993
+ </div>
994
+ {featured.map((r) => (
995
+ <HubResultCard
996
+ key={r.identifier}
997
+ result={r}
998
+ installed={isInstalled(r.identifier)}
999
+ onOpen={() => setDetail(r)}
1000
+ onInstall={() => void install(r.identifier)}
1001
+ />
1002
+ ))}
1003
+ </div>
1004
+ ) : (
1005
+ <Card className="rounded-none">
1006
+ <CardContent className="py-10 text-center text-sm text-muted-foreground">
1007
+ Search the hub above to browse installable skills from the
1008
+ connected sources.
1009
+ </CardContent>
1010
+ </Card>
1011
+ )}
1012
+ </>
1013
+ )}
1014
+
1015
+ {/* ── Searching spinner ── */}
713
1016
  {searching && (
714
1017
  <div className="flex items-center justify-center py-8">
715
1018
  <Spinner className="text-xl text-primary" />
716
1019
  </div>
717
1020
  )}
718
1021
 
719
- {!searching && searched && results.length === 0 && (
720
- <Card className="rounded-none">
721
- <CardContent className="py-8 text-center text-sm text-muted-foreground">
722
- No matching skills found in the hub.
723
- </CardContent>
724
- </Card>
1022
+ {/* ── Search results ── */}
1023
+ {!searching && searched && (
1024
+ <>
1025
+ <SearchMeta
1026
+ count={results.length}
1027
+ sourceCounts={sourceCounts}
1028
+ timedOut={timedOut}
1029
+ ms={searchMs}
1030
+ />
1031
+ {results.length === 0 ? (
1032
+ <Card className="rounded-none">
1033
+ <CardContent className="py-8 text-center text-sm text-muted-foreground">
1034
+ No matching skills found in the hub.
1035
+ </CardContent>
1036
+ </Card>
1037
+ ) : (
1038
+ results.map((r) => (
1039
+ <HubResultCard
1040
+ key={r.identifier}
1041
+ result={r}
1042
+ installed={isInstalled(r.identifier)}
1043
+ onOpen={() => setDetail(r)}
1044
+ onInstall={() => void install(r.identifier)}
1045
+ />
1046
+ ))
1047
+ )}
1048
+ </>
725
1049
  )}
726
1050
 
727
- {results.map((r) => (
728
- <Card key={r.identifier} className="rounded-none">
729
- <CardContent className="py-3 flex items-start gap-3">
730
- <div className="flex-1 min-w-0">
731
- <div className="flex items-center gap-2 mb-0.5">
732
- <span className="font-mono-ui text-sm">{r.name}</span>
733
- <Badge tone="secondary" className="text-xs">{r.source}</Badge>
734
- <Badge tone="outline" className="text-xs">{r.trust_level}</Badge>
735
- </div>
736
- <p className="text-xs text-text-secondary">{r.description}</p>
737
- <p className="text-xs font-mono text-text-tertiary truncate mt-0.5">
738
- {r.identifier}
739
- </p>
740
- </div>
1051
+ {/* ── Detail dialog: preview + scan ── */}
1052
+ {detail && (
1053
+ <SkillDetailDialog
1054
+ result={detail}
1055
+ installed={isInstalled(detail.identifier)}
1056
+ onClose={() => setDetail(null)}
1057
+ onInstall={() => void install(detail.identifier)}
1058
+ showToast={showToast}
1059
+ />
1060
+ )}
1061
+ </div>
1062
+ );
1063
+ }
1064
+
1065
+ /* ---- Connected hubs strip ---- */
1066
+ function ConnectedHubs({
1067
+ sources,
1068
+ loading,
1069
+ }: {
1070
+ sources: SkillHubSource[];
1071
+ loading: boolean;
1072
+ }) {
1073
+ if (loading) {
1074
+ return (
1075
+ <p className="text-xs text-muted-foreground">Connecting to skill hubs…</p>
1076
+ );
1077
+ }
1078
+ if (sources.length === 0) {
1079
+ return (
1080
+ <p className="text-xs text-muted-foreground">
1081
+ Results come from the same sources as{" "}
1082
+ <span className="font-mono">hermes skills search</span>.
1083
+ </p>
1084
+ );
1085
+ }
1086
+ return (
1087
+ <div className="flex flex-wrap items-center gap-1.5">
1088
+ <span className="flex items-center gap-1 text-xs text-text-tertiary">
1089
+ <Globe className="h-3 w-3" />
1090
+ Connected hubs:
1091
+ </span>
1092
+ {sources.map((s) => {
1093
+ const down =
1094
+ (s.id === "hermes-index" && s.available === false) ||
1095
+ (s.id === "github" && s.rate_limited === true);
1096
+ return (
1097
+ <Badge
1098
+ key={s.id}
1099
+ tone={down ? "outline" : "secondary"}
1100
+ className={cn("text-xs", down && "opacity-60")}
1101
+ title={
1102
+ s.id === "github" && s.rate_limited
1103
+ ? "GitHub API rate-limited — set GITHUB_TOKEN to raise the limit"
1104
+ : s.id === "hermes-index" && s.available === false
1105
+ ? "Centralized index unavailable — falling back to live sources"
1106
+ : undefined
1107
+ }
1108
+ >
1109
+ {s.label}
1110
+ {s.id === "github" && s.rate_limited ? " (rate-limited)" : ""}
1111
+ </Badge>
1112
+ );
1113
+ })}
1114
+ </div>
1115
+ );
1116
+ }
1117
+
1118
+ /* ---- Search result-count + per-source breakdown ---- */
1119
+ function SearchMeta({
1120
+ count,
1121
+ sourceCounts,
1122
+ timedOut,
1123
+ ms,
1124
+ }: {
1125
+ count: number;
1126
+ sourceCounts: Record<string, number>;
1127
+ timedOut: string[];
1128
+ ms: number | null;
1129
+ }) {
1130
+ const entries = Object.entries(sourceCounts).filter(([, n]) => n > 0);
1131
+ return (
1132
+ <div className="flex flex-wrap items-center gap-2 px-1 text-xs text-text-tertiary">
1133
+ <Badge tone="secondary" className="text-xs">
1134
+ {count} result{count !== 1 ? "s" : ""}
1135
+ </Badge>
1136
+ {ms != null && <span>{(ms / 1000).toFixed(1)}s</span>}
1137
+ {entries.length > 0 && (
1138
+ <span className="flex flex-wrap items-center gap-1.5">
1139
+ {entries.map(([sid, n]) => (
1140
+ <span key={sid} className="font-mono">
1141
+ {sid}:{n}
1142
+ </span>
1143
+ ))}
1144
+ </span>
1145
+ )}
1146
+ {timedOut.length > 0 && (
1147
+ <span className="flex items-center gap-1 text-amber-400">
1148
+ <AlertTriangle className="h-3 w-3" />
1149
+ {timedOut.join(", ")} timed out
1150
+ </span>
1151
+ )}
1152
+ </div>
1153
+ );
1154
+ }
1155
+
1156
+ /* ---- One result card ---- */
1157
+ function HubResultCard({
1158
+ result,
1159
+ installed,
1160
+ onOpen,
1161
+ onInstall,
1162
+ }: {
1163
+ result: SkillHubResult;
1164
+ installed: boolean;
1165
+ onOpen: () => void;
1166
+ onInstall: () => void;
1167
+ }) {
1168
+ const trust = trustVisual(result.trust_level);
1169
+ return (
1170
+ <Card className="rounded-none transition-colors hover:bg-muted/30">
1171
+ <CardContent className="py-3 flex items-start gap-3">
1172
+ <button
1173
+ type="button"
1174
+ className="flex-1 min-w-0 text-left"
1175
+ onClick={onOpen}
1176
+ aria-label={`Open ${result.name}`}
1177
+ >
1178
+ <div className="flex flex-wrap items-center gap-2 mb-0.5">
1179
+ <span className="font-mono-ui text-sm hover:underline">
1180
+ {result.name}
1181
+ </span>
1182
+ <Badge tone={trust.tone} className="text-xs">
1183
+ {trust.label}
1184
+ </Badge>
1185
+ <Badge tone="secondary" className="text-xs">
1186
+ {result.source}
1187
+ </Badge>
1188
+ {installed && (
1189
+ <Badge tone="success" className="text-xs">
1190
+ installed
1191
+ </Badge>
1192
+ )}
1193
+ </div>
1194
+ <p className="text-xs text-text-secondary line-clamp-2">
1195
+ {result.description}
1196
+ </p>
1197
+ <div className="flex flex-wrap items-center gap-1 mt-1">
1198
+ {result.tags.slice(0, 5).map((tag) => (
1199
+ <span
1200
+ key={tag}
1201
+ className="text-[0.65rem] font-mono text-text-tertiary border border-border px-1 py-px"
1202
+ >
1203
+ {tag}
1204
+ </span>
1205
+ ))}
1206
+ </div>
1207
+ <p className="text-xs font-mono text-text-tertiary truncate mt-1">
1208
+ {result.identifier}
1209
+ </p>
1210
+ </button>
1211
+ <div className="flex shrink-0 flex-col gap-1.5">
1212
+ <Button
1213
+ size="sm"
1214
+ outlined
1215
+ onClick={onOpen}
1216
+ prefix={<FileText className="h-3.5 w-3.5" />}
1217
+ >
1218
+ Details
1219
+ </Button>
1220
+ {installed ? (
1221
+ <Button size="sm" ghost disabled prefix={<CheckCircle2 className="h-3.5 w-3.5" />}>
1222
+ Installed
1223
+ </Button>
1224
+ ) : (
741
1225
  <Button
742
1226
  size="sm"
743
- outlined
744
- className="shrink-0"
745
- onClick={() => void install(r.identifier)}
1227
+ onClick={onInstall}
746
1228
  prefix={<Download className="h-3.5 w-3.5" />}
747
1229
  >
748
1230
  Install
749
1231
  </Button>
750
- </CardContent>
751
- </Card>
752
- ))}
1232
+ )}
1233
+ </div>
1234
+ </CardContent>
1235
+ </Card>
1236
+ );
1237
+ }
1238
+
1239
+ /* ---- Detail dialog: SKILL.md preview + on-demand security scan ---- */
1240
+ function SkillDetailDialog({
1241
+ result,
1242
+ installed,
1243
+ onClose,
1244
+ onInstall,
1245
+ showToast,
1246
+ }: {
1247
+ result: SkillHubResult;
1248
+ installed: boolean;
1249
+ onClose: () => void;
1250
+ onInstall: () => void;
1251
+ showToast: (msg: string, kind: "success" | "error") => void;
1252
+ }) {
1253
+ const [tab, setTab] = useState<"readme" | "scan">("readme");
1254
+ const [preview, setPreview] = useState<SkillHubPreview | null>(null);
1255
+ const [previewLoading, setPreviewLoading] = useState(true);
1256
+ const [scan, setScan] = useState<SkillHubScan | null>(null);
1257
+ const [scanning, setScanning] = useState(false);
1258
+ const trust = trustVisual(result.trust_level);
1259
+
1260
+ useEffect(() => {
1261
+ let cancelled = false;
1262
+ setPreviewLoading(true);
1263
+ api
1264
+ .previewSkillFromHub(result.identifier)
1265
+ .then((p) => !cancelled && setPreview(p))
1266
+ .catch((e) => {
1267
+ if (!cancelled) showToast(`Preview failed: ${e}`, "error");
1268
+ })
1269
+ .finally(() => !cancelled && setPreviewLoading(false));
1270
+ return () => {
1271
+ cancelled = true;
1272
+ };
1273
+ }, [result.identifier, showToast]);
1274
+
1275
+ const runScan = useCallback(async () => {
1276
+ setScanning(true);
1277
+ setTab("scan");
1278
+ try {
1279
+ const s = await api.scanSkillFromHub(result.identifier);
1280
+ setScan(s);
1281
+ } catch (e) {
1282
+ showToast(`Scan failed: ${e}`, "error");
1283
+ } finally {
1284
+ setScanning(false);
1285
+ }
1286
+ }, [result.identifier, showToast]);
1287
+
1288
+ return (
1289
+ <Dialog open onOpenChange={(o: boolean) => !o && onClose()}>
1290
+ <DialogContent className="max-w-3xl rounded-none">
1291
+ <DialogHeader>
1292
+ <DialogTitle className="flex flex-wrap items-center gap-2 text-sm">
1293
+ <Package className="h-4 w-4" />
1294
+ {result.name}
1295
+ <Badge tone={trust.tone} className="text-xs">
1296
+ {trust.label}
1297
+ </Badge>
1298
+ <Badge tone="secondary" className="text-xs">
1299
+ {result.source}
1300
+ </Badge>
1301
+ {installed && (
1302
+ <Badge tone="success" className="text-xs">
1303
+ installed
1304
+ </Badge>
1305
+ )}
1306
+ </DialogTitle>
1307
+ <DialogDescription className="sr-only">
1308
+ Preview the SKILL.md source and run a security scan for {result.name}{" "}
1309
+ before installing.
1310
+ </DialogDescription>
1311
+ </DialogHeader>
1312
+
1313
+ <div className="mt-1 flex flex-col gap-1">
1314
+ <p className="text-xs text-text-secondary">{result.description}</p>
1315
+ <p className="text-xs font-mono text-text-tertiary truncate">
1316
+ {result.identifier}
1317
+ </p>
1318
+ </div>
1319
+
1320
+ {/* Action row */}
1321
+ <div className="mt-3 flex flex-wrap items-center gap-2 border-y border-border py-2.5">
1322
+ <Button
1323
+ size="sm"
1324
+ outlined={tab !== "readme"}
1325
+ onClick={() => setTab("readme")}
1326
+ prefix={<FileText className="h-3.5 w-3.5" />}
1327
+ >
1328
+ Read SKILL.md
1329
+ </Button>
1330
+ <Button
1331
+ size="sm"
1332
+ outlined={tab !== "scan"}
1333
+ onClick={() => void runScan()}
1334
+ disabled={scanning}
1335
+ prefix={
1336
+ scanning ? (
1337
+ <Loader2 className="h-3.5 w-3.5 animate-spin" />
1338
+ ) : (
1339
+ <Shield className="h-3.5 w-3.5" />
1340
+ )
1341
+ }
1342
+ >
1343
+ {scan ? "Re-scan" : "Security scan"}
1344
+ </Button>
1345
+ <div className="ml-auto flex items-center gap-3">
1346
+ {result.repo && (
1347
+ <a
1348
+ href={`https://github.com/${result.repo}`}
1349
+ target="_blank"
1350
+ rel="noreferrer"
1351
+ className="inline-flex items-center gap-1 text-xs text-primary hover:underline"
1352
+ >
1353
+ <ExternalLink className="h-3.5 w-3.5" />
1354
+ {result.repo}
1355
+ </a>
1356
+ )}
1357
+ {installed ? (
1358
+ <Button size="sm" ghost disabled prefix={<CheckCircle2 className="h-3.5 w-3.5" />}>
1359
+ Installed
1360
+ </Button>
1361
+ ) : (
1362
+ <Button
1363
+ size="sm"
1364
+ onClick={onInstall}
1365
+ prefix={<Download className="h-3.5 w-3.5" />}
1366
+ >
1367
+ Install
1368
+ </Button>
1369
+ )}
1370
+ </div>
1371
+ </div>
1372
+
1373
+ {/* Body */}
1374
+ <div className="mt-3 max-h-[55vh] overflow-auto">
1375
+ {tab === "readme" ? (
1376
+ previewLoading ? (
1377
+ <div className="flex items-center justify-center py-12">
1378
+ <Spinner className="text-xl text-primary" />
1379
+ </div>
1380
+ ) : preview ? (
1381
+ <div className="flex flex-col gap-2.5">
1382
+ {preview.tags.length > 0 && (
1383
+ <div className="flex flex-wrap items-center gap-1">
1384
+ {preview.tags.map((tag) => (
1385
+ <span
1386
+ key={tag}
1387
+ className="text-[0.65rem] font-mono text-text-tertiary border border-border px-1 py-px"
1388
+ >
1389
+ {tag}
1390
+ </span>
1391
+ ))}
1392
+ </div>
1393
+ )}
1394
+ {preview.files.length > 0 && (
1395
+ <div className="text-xs text-text-tertiary">
1396
+ <span className="font-mondwest tracking-[0.1em] uppercase">
1397
+ Files:{" "}
1398
+ </span>
1399
+ <span className="font-mono">{preview.files.join(" ")}</span>
1400
+ </div>
1401
+ )}
1402
+ <pre className="whitespace-pre-wrap break-words bg-background/50 border border-border p-3 text-xs font-mono text-text-secondary leading-relaxed">
1403
+ {(preview.skill_md || "").trim() || "(SKILL.md is empty)"}
1404
+ </pre>
1405
+ </div>
1406
+ ) : (
1407
+ <p className="text-sm text-muted-foreground text-center py-10">
1408
+ Couldn't load the skill source.
1409
+ </p>
1410
+ )
1411
+ ) : (
1412
+ <ScanPanel scan={scan} scanning={scanning} />
1413
+ )}
1414
+ </div>
1415
+ </DialogContent>
1416
+ </Dialog>
1417
+ );
1418
+ }
1419
+
1420
+ /* ---- Visual security-scan result ---- */
1421
+ function ScanPanel({
1422
+ scan,
1423
+ scanning,
1424
+ }: {
1425
+ scan: SkillHubScan | null;
1426
+ scanning: boolean;
1427
+ }) {
1428
+ if (scanning && !scan) {
1429
+ return (
1430
+ <div className="flex flex-col items-center justify-center gap-2 py-12">
1431
+ <Loader2 className="h-6 w-6 animate-spin text-primary" />
1432
+ <span className="text-xs text-muted-foreground">
1433
+ Fetching, quarantining, and scanning…
1434
+ </span>
1435
+ </div>
1436
+ );
1437
+ }
1438
+ if (!scan) {
1439
+ return (
1440
+ <p className="text-sm text-muted-foreground text-center py-10">
1441
+ Run a security scan to inspect this skill for risky patterns before
1442
+ installing.
1443
+ </p>
1444
+ );
1445
+ }
1446
+
1447
+ const v = verdictVisual(scan.verdict);
1448
+ const policyTone =
1449
+ scan.policy === "allow"
1450
+ ? "success"
1451
+ : scan.policy === "ask"
1452
+ ? "warning"
1453
+ : "destructive";
1454
+ const policyLabel =
1455
+ scan.policy === "allow"
1456
+ ? "Install allowed"
1457
+ : scan.policy === "ask"
1458
+ ? "Needs confirmation"
1459
+ : "Install blocked";
1460
+
1461
+ return (
1462
+ <div className="flex flex-col gap-3">
1463
+ {/* Verdict header */}
1464
+ <div className="flex flex-wrap items-center gap-2 border border-border p-3">
1465
+ <v.Icon
1466
+ className={cn(
1467
+ "h-6 w-6",
1468
+ scan.verdict === "safe"
1469
+ ? "text-emerald-400"
1470
+ : scan.verdict === "dangerous"
1471
+ ? "text-red-400"
1472
+ : "text-amber-400",
1473
+ )}
1474
+ />
1475
+ <div className="flex flex-col">
1476
+ <div className="flex items-center gap-2">
1477
+ <span className="text-sm font-medium">Verdict: {v.label}</span>
1478
+ <Badge tone={v.tone} className="text-xs">
1479
+ {scan.verdict}
1480
+ </Badge>
1481
+ </div>
1482
+ <span className="text-xs text-text-tertiary">
1483
+ {scan.trust_level} source · {scan.findings.length} finding
1484
+ {scan.findings.length !== 1 ? "s" : ""}
1485
+ </span>
1486
+ </div>
1487
+ <Badge tone={policyTone} className="ml-auto text-xs">
1488
+ {policyLabel}
1489
+ </Badge>
1490
+ </div>
1491
+
1492
+ {/* Severity tally */}
1493
+ <div className="flex flex-wrap items-center gap-1.5">
1494
+ {(["critical", "high", "medium", "low"] as const).map((sev) => {
1495
+ const n = scan.severity_counts[sev] || 0;
1496
+ if (n === 0) return null;
1497
+ return (
1498
+ <Badge key={sev} tone={SEVERITY_TONE[sev]} className="text-xs">
1499
+ {n} {sev}
1500
+ </Badge>
1501
+ );
1502
+ })}
1503
+ {scan.findings.length === 0 && (
1504
+ <span className="flex items-center gap-1 text-xs text-emerald-400">
1505
+ <CheckCircle2 className="h-3.5 w-3.5" />
1506
+ No risky patterns detected
1507
+ </span>
1508
+ )}
1509
+ </div>
1510
+
1511
+ <p className="text-xs text-text-tertiary">{scan.policy_reason}</p>
1512
+
1513
+ {/* Findings */}
1514
+ {scan.findings.length > 0 && (
1515
+ <div className="flex flex-col border border-border divide-y divide-border">
1516
+ {scan.findings.map((f, i) => (
1517
+ <div key={i} className="flex items-start gap-2 p-2">
1518
+ <Badge tone={SEVERITY_TONE[f.severity] || "outline"} className="text-xs shrink-0">
1519
+ {f.severity}
1520
+ </Badge>
1521
+ <div className="flex-1 min-w-0">
1522
+ <div className="flex flex-wrap items-center gap-2">
1523
+ <span className="text-xs font-medium">{f.category}</span>
1524
+ <span className="text-xs font-mono text-text-tertiary truncate">
1525
+ {f.file}:{f.line}
1526
+ </span>
1527
+ </div>
1528
+ <p className="text-xs text-text-secondary">{f.description}</p>
1529
+ </div>
1530
+ </div>
1531
+ ))}
1532
+ </div>
1533
+ )}
753
1534
  </div>
754
1535
  );
755
1536
  }