@openparachute/agent 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (501) hide show
  1. package/.claude/scheduled_tasks.lock +1 -0
  2. package/.claude/settings.json +5 -0
  3. package/.claude/skills/add-atomic-chat-tool/SKILL.md +243 -0
  4. package/.claude/skills/add-atomic-chat-tool/atomic-chat-mcp-stdio.ts +229 -0
  5. package/.claude/skills/add-codex/SKILL.md +161 -0
  6. package/.claude/skills/add-dashboard/SKILL.md +138 -0
  7. package/.claude/skills/add-dashboard/resources/dashboard-pusher.ts +495 -0
  8. package/.claude/skills/add-emacs/SKILL.md +296 -0
  9. package/.claude/skills/add-gcal-tool/SKILL.md +210 -0
  10. package/.claude/skills/add-gchat/REMOVE.md +6 -0
  11. package/.claude/skills/add-gchat/SKILL.md +92 -0
  12. package/.claude/skills/add-gchat/VERIFY.md +3 -0
  13. package/.claude/skills/add-github/REMOVE.md +6 -0
  14. package/.claude/skills/add-github/SKILL.md +148 -0
  15. package/.claude/skills/add-github/VERIFY.md +3 -0
  16. package/.claude/skills/add-gmail-tool/SKILL.md +229 -0
  17. package/.claude/skills/add-imessage/REMOVE.md +6 -0
  18. package/.claude/skills/add-imessage/SKILL.md +113 -0
  19. package/.claude/skills/add-imessage/VERIFY.md +3 -0
  20. package/.claude/skills/add-karpathy-llm-wiki/SKILL.md +110 -0
  21. package/.claude/skills/add-karpathy-llm-wiki/llm-wiki.md +75 -0
  22. package/.claude/skills/add-linear/REMOVE.md +6 -0
  23. package/.claude/skills/add-linear/SKILL.md +168 -0
  24. package/.claude/skills/add-linear/VERIFY.md +3 -0
  25. package/.claude/skills/add-macos-statusbar/SKILL.md +133 -0
  26. package/.claude/skills/add-macos-statusbar/add/src/statusbar.swift +147 -0
  27. package/.claude/skills/add-matrix/REMOVE.md +6 -0
  28. package/.claude/skills/add-matrix/SKILL.md +148 -0
  29. package/.claude/skills/add-matrix/VERIFY.md +3 -0
  30. package/.claude/skills/add-ollama-provider/SKILL.md +179 -0
  31. package/.claude/skills/add-ollama-tool/SKILL.md +193 -0
  32. package/.claude/skills/add-opencode/SKILL.md +229 -0
  33. package/.claude/skills/add-parallel/SKILL.md +290 -0
  34. package/.claude/skills/add-resend/REMOVE.md +6 -0
  35. package/.claude/skills/add-resend/SKILL.md +93 -0
  36. package/.claude/skills/add-resend/VERIFY.md +3 -0
  37. package/.claude/skills/add-signal/REMOVE.md +13 -0
  38. package/.claude/skills/add-signal/SKILL.md +318 -0
  39. package/.claude/skills/add-signal/VERIFY.md +5 -0
  40. package/.claude/skills/add-slack/REMOVE.md +6 -0
  41. package/.claude/skills/add-slack/SKILL.md +112 -0
  42. package/.claude/skills/add-slack/VERIFY.md +3 -0
  43. package/.claude/skills/add-teams/REMOVE.md +6 -0
  44. package/.claude/skills/add-teams/SKILL.md +207 -0
  45. package/.claude/skills/add-teams/VERIFY.md +3 -0
  46. package/.claude/skills/add-vercel/SKILL.md +147 -0
  47. package/.claude/skills/add-vercel/container-skills/vercel-cli/SKILL.md +103 -0
  48. package/.claude/skills/add-webex/REMOVE.md +6 -0
  49. package/.claude/skills/add-webex/SKILL.md +88 -0
  50. package/.claude/skills/add-webex/VERIFY.md +3 -0
  51. package/.claude/skills/add-wechat/REMOVE.md +49 -0
  52. package/.claude/skills/add-wechat/SKILL.md +170 -0
  53. package/.claude/skills/add-wechat/scripts/wire-dm.ts +172 -0
  54. package/.claude/skills/add-whatsapp/SKILL.md +264 -0
  55. package/.claude/skills/add-whatsapp-cloud/REMOVE.md +6 -0
  56. package/.claude/skills/add-whatsapp-cloud/SKILL.md +95 -0
  57. package/.claude/skills/add-whatsapp-cloud/VERIFY.md +3 -0
  58. package/.claude/skills/claw/SKILL.md +131 -0
  59. package/.claude/skills/claw/scripts/claw +374 -0
  60. package/.claude/skills/convert-to-apple-container/SKILL.md +212 -0
  61. package/.claude/skills/customize/SKILL.md +110 -0
  62. package/.claude/skills/debug/SKILL.md +349 -0
  63. package/.claude/skills/get-qodo-rules/SKILL.md +122 -0
  64. package/.claude/skills/get-qodo-rules/references/output-format.md +41 -0
  65. package/.claude/skills/get-qodo-rules/references/pagination.md +33 -0
  66. package/.claude/skills/get-qodo-rules/references/repository-scope.md +26 -0
  67. package/.claude/skills/init-first-agent/SKILL.md +120 -0
  68. package/.claude/skills/init-onecli/SKILL.md +270 -0
  69. package/.claude/skills/manage-channels/SKILL.md +87 -0
  70. package/.claude/skills/manage-mounts/SKILL.md +47 -0
  71. package/.claude/skills/migrate-from-openclaw/MIGRATE_CRONS.md +100 -0
  72. package/.claude/skills/migrate-from-openclaw/SKILL.md +447 -0
  73. package/.claude/skills/migrate-from-openclaw/scripts/discover-openclaw.ts +734 -0
  74. package/.claude/skills/migrate-from-openclaw/scripts/extract-channel-credentials.ts +476 -0
  75. package/.claude/skills/migrate-nanoclaw/SKILL.md +484 -0
  76. package/.claude/skills/migrate-nanoclaw/diagnostics.md +51 -0
  77. package/.claude/skills/qodo-pr-resolver/SKILL.md +326 -0
  78. package/.claude/skills/qodo-pr-resolver/resources/providers.md +329 -0
  79. package/.claude/skills/update-nanoclaw/SKILL.md +243 -0
  80. package/.claude/skills/update-nanoclaw/diagnostics.md +48 -0
  81. package/.claude/skills/update-skills/SKILL.md +130 -0
  82. package/.claude/skills/use-native-credential-proxy/SKILL.md +167 -0
  83. package/.claude/skills/x-integration/SKILL.md +417 -0
  84. package/.claude/skills/x-integration/agent.ts +243 -0
  85. package/.claude/skills/x-integration/host.ts +155 -0
  86. package/.claude/skills/x-integration/lib/browser.ts +148 -0
  87. package/.claude/skills/x-integration/lib/config.ts +62 -0
  88. package/.claude/skills/x-integration/scripts/like.ts +56 -0
  89. package/.claude/skills/x-integration/scripts/post.ts +66 -0
  90. package/.claude/skills/x-integration/scripts/quote.ts +80 -0
  91. package/.claude/skills/x-integration/scripts/reply.ts +74 -0
  92. package/.claude/skills/x-integration/scripts/retweet.ts +62 -0
  93. package/.claude/skills/x-integration/scripts/setup.ts +87 -0
  94. package/.github/CODEOWNERS +10 -0
  95. package/.github/PULL_REQUEST_TEMPLATE.md +18 -0
  96. package/.github/workflows/bump-version.yml +35 -0
  97. package/.github/workflows/ci.yml +39 -0
  98. package/.github/workflows/label-pr.yml +40 -0
  99. package/.github/workflows/update-tokens.yml +43 -0
  100. package/.husky/pre-commit +1 -0
  101. package/.mcp.json +3 -0
  102. package/.nvmrc +1 -0
  103. package/.parachute/module.json +14 -0
  104. package/.prettierrc +4 -0
  105. package/CHANGELOG.md +215 -0
  106. package/CLAUDE.md +307 -0
  107. package/CODE_OF_CONDUCT.md +128 -0
  108. package/CONTRIBUTING.md +159 -0
  109. package/CONTRIBUTORS.md +26 -0
  110. package/LICENSE +21 -0
  111. package/README.md +190 -0
  112. package/README_ja.md +194 -0
  113. package/README_zh.md +194 -0
  114. package/assets/nanoclaw-favicon.png +0 -0
  115. package/assets/nanoclaw-icon.png +0 -0
  116. package/assets/nanoclaw-logo-dark.png +0 -0
  117. package/assets/nanoclaw-logo.png +0 -0
  118. package/assets/nanoclaw-profile.jpeg +0 -0
  119. package/assets/nanoclaw-sales.png +0 -0
  120. package/assets/social-preview.jpg +0 -0
  121. package/config-examples/mount-allowlist.json +25 -0
  122. package/container/.dockerignore +2 -0
  123. package/container/CLAUDE.md +21 -0
  124. package/container/Dockerfile +121 -0
  125. package/container/agent-runner/bun.lock +243 -0
  126. package/container/agent-runner/package.json +22 -0
  127. package/container/agent-runner/scripts/sdk-signal-probe.ts +169 -0
  128. package/container/agent-runner/src/config.ts +55 -0
  129. package/container/agent-runner/src/db/connection.ts +267 -0
  130. package/container/agent-runner/src/db/index.ts +20 -0
  131. package/container/agent-runner/src/db/messages-in.ts +138 -0
  132. package/container/agent-runner/src/db/messages-out.ts +143 -0
  133. package/container/agent-runner/src/db/session-routing.ts +30 -0
  134. package/container/agent-runner/src/db/session-state.test.ts +100 -0
  135. package/container/agent-runner/src/db/session-state.ts +79 -0
  136. package/container/agent-runner/src/destinations.ts +135 -0
  137. package/container/agent-runner/src/formatter.test.ts +167 -0
  138. package/container/agent-runner/src/formatter.ts +260 -0
  139. package/container/agent-runner/src/index.ts +110 -0
  140. package/container/agent-runner/src/integration.test.ts +121 -0
  141. package/container/agent-runner/src/mcp-tools/agents.instructions.md +26 -0
  142. package/container/agent-runner/src/mcp-tools/agents.ts +66 -0
  143. package/container/agent-runner/src/mcp-tools/core.instructions.md +27 -0
  144. package/container/agent-runner/src/mcp-tools/core.ts +262 -0
  145. package/container/agent-runner/src/mcp-tools/index.ts +22 -0
  146. package/container/agent-runner/src/mcp-tools/interactive.instructions.md +22 -0
  147. package/container/agent-runner/src/mcp-tools/interactive.ts +169 -0
  148. package/container/agent-runner/src/mcp-tools/scheduling.instructions.md +40 -0
  149. package/container/agent-runner/src/mcp-tools/scheduling.ts +299 -0
  150. package/container/agent-runner/src/mcp-tools/self-mod.instructions.md +25 -0
  151. package/container/agent-runner/src/mcp-tools/self-mod.ts +120 -0
  152. package/container/agent-runner/src/mcp-tools/server.ts +54 -0
  153. package/container/agent-runner/src/mcp-tools/types.ts +6 -0
  154. package/container/agent-runner/src/poll-loop.test.ts +248 -0
  155. package/container/agent-runner/src/poll-loop.ts +437 -0
  156. package/container/agent-runner/src/providers/claude.ts +379 -0
  157. package/container/agent-runner/src/providers/factory.test.ts +19 -0
  158. package/container/agent-runner/src/providers/factory.ts +13 -0
  159. package/container/agent-runner/src/providers/index.ts +6 -0
  160. package/container/agent-runner/src/providers/mock.ts +77 -0
  161. package/container/agent-runner/src/providers/provider-registry.ts +33 -0
  162. package/container/agent-runner/src/providers/types.ts +82 -0
  163. package/container/agent-runner/src/scheduling/task-script.ts +121 -0
  164. package/container/agent-runner/src/timezone.test.ts +93 -0
  165. package/container/agent-runner/src/timezone.ts +107 -0
  166. package/container/agent-runner/tsconfig.json +14 -0
  167. package/container/build.sh +48 -0
  168. package/container/entrypoint.sh +16 -0
  169. package/container/skills/agent-browser/SKILL.md +159 -0
  170. package/container/skills/frontend-engineer/SKILL.md +157 -0
  171. package/container/skills/self-customize/SKILL.md +87 -0
  172. package/container/skills/slack-formatting/SKILL.md +94 -0
  173. package/container/skills/vercel-cli/SKILL.md +111 -0
  174. package/container/skills/welcome/SKILL.md +85 -0
  175. package/docs/APPLE-CONTAINER-NETWORKING.md +90 -0
  176. package/docs/BRANCH-FORK-MAINTENANCE.md +81 -0
  177. package/docs/README.md +25 -0
  178. package/docs/SDK_DEEP_DIVE.md +643 -0
  179. package/docs/SECURITY.md +162 -0
  180. package/docs/agent-runner-details.md +749 -0
  181. package/docs/api-details.md +365 -0
  182. package/docs/architecture-diagram.html +422 -0
  183. package/docs/architecture-diagram.md +215 -0
  184. package/docs/architecture.md +751 -0
  185. package/docs/audit/2026-04-30-channel-endpoint-audit.md +36 -0
  186. package/docs/build-and-runtime.md +80 -0
  187. package/docs/cross-mount-stress/README.md +112 -0
  188. package/docs/cross-mount-stress/container-writer-retry.mjs +55 -0
  189. package/docs/cross-mount-stress/container-writer-slow.mjs +42 -0
  190. package/docs/cross-mount-stress/container-writer.mjs +47 -0
  191. package/docs/cross-mount-stress/host-writer-retry.mjs +55 -0
  192. package/docs/cross-mount-stress/host-writer-slow.mjs +43 -0
  193. package/docs/cross-mount-stress/host-writer.mjs +47 -0
  194. package/docs/db-central.md +316 -0
  195. package/docs/db-session.md +183 -0
  196. package/docs/db.md +119 -0
  197. package/docs/design/2026-04-29-vault-management-ui.md +231 -0
  198. package/docs/design/2026-04-30-channel-wiring-rework.md +234 -0
  199. package/docs/design/2026-05-01-channel-wiring-approvals-deep-dive.md +272 -0
  200. package/docs/design/2026-05-02-channel-policy-and-approval-routing.md +250 -0
  201. package/docs/docker-sandboxes.md +359 -0
  202. package/docs/isolation-model.md +88 -0
  203. package/docs/ollama.md +79 -0
  204. package/docs/parachute-integration.md +109 -0
  205. package/docs/post-night-rebirth-reflections.md +151 -0
  206. package/eslint.config.js +32 -0
  207. package/package.json +54 -0
  208. package/pnpm-workspace.yaml +8 -0
  209. package/repo-tokens/README.md +113 -0
  210. package/repo-tokens/action.yml +186 -0
  211. package/repo-tokens/badge.svg +23 -0
  212. package/repo-tokens/examples/green.svg +14 -0
  213. package/repo-tokens/examples/red.svg +14 -0
  214. package/repo-tokens/examples/yellow-green.svg +14 -0
  215. package/repo-tokens/examples/yellow.svg +14 -0
  216. package/scripts/chat.ts +101 -0
  217. package/scripts/cleanup-sessions.sh +150 -0
  218. package/scripts/init-cli-agent.ts +171 -0
  219. package/scripts/init-first-agent.ts +377 -0
  220. package/scripts/parachute.ts +158 -0
  221. package/scripts/run-migrations.ts +105 -0
  222. package/scripts/sanity-live-poll.ts +95 -0
  223. package/scripts/seed-discord.ts +79 -0
  224. package/scripts/test-v2-agent.ts +106 -0
  225. package/scripts/test-v2-channel-e2e.ts +265 -0
  226. package/scripts/test-v2-host.ts +184 -0
  227. package/src/channels/adapter.ts +214 -0
  228. package/src/channels/ask-question.ts +46 -0
  229. package/src/channels/channel-registry.test.ts +421 -0
  230. package/src/channels/channel-registry.ts +313 -0
  231. package/src/channels/chat-sdk-bridge.test.ts +84 -0
  232. package/src/channels/chat-sdk-bridge.ts +652 -0
  233. package/src/channels/cli.ts +276 -0
  234. package/src/channels/discord.ts +90 -0
  235. package/src/channels/index.ts +17 -0
  236. package/src/channels/telegram-markdown-sanitize.test.ts +78 -0
  237. package/src/channels/telegram-markdown-sanitize.ts +55 -0
  238. package/src/channels/telegram-pairing.test.ts +254 -0
  239. package/src/channels/telegram-pairing.ts +339 -0
  240. package/src/channels/telegram.ts +279 -0
  241. package/src/channels/trust-hint.test.ts +48 -0
  242. package/src/channels/trust-hint.ts +75 -0
  243. package/src/claude-md-compose.migrate.test.ts +64 -0
  244. package/src/claude-md-compose.ts +205 -0
  245. package/src/command-gate.ts +63 -0
  246. package/src/config.test.ts +93 -0
  247. package/src/config.ts +108 -0
  248. package/src/container-config.ts +167 -0
  249. package/src/container-runner.test.ts +32 -0
  250. package/src/container-runner.ts +576 -0
  251. package/src/container-runtime.test.ts +169 -0
  252. package/src/container-runtime.ts +92 -0
  253. package/src/db/_bun-sqlite-shim.ts +88 -0
  254. package/src/db/agent-activity.test.ts +155 -0
  255. package/src/db/agent-activity.ts +121 -0
  256. package/src/db/agent-groups.ts +77 -0
  257. package/src/db/connection.migrate.test.ts +143 -0
  258. package/src/db/connection.ts +224 -0
  259. package/src/db/db-v2.test.ts +440 -0
  260. package/src/db/dropped-messages.ts +44 -0
  261. package/src/db/index.ts +40 -0
  262. package/src/db/messaging-groups.ts +252 -0
  263. package/src/db/migrations/001-initial.ts +112 -0
  264. package/src/db/migrations/002-chat-sdk-state.ts +36 -0
  265. package/src/db/migrations/008-dropped-messages.ts +27 -0
  266. package/src/db/migrations/009-drop-pending-credentials.ts +13 -0
  267. package/src/db/migrations/010-engage-modes.ts +103 -0
  268. package/src/db/migrations/011-pending-sender-approvals.ts +40 -0
  269. package/src/db/migrations/012-channel-registration.ts +48 -0
  270. package/src/db/migrations/013-approval-render-metadata.ts +27 -0
  271. package/src/db/migrations/014-secrets.ts +44 -0
  272. package/src/db/migrations/015-secrets-drop-host-pattern.ts +18 -0
  273. package/src/db/migrations/016-secret-assignments.ts +30 -0
  274. package/src/db/migrations/017-agent-activity.ts +40 -0
  275. package/src/db/migrations/018-oauth-app-configs.ts +34 -0
  276. package/src/db/migrations/019-oauth-app-connections.ts +48 -0
  277. package/src/db/migrations/020-agent-app-connections.ts +28 -0
  278. package/src/db/migrations/021-pending-oauth-states.ts +35 -0
  279. package/src/db/migrations/022-app-connections-provider.ts +25 -0
  280. package/src/db/migrations/023-agent-group-secret-mode.test.ts +124 -0
  281. package/src/db/migrations/023-agent-group-secret-mode.ts +65 -0
  282. package/src/db/migrations/024-collapse-approvals.test.ts +249 -0
  283. package/src/db/migrations/024-collapse-approvals.ts +182 -0
  284. package/src/db/migrations/025-secret-mode-check.test.ts +155 -0
  285. package/src/db/migrations/025-secret-mode-check.ts +49 -0
  286. package/src/db/migrations/026-user-dms-bot-id.test.ts +116 -0
  287. package/src/db/migrations/026-user-dms-bot-id.ts +54 -0
  288. package/src/db/migrations/027-provider-credentials.ts +41 -0
  289. package/src/db/migrations/_test-helpers.ts +41 -0
  290. package/src/db/migrations/index.ts +127 -0
  291. package/src/db/migrations/module-agent-to-agent-destinations.ts +84 -0
  292. package/src/db/migrations/module-approvals-pending-approvals.ts +42 -0
  293. package/src/db/migrations/module-approvals-title-options.ts +40 -0
  294. package/src/db/schema.ts +258 -0
  295. package/src/db/session-db.test.ts +93 -0
  296. package/src/db/session-db.ts +325 -0
  297. package/src/db/sessions.ts +241 -0
  298. package/src/delivery.test.ts +148 -0
  299. package/src/delivery.ts +445 -0
  300. package/src/env.ts +74 -0
  301. package/src/group-folder.test.ts +35 -0
  302. package/src/group-folder.ts +44 -0
  303. package/src/group-init.ts +92 -0
  304. package/src/host-core.test.ts +456 -0
  305. package/src/host-sweep.test.ts +146 -0
  306. package/src/host-sweep.ts +287 -0
  307. package/src/index.ts +227 -0
  308. package/src/install-slug.ts +33 -0
  309. package/src/log.test.ts +81 -0
  310. package/src/log.ts +117 -0
  311. package/src/mcp/http.ts +72 -0
  312. package/src/mcp/server.ts +92 -0
  313. package/src/mcp/stdio.ts +51 -0
  314. package/src/mcp/tools/activity.ts +88 -0
  315. package/src/mcp/tools/agent-groups.ts +183 -0
  316. package/src/mcp/tools/approvals.ts +122 -0
  317. package/src/mcp/tools/channels.ts +199 -0
  318. package/src/mcp/tools/index.ts +27 -0
  319. package/src/mcp/tools/oauth.ts +48 -0
  320. package/src/mcp/tools/secrets.ts +169 -0
  321. package/src/mcp/tools/sessions.ts +135 -0
  322. package/src/mcp/types.ts +51 -0
  323. package/src/modules/agent-to-agent/agent-route.test.ts +46 -0
  324. package/src/modules/agent-to-agent/agent-route.ts +223 -0
  325. package/src/modules/agent-to-agent/create-agent.ts +127 -0
  326. package/src/modules/agent-to-agent/db/agent-destinations.ts +135 -0
  327. package/src/modules/agent-to-agent/index.ts +22 -0
  328. package/src/modules/agent-to-agent/write-destinations.ts +59 -0
  329. package/src/modules/approvals/agent.md +45 -0
  330. package/src/modules/approvals/index.ts +21 -0
  331. package/src/modules/approvals/picks.test.ts +291 -0
  332. package/src/modules/approvals/primitive.ts +279 -0
  333. package/src/modules/approvals/project.md +27 -0
  334. package/src/modules/approvals/response-handler.ts +87 -0
  335. package/src/modules/index.ts +24 -0
  336. package/src/modules/interactive/agent.md +21 -0
  337. package/src/modules/interactive/index.ts +69 -0
  338. package/src/modules/interactive/project.md +12 -0
  339. package/src/modules/mount-security/index.ts +448 -0
  340. package/src/modules/mount-security/migrate.test.ts +91 -0
  341. package/src/modules/permissions/access.ts +28 -0
  342. package/src/modules/permissions/channel-approval.test.ts +389 -0
  343. package/src/modules/permissions/channel-approval.ts +188 -0
  344. package/src/modules/permissions/db/agent-group-members.ts +44 -0
  345. package/src/modules/permissions/db/pending-channel-approvals.test.ts +86 -0
  346. package/src/modules/permissions/db/pending-channel-approvals.ts +66 -0
  347. package/src/modules/permissions/db/pending-sender-approvals.ts +60 -0
  348. package/src/modules/permissions/db/user-dms.ts +58 -0
  349. package/src/modules/permissions/db/user-roles.ts +85 -0
  350. package/src/modules/permissions/db/users.ts +38 -0
  351. package/src/modules/permissions/index.ts +421 -0
  352. package/src/modules/permissions/permissions.test.ts +358 -0
  353. package/src/modules/permissions/sender-approval.test.ts +470 -0
  354. package/src/modules/permissions/sender-approval.ts +165 -0
  355. package/src/modules/permissions/user-dm.ts +200 -0
  356. package/src/modules/provider-credentials/db.ts +121 -0
  357. package/src/modules/provider-credentials/index.ts +12 -0
  358. package/src/modules/provider-credentials/spawn.test.ts +206 -0
  359. package/src/modules/provider-credentials/spawn.ts +114 -0
  360. package/src/modules/scheduling/actions.ts +113 -0
  361. package/src/modules/scheduling/db.test.ts +282 -0
  362. package/src/modules/scheduling/db.ts +148 -0
  363. package/src/modules/scheduling/index.ts +34 -0
  364. package/src/modules/scheduling/recurrence.test.ts +98 -0
  365. package/src/modules/scheduling/recurrence.ts +54 -0
  366. package/src/modules/self-mod/agent.md +30 -0
  367. package/src/modules/self-mod/apply.ts +85 -0
  368. package/src/modules/self-mod/index.ts +30 -0
  369. package/src/modules/self-mod/project.md +39 -0
  370. package/src/modules/self-mod/request.ts +91 -0
  371. package/src/modules/typing/index.ts +165 -0
  372. package/src/oauth/agent-app-connections.ts +103 -0
  373. package/src/oauth/app-configs.test.ts +64 -0
  374. package/src/oauth/app-configs.ts +114 -0
  375. package/src/oauth/app-connections.test.ts +109 -0
  376. package/src/oauth/app-connections.ts +178 -0
  377. package/src/oauth/crypto.ts +56 -0
  378. package/src/oauth/flow.ts +104 -0
  379. package/src/oauth/providers/google.test.ts +38 -0
  380. package/src/oauth/providers/google.ts +46 -0
  381. package/src/oauth/providers/index.ts +48 -0
  382. package/src/oauth/state-store.test.ts +54 -0
  383. package/src/oauth/state-store.ts +93 -0
  384. package/src/parachute/README.md +27 -0
  385. package/src/parachute/create-agent.test.ts +83 -0
  386. package/src/parachute/create-agent.ts +122 -0
  387. package/src/parachute/group-status.test.ts +165 -0
  388. package/src/parachute/group-status.ts +136 -0
  389. package/src/parachute/types.ts +41 -0
  390. package/src/parachute/vault-mcp.test.ts +251 -0
  391. package/src/parachute/vault-mcp.ts +232 -0
  392. package/src/platform-id.test.ts +104 -0
  393. package/src/platform-id.ts +109 -0
  394. package/src/providers/index.ts +6 -0
  395. package/src/providers/provider-container-registry.ts +58 -0
  396. package/src/response-registry.ts +45 -0
  397. package/src/router.ts +530 -0
  398. package/src/secrets/crypto.test.ts +45 -0
  399. package/src/secrets/crypto.ts +55 -0
  400. package/src/secrets/index.ts +355 -0
  401. package/src/secrets/master-key.ts +70 -0
  402. package/src/secrets/secrets.test.ts +354 -0
  403. package/src/session-manager.migrate.test.ts +59 -0
  404. package/src/session-manager.ts +433 -0
  405. package/src/startup-bootstrap.test.ts +226 -0
  406. package/src/startup-bootstrap.ts +207 -0
  407. package/src/state-sqlite.ts +182 -0
  408. package/src/timezone.test.ts +64 -0
  409. package/src/timezone.ts +37 -0
  410. package/src/types.ts +230 -0
  411. package/src/web/auth.test.ts +335 -0
  412. package/src/web/auth.ts +214 -0
  413. package/src/web/discord-validate.test.ts +77 -0
  414. package/src/web/discord-validate.ts +88 -0
  415. package/src/web/hub-discovery.test.ts +98 -0
  416. package/src/web/hub-discovery.ts +69 -0
  417. package/src/web/routes/activity.ts +106 -0
  418. package/src/web/routes/agent-provider.test.ts +282 -0
  419. package/src/web/routes/agent-provider.ts +309 -0
  420. package/src/web/routes/approvals.ts +185 -0
  421. package/src/web/routes/apps.ts +434 -0
  422. package/src/web/routes/channels-mg-detail.test.ts +324 -0
  423. package/src/web/routes/channels-mga-detail.test.ts +425 -0
  424. package/src/web/routes/channels.ts +489 -0
  425. package/src/web/routes/oauth-providers.ts +42 -0
  426. package/src/web/routes/secrets.test.ts +175 -0
  427. package/src/web/routes/secrets.ts +282 -0
  428. package/src/web/routes/sessions.ts +123 -0
  429. package/src/web/routes/settings.test.ts +106 -0
  430. package/src/web/routes/settings.ts +247 -0
  431. package/src/web/routes/setup-status.ts +205 -0
  432. package/src/web/routes/vaults.test.ts +389 -0
  433. package/src/web/routes/vaults.ts +225 -0
  434. package/src/web/server-version.test.ts +16 -0
  435. package/src/web/server.ts +1003 -0
  436. package/src/web/services-manifest.test.ts +120 -0
  437. package/src/web/services-manifest.ts +61 -0
  438. package/src/web/static-serve.test.ts +255 -0
  439. package/src/web/static-serve.ts +104 -0
  440. package/src/web/telegram-validate.test.ts +116 -0
  441. package/src/web/telegram-validate.ts +107 -0
  442. package/src/web/vault-proxy.test.ts +214 -0
  443. package/src/web/vault-proxy.ts +120 -0
  444. package/src/web/wire-channel.ts +181 -0
  445. package/src/webhook-server.ts +134 -0
  446. package/tsconfig.json +21 -0
  447. package/vitest.config.ts +18 -0
  448. package/web/README.md +63 -0
  449. package/web/ui/index.html +13 -0
  450. package/web/ui/package.json +35 -0
  451. package/web/ui/pnpm-lock.yaml +2164 -0
  452. package/web/ui/scripts/verify-base.mjs +31 -0
  453. package/web/ui/src/App.tsx +88 -0
  454. package/web/ui/src/components/ActivityFeed.tsx +444 -0
  455. package/web/ui/src/components/AgentGroupPicker.tsx +263 -0
  456. package/web/ui/src/components/AgentProviderCards.tsx +220 -0
  457. package/web/ui/src/components/CredentialForm.tsx +214 -0
  458. package/web/ui/src/components/ScopeGrants.tsx +74 -0
  459. package/web/ui/src/components/StatusDot.tsx +43 -0
  460. package/web/ui/src/components/VaultPicker.tsx +127 -0
  461. package/web/ui/src/components/setup/AdapterInstallStep.tsx +178 -0
  462. package/web/ui/src/components/setup/AgentGroupStep.tsx +43 -0
  463. package/web/ui/src/components/setup/ChannelPickStep.tsx +74 -0
  464. package/web/ui/src/components/setup/DoneStep.tsx +49 -0
  465. package/web/ui/src/components/setup/PrereqStep.tsx +129 -0
  466. package/web/ui/src/components/setup/TestConnectionStep.tsx +108 -0
  467. package/web/ui/src/components/setup/TestMessageStep.tsx +104 -0
  468. package/web/ui/src/components/setup/WireChannelStep.tsx +166 -0
  469. package/web/ui/src/components/setup/types.ts +105 -0
  470. package/web/ui/src/lib/api.test.ts +410 -0
  471. package/web/ui/src/lib/api.ts +1210 -0
  472. package/web/ui/src/lib/auth.test.ts +139 -0
  473. package/web/ui/src/lib/auth.ts +348 -0
  474. package/web/ui/src/lib/channel-adapters.ts +136 -0
  475. package/web/ui/src/main.tsx +19 -0
  476. package/web/ui/src/routes/ApprovalsList.tsx +294 -0
  477. package/web/ui/src/routes/Apps.tsx +613 -0
  478. package/web/ui/src/routes/ChannelWireDetail.test.tsx +233 -0
  479. package/web/ui/src/routes/ChannelWireDetail.tsx +403 -0
  480. package/web/ui/src/routes/ChannelsList.tsx +158 -0
  481. package/web/ui/src/routes/GroupDetail.tsx +755 -0
  482. package/web/ui/src/routes/GroupList.tsx +187 -0
  483. package/web/ui/src/routes/MessagingGroupDetail.test.tsx +233 -0
  484. package/web/ui/src/routes/MessagingGroupDetail.tsx +306 -0
  485. package/web/ui/src/routes/NewGroupWizard.tsx +390 -0
  486. package/web/ui/src/routes/OAuthCallback.tsx +56 -0
  487. package/web/ui/src/routes/SecretsList.tsx +921 -0
  488. package/web/ui/src/routes/SessionsList.tsx +220 -0
  489. package/web/ui/src/routes/SettingsAgentProvider.tsx +109 -0
  490. package/web/ui/src/routes/SettingsApprovals.tsx +234 -0
  491. package/web/ui/src/routes/SetupWizard.tsx +219 -0
  492. package/web/ui/src/routes/VaultDetail.test.tsx +361 -0
  493. package/web/ui/src/routes/VaultDetail.tsx +960 -0
  494. package/web/ui/src/routes/VaultsList.tsx +295 -0
  495. package/web/ui/src/routes/WireChannelPage.tsx +413 -0
  496. package/web/ui/src/styles.css +608 -0
  497. package/web/ui/src/test/setup.ts +23 -0
  498. package/web/ui/src/vite-env.d.ts +10 -0
  499. package/web/ui/tsconfig.json +20 -0
  500. package/web/ui/vite.config.ts +34 -0
  501. package/web/ui/vitest.config.ts +25 -0
@@ -0,0 +1,220 @@
1
+ /**
2
+ * /sessions — global session listing across every agent group.
3
+ *
4
+ * The per-group view is on /groups/:folder; this is the flat
5
+ * cross-group inventory the operator uses to spot stuck containers,
6
+ * compare activity, and reap idle sessions.
7
+ *
8
+ * "Close" is a hard-stop: the server flips status to closed and signals
9
+ * the container to exit. Re-opening a closed session is a re-spawn (new
10
+ * container, new id) — the button copy reflects that.
11
+ *
12
+ * The list polls every 7s like /groups so heartbeat freshness stays
13
+ * accurate without a manual refresh. Unlike /groups, sessions don't
14
+ * dedupe — each row is its own thing — so we sort by lastActiveAt desc
15
+ * to keep the most recent at the top.
16
+ */
17
+ import { useCallback, useEffect, useState } from 'react';
18
+ import { Link } from 'react-router-dom';
19
+ import { formatRelative } from '../components/StatusDot.tsx';
20
+ import { closeSession, listSessions, type SessionView } from '../lib/api.ts';
21
+
22
+ const POLL_MS = 7_000;
23
+
24
+ export function SessionsList() {
25
+ const [state, setState] = useState<
26
+ | { kind: 'loading' }
27
+ | { kind: 'ok'; sessions: SessionView[] }
28
+ | { kind: 'error'; message: string }
29
+ >({ kind: 'loading' });
30
+ const [busyId, setBusyId] = useState<string | null>(null);
31
+ const [actionError, setActionError] = useState<string | null>(null);
32
+ const [reloadKey, setReloadKey] = useState(0);
33
+
34
+ const reload = useCallback(() => setReloadKey((k) => k + 1), []);
35
+
36
+ useEffect(() => {
37
+ let cancelled = false;
38
+ listSessions()
39
+ .then((sessions) => !cancelled && setState({ kind: 'ok', sessions }))
40
+ .catch((err) => {
41
+ if (!cancelled) {
42
+ setState({
43
+ kind: 'error',
44
+ message: err instanceof Error ? err.message : String(err),
45
+ });
46
+ }
47
+ });
48
+ return () => {
49
+ cancelled = true;
50
+ };
51
+ }, [reloadKey]);
52
+
53
+ useEffect(() => {
54
+ if (state.kind !== 'ok') return;
55
+ let cancelled = false;
56
+ const t = setInterval(() => {
57
+ listSessions()
58
+ .then((sessions) => !cancelled && setState({ kind: 'ok', sessions }))
59
+ .catch(() => {});
60
+ }, POLL_MS);
61
+ return () => {
62
+ cancelled = true;
63
+ clearInterval(t);
64
+ };
65
+ }, [state.kind]);
66
+
67
+ const onClose = async (s: SessionView) => {
68
+ if (!confirm(`Close session ${s.id.slice(0, 8)}… for ${s.agentGroupName}? The container will be stopped.`)) {
69
+ return;
70
+ }
71
+ setBusyId(s.id);
72
+ setActionError(null);
73
+ try {
74
+ await closeSession(s.id);
75
+ reload();
76
+ } catch (err) {
77
+ setActionError(err instanceof Error ? err.message : String(err));
78
+ } finally {
79
+ setBusyId(null);
80
+ }
81
+ };
82
+
83
+ if (state.kind === 'loading') {
84
+ return (
85
+ <div>
86
+ <h2>Sessions</h2>
87
+ <ul className="skeleton-list" aria-busy="true">
88
+ <li className="skeleton skeleton-row" />
89
+ <li className="skeleton skeleton-row" />
90
+ </ul>
91
+ </div>
92
+ );
93
+ }
94
+
95
+ if (state.kind === 'error') {
96
+ return (
97
+ <div>
98
+ <h2>Sessions</h2>
99
+ <div className="error-banner">
100
+ Couldn't load sessions: <code>{state.message}</code>
101
+ </div>
102
+ <div className="actions" style={{ marginTop: '1rem' }}>
103
+ <button onClick={reload}>Retry</button>
104
+ </div>
105
+ </div>
106
+ );
107
+ }
108
+
109
+ const active = state.sessions.filter((s) => s.status === 'active');
110
+ const closed = state.sessions.filter((s) => s.status !== 'active');
111
+ const sortByActive = (a: SessionView, b: SessionView) => {
112
+ const aT = a.lastActiveAt ? Date.parse(a.lastActiveAt) : 0;
113
+ const bT = b.lastActiveAt ? Date.parse(b.lastActiveAt) : 0;
114
+ return bT - aT;
115
+ };
116
+ active.sort(sortByActive);
117
+ closed.sort(sortByActive);
118
+
119
+ return (
120
+ <div>
121
+ <div className="list-header">
122
+ <h2>Sessions ({active.length} active)</h2>
123
+ <button className="secondary" onClick={reload}>Refresh</button>
124
+ </div>
125
+
126
+ <p className="muted">
127
+ Per-session containers across every agent group. Stuck containers can be hard-closed here; re-opening is a
128
+ new spawn from the agent group page.
129
+ </p>
130
+
131
+ {actionError && <div className="error-banner">{actionError}</div>}
132
+
133
+ {state.sessions.length === 0 && (
134
+ <div className="empty empty-rich" style={{ marginTop: '1rem' }}>
135
+ <p className="empty-headline">No sessions yet.</p>
136
+ <p className="muted">
137
+ Sessions are spawned when a wired channel receives an inbound message, or via the spawn button on an{' '}
138
+ <Link to="/">agent group</Link>.
139
+ </p>
140
+ </div>
141
+ )}
142
+
143
+ {active.length > 0 && (
144
+ <div style={{ marginTop: '1rem' }}>
145
+ {active.map((s) => (
146
+ <SessionRow key={s.id} s={s} busy={busyId === s.id} onClose={onClose} />
147
+ ))}
148
+ </div>
149
+ )}
150
+
151
+ {closed.length > 0 && (
152
+ <div style={{ marginTop: '2rem' }}>
153
+ <h3 style={{ fontSize: '0.85rem', color: 'var(--fg-muted)', textTransform: 'uppercase', letterSpacing: '0.06em', fontWeight: 500 }}>
154
+ Closed (recent)
155
+ </h3>
156
+ {closed.slice(0, 20).map((s) => (
157
+ <SessionRow key={s.id} s={s} busy={false} onClose={() => {}} closed />
158
+ ))}
159
+ </div>
160
+ )}
161
+ </div>
162
+ );
163
+ }
164
+
165
+ function SessionRow({
166
+ s,
167
+ busy,
168
+ onClose,
169
+ closed,
170
+ }: {
171
+ s: SessionView;
172
+ busy: boolean;
173
+ onClose: (s: SessionView) => void;
174
+ closed?: boolean;
175
+ }) {
176
+ const containerColor =
177
+ s.containerStatus === 'running' ? 'var(--accent)' : s.containerStatus === 'idle' ? 'var(--warn)' : 'var(--fg-dim)';
178
+ return (
179
+ <div
180
+ style={{
181
+ display: 'flex',
182
+ alignItems: 'center',
183
+ gap: '1rem',
184
+ padding: '0.75rem 1rem',
185
+ background: 'white',
186
+ border: '1px solid var(--border)',
187
+ borderRadius: '8px',
188
+ marginBottom: '0.5rem',
189
+ opacity: closed ? 0.7 : 1,
190
+ }}
191
+ >
192
+ <div style={{ flex: 1, minWidth: 0 }}>
193
+ <div style={{ display: 'flex', alignItems: 'center', gap: '0.5rem', flexWrap: 'wrap' }}>
194
+ <Link to={`/groups/${encodeURIComponent(s.agentGroupFolder)}`} style={{ fontWeight: 500 }}>
195
+ {s.agentGroupName}
196
+ </Link>
197
+ <span style={{ color: containerColor, fontSize: '0.85rem' }}>● {s.containerStatus}</span>
198
+ {!s.alive && s.status === 'active' && <span className="tag warn">no heartbeat</span>}
199
+ </div>
200
+ <div className="dim" style={{ marginTop: '0.25rem', fontSize: '0.85rem' }}>
201
+ <code>{s.id.slice(0, 12)}…</code>
202
+ {s.messagingGroupId && <> · channel <code>{s.messagingGroupId.slice(0, 8)}…</code></>}
203
+ {s.lastActiveAt && <> · last active {formatRelative(s.lastActiveAt)}</>}
204
+ {!s.lastActiveAt && <> · created {formatRelative(s.createdAt)}</>}
205
+ </div>
206
+ </div>
207
+ {!closed && (
208
+ <button
209
+ type="button"
210
+ className="secondary"
211
+ onClick={() => onClose(s)}
212
+ disabled={busy}
213
+ style={{ borderColor: 'var(--error)', color: 'var(--error)' }}
214
+ >
215
+ {busy ? 'Closing…' : 'Close'}
216
+ </button>
217
+ )}
218
+ </div>
219
+ );
220
+ }
@@ -0,0 +1,109 @@
1
+ /**
2
+ * /settings/agent-provider — install-wide agent provider source.
3
+ *
4
+ * Three options, all paste-only:
5
+ * - Claude setup token (`claude setup-token` on a subscription host).
6
+ * - Anthropic API key (Console).
7
+ * - External provider server (self-hosted proxy or vendor speaking the
8
+ * Anthropic API).
9
+ *
10
+ * The page never displays plaintext secrets — the API returns a `hasApiKey`
11
+ * boolean only. Per-agent-group overrides live on each group's detail page
12
+ * (paraclaw#86).
13
+ */
14
+ import { useCallback, useEffect, useState } from 'react';
15
+
16
+ import { AgentProviderCards } from '../components/AgentProviderCards.tsx';
17
+ import { getAgentProvider, setAgentProvider, type AgentProviderSource, type AgentProviderView } from '../lib/api.ts';
18
+
19
+ type SaveState = { kind: 'idle' } | { kind: 'saving' } | { kind: 'error'; message: string };
20
+
21
+ export function SettingsAgentProvider() {
22
+ const [state, setState] = useState<
23
+ { kind: 'loading' } | { kind: 'ok'; view: AgentProviderView } | { kind: 'error'; message: string }
24
+ >({ kind: 'loading' });
25
+ const [save, setSave] = useState<SaveState>({ kind: 'idle' });
26
+ const [reloadKey, setReloadKey] = useState(0);
27
+
28
+ const reload = useCallback(() => setReloadKey((k) => k + 1), []);
29
+
30
+ useEffect(() => {
31
+ let cancelled = false;
32
+ getAgentProvider()
33
+ .then((view) => !cancelled && setState({ kind: 'ok', view }))
34
+ .catch((err) => {
35
+ if (!cancelled) {
36
+ setState({ kind: 'error', message: err instanceof Error ? err.message : String(err) });
37
+ }
38
+ });
39
+ return () => {
40
+ cancelled = true;
41
+ };
42
+ }, [reloadKey]);
43
+
44
+ const submit = async (input: { source: AgentProviderSource; apiKey?: string; serverUrl?: string }) => {
45
+ setSave({ kind: 'saving' });
46
+ try {
47
+ const view = await setAgentProvider(input);
48
+ setState({ kind: 'ok', view });
49
+ setSave({ kind: 'idle' });
50
+ } catch (err) {
51
+ setSave({ kind: 'error', message: err instanceof Error ? err.message : String(err) });
52
+ }
53
+ };
54
+
55
+ if (state.kind === 'loading') {
56
+ return (
57
+ <div>
58
+ <h2>Settings · Agent provider</h2>
59
+ <ul className="skeleton-list" aria-busy="true">
60
+ <li className="skeleton skeleton-row" />
61
+ <li className="skeleton skeleton-row" />
62
+ <li className="skeleton skeleton-row" />
63
+ </ul>
64
+ </div>
65
+ );
66
+ }
67
+ if (state.kind === 'error') {
68
+ return (
69
+ <div>
70
+ <h2>Settings · Agent provider</h2>
71
+ <div className="error-banner">
72
+ Couldn't load settings: <code>{state.message}</code>
73
+ </div>
74
+ <div className="actions" style={{ marginTop: '1rem' }}>
75
+ <button onClick={reload}>Retry</button>
76
+ </div>
77
+ </div>
78
+ );
79
+ }
80
+
81
+ const { view } = state;
82
+ return (
83
+ <div>
84
+ <div className="list-header">
85
+ <h2>Settings · Agent provider</h2>
86
+ <button className="secondary" onClick={reload}>
87
+ Refresh
88
+ </button>
89
+ </div>
90
+ <nav className="muted" style={{ marginBottom: '0.75rem' }}>
91
+ <a href="approvals">Approval routing</a>
92
+ {' · '}
93
+ <a href="agent-provider">Agent provider</a>
94
+ </nav>
95
+ <p className="muted">
96
+ Where the agent gets its Claude credentials. One source per install — applies to every agent group unless a
97
+ specific group sets an override on its detail page. Changing the source takes effect on the next session spawn.
98
+ </p>
99
+
100
+ {save.kind === 'error' && (
101
+ <div className="error-banner" style={{ marginBottom: '1rem' }}>
102
+ {save.message}
103
+ </div>
104
+ )}
105
+
106
+ <AgentProviderCards view={view} busy={save.kind === 'saving'} onSubmit={submit} />
107
+ </div>
108
+ );
109
+ }
@@ -0,0 +1,234 @@
1
+ /**
2
+ * /settings/approvals — default approval-routing bot per (approver, channel).
3
+ *
4
+ * Backs the `bot_id = ''` channel-default slot in `user_dms`. When an
5
+ * approval card needs to land for a given approver and the inbound
6
+ * came in on a bot we don't have a cached DM for, the picker falls
7
+ * through to this configured default. Without a default the picker
8
+ * cold-resolves through whichever adapter happens to be registered
9
+ * first — fine on single-bot installs, surprising on multi-bot.
10
+ *
11
+ * One row per (approver, channel) pair where either the approver has a
12
+ * cached DM or there's at least one active adapter for the channel.
13
+ * Operator changes the default by picking from the channel's active
14
+ * bots; the server cold-resolves through the chosen bot to confirm it
15
+ * can DM the user before re-pointing the slot.
16
+ */
17
+ import { useCallback, useEffect, useMemo, useState } from 'react';
18
+ import {
19
+ listApprovalRouting,
20
+ setApprovalRoutingDefault,
21
+ type ApprovalRoutingRow,
22
+ } from '../lib/api.ts';
23
+
24
+ interface SaveError {
25
+ rowKey: string;
26
+ message: string;
27
+ }
28
+
29
+ function rowKey(r: ApprovalRoutingRow): string {
30
+ return `${r.userId}|${r.channelType}`;
31
+ }
32
+
33
+ export function SettingsApprovals() {
34
+ const [state, setState] = useState<
35
+ | { kind: 'loading' }
36
+ | { kind: 'ok'; rows: ApprovalRoutingRow[] }
37
+ | { kind: 'error'; message: string }
38
+ >({ kind: 'loading' });
39
+ const [busyKey, setBusyKey] = useState<string | null>(null);
40
+ const [saveError, setSaveError] = useState<SaveError | null>(null);
41
+ const [reloadKey, setReloadKey] = useState(0);
42
+
43
+ const reload = useCallback(() => setReloadKey((k) => k + 1), []);
44
+
45
+ useEffect(() => {
46
+ let cancelled = false;
47
+ listApprovalRouting()
48
+ .then((rows) => !cancelled && setState({ kind: 'ok', rows }))
49
+ .catch((err) => {
50
+ if (!cancelled) {
51
+ setState({
52
+ kind: 'error',
53
+ message: err instanceof Error ? err.message : String(err),
54
+ });
55
+ }
56
+ });
57
+ return () => {
58
+ cancelled = true;
59
+ };
60
+ }, [reloadKey]);
61
+
62
+ const onPickBot = async (row: ApprovalRoutingRow, botId: string) => {
63
+ if (botId === (row.currentBotId ?? '')) return;
64
+ const key = rowKey(row);
65
+ setBusyKey(key);
66
+ setSaveError(null);
67
+ try {
68
+ const updated = await setApprovalRoutingDefault(row.userId, row.channelType, botId);
69
+ setState((s) => {
70
+ if (s.kind !== 'ok') return s;
71
+ return {
72
+ kind: 'ok',
73
+ rows: s.rows.map((r) => (rowKey(r) === key ? updated : r)),
74
+ };
75
+ });
76
+ } catch (err) {
77
+ setSaveError({ rowKey: key, message: err instanceof Error ? err.message : String(err) });
78
+ } finally {
79
+ setBusyKey(null);
80
+ }
81
+ };
82
+
83
+ if (state.kind === 'loading') {
84
+ return (
85
+ <div>
86
+ <h2>Settings · Approval routing</h2>
87
+ <ul className="skeleton-list" aria-busy="true">
88
+ <li className="skeleton skeleton-row" />
89
+ <li className="skeleton skeleton-row" />
90
+ </ul>
91
+ </div>
92
+ );
93
+ }
94
+
95
+ if (state.kind === 'error') {
96
+ return (
97
+ <div>
98
+ <h2>Settings · Approval routing</h2>
99
+ <div className="error-banner">
100
+ Couldn't load settings: <code>{state.message}</code>
101
+ </div>
102
+ <div className="actions" style={{ marginTop: '1rem' }}>
103
+ <button onClick={reload}>Retry</button>
104
+ </div>
105
+ </div>
106
+ );
107
+ }
108
+
109
+ return (
110
+ <div>
111
+ <div className="list-header">
112
+ <h2>Settings · Approval routing</h2>
113
+ <button className="secondary" onClick={reload}>Refresh</button>
114
+ </div>
115
+
116
+ <nav className="muted" style={{ marginBottom: '0.75rem' }}>
117
+ <a href="approvals">Approval routing</a>
118
+ {' · '}
119
+ <a href="agent-provider">Agent provider</a>
120
+ </nav>
121
+
122
+ <p className="muted">
123
+ Pick the bot that should deliver approval cards when the inbound bot can't reach you. Each
124
+ approver has their own per-channel default. Adding a new bot? Wire it from{' '}
125
+ <a href="channels/new">Channels → New</a> first — only running adapters appear here.
126
+ </p>
127
+
128
+ {state.rows.length === 0 && (
129
+ <div className="empty empty-rich" style={{ marginTop: '1rem' }}>
130
+ <p className="empty-headline">Nothing to configure yet.</p>
131
+ <p className="muted">
132
+ Add an owner or admin (and wire at least one channel) and they'll show up here.
133
+ </p>
134
+ </div>
135
+ )}
136
+
137
+ {state.rows.map((row) => (
138
+ <SettingsRow
139
+ key={rowKey(row)}
140
+ row={row}
141
+ busy={busyKey === rowKey(row)}
142
+ error={saveError && saveError.rowKey === rowKey(row) ? saveError.message : null}
143
+ onPick={(botId) => onPickBot(row, botId)}
144
+ />
145
+ ))}
146
+ </div>
147
+ );
148
+ }
149
+
150
+ function SettingsRow({
151
+ row,
152
+ busy,
153
+ error,
154
+ onPick,
155
+ }: {
156
+ row: ApprovalRoutingRow;
157
+ busy: boolean;
158
+ error: string | null;
159
+ onPick: (botId: string) => void;
160
+ }) {
161
+ const value = row.currentBotId ?? '';
162
+ // Multi-bot install OR a default already pinned to a specific bot.
163
+ // If there's only one available bot AND the default is either unset
164
+ // or already that bot, we render an info row — there's no choice
165
+ // to be made and a dropdown of one option is just visual noise.
166
+ const trivial = useMemo(() => {
167
+ if (row.availableBots.length <= 1) {
168
+ const only = row.availableBots[0]?.botId ?? '';
169
+ if (!value || value === only) return true;
170
+ }
171
+ return false;
172
+ }, [row.availableBots, value]);
173
+
174
+ return (
175
+ <div
176
+ style={{
177
+ background: 'white',
178
+ border: '1px solid var(--border)',
179
+ borderRadius: '8px',
180
+ padding: '1rem 1.25rem',
181
+ marginBottom: '0.75rem',
182
+ }}
183
+ >
184
+ <div style={{ display: 'flex', alignItems: 'center', gap: '0.5rem', flexWrap: 'wrap' }}>
185
+ <strong>{row.channelType}</strong>
186
+ <span className="tag muted">{row.userId}</span>
187
+ </div>
188
+
189
+ <div style={{ marginTop: '0.5rem' }}>
190
+ {trivial ? (
191
+ <p className="muted" style={{ margin: 0 }}>
192
+ {row.availableBots.length === 0 ? (
193
+ <>No active bot for this channel — wire one to enable routing.</>
194
+ ) : (
195
+ <>
196
+ Routing through <code>{row.availableBots[0]!.label}</code> (the only active bot).
197
+ </>
198
+ )}
199
+ </p>
200
+ ) : (
201
+ <label style={{ display: 'flex', alignItems: 'center', gap: '0.5rem' }}>
202
+ <span className="muted">Default bot:</span>
203
+ <select
204
+ value={value}
205
+ disabled={busy}
206
+ onChange={(e) => onPick(e.target.value)}
207
+ style={{ minWidth: '14rem' }}
208
+ >
209
+ {value === '' && <option value="">(unset — first adapter wins)</option>}
210
+ {row.availableBots.map((b) => (
211
+ <option key={b.botId} value={b.botId}>
212
+ {b.label}
213
+ </option>
214
+ ))}
215
+ {/* Surface a stale current selection (e.g. its adapter was
216
+ taken offline) so the operator sees what's there before
217
+ switching it. */}
218
+ {value !== '' && !row.availableBots.find((b) => b.botId === value) && (
219
+ <option value={value}>{value} (offline)</option>
220
+ )}
221
+ </select>
222
+ {busy && <span className="dim">saving…</span>}
223
+ </label>
224
+ )}
225
+ </div>
226
+
227
+ {error && (
228
+ <div className="error-banner" style={{ marginTop: '0.5rem' }}>
229
+ {error}
230
+ </div>
231
+ )}
232
+ </div>
233
+ );
234
+ }