@desplega.ai/agent-swarm 1.20.0 → 1.51.2

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 (561) hide show
  1. package/README.md +271 -169
  2. package/openapi.json +5015 -0
  3. package/package.json +40 -7
  4. package/plugin/commands/close-issue.md +7 -3
  5. package/plugin/commands/create-pr.md +18 -12
  6. package/plugin/commands/implement-issue.md +7 -3
  7. package/plugin/commands/respond-github.md +8 -4
  8. package/plugin/commands/review-pr.md +44 -10
  9. package/plugin/commands/start-leader.md +1 -3
  10. package/plugin/commands/start-worker.md +1 -3
  11. package/plugin/commands/work-on-task.md +22 -3
  12. package/plugin/pi-skills/close-issue/SKILL.md +90 -0
  13. package/plugin/pi-skills/create-pr/SKILL.md +99 -0
  14. package/plugin/pi-skills/implement-issue/SKILL.md +135 -0
  15. package/plugin/pi-skills/investigate-sentry-issue/SKILL.md +138 -0
  16. package/plugin/pi-skills/respond-github/SKILL.md +98 -0
  17. package/plugin/pi-skills/review-offered-task/SKILL.md +45 -0
  18. package/plugin/pi-skills/review-pr/SKILL.md +261 -0
  19. package/plugin/pi-skills/start-leader/SKILL.md +121 -0
  20. package/plugin/pi-skills/start-worker/SKILL.md +60 -0
  21. package/plugin/pi-skills/swarm-chat/SKILL.md +82 -0
  22. package/plugin/pi-skills/todos/SKILL.md +66 -0
  23. package/plugin/pi-skills/work-on-task/SKILL.md +65 -0
  24. package/plugin/skills/artifacts/examples/approval-flow.ts +34 -0
  25. package/plugin/skills/artifacts/examples/hono-dashboard.ts +31 -0
  26. package/plugin/skills/artifacts/examples/multi-artifact.ts +20 -0
  27. package/plugin/skills/artifacts/examples/static-report.sh +17 -0
  28. package/plugin/skills/artifacts/skill.md +71 -0
  29. package/src/agentmail/app.ts +65 -0
  30. package/src/agentmail/handlers.ts +262 -0
  31. package/src/agentmail/index.ts +9 -0
  32. package/src/agentmail/templates.ts +111 -0
  33. package/src/agentmail/types.ts +51 -0
  34. package/src/artifact-sdk/browser-sdk.ts +30 -0
  35. package/src/artifact-sdk/index.ts +2 -0
  36. package/src/artifact-sdk/localtunnel.d.ts +20 -0
  37. package/src/artifact-sdk/port.ts +12 -0
  38. package/src/artifact-sdk/server.ts +156 -0
  39. package/src/artifact-sdk/tunnel.ts +19 -0
  40. package/src/be/chunking.ts +193 -0
  41. package/src/be/db-queries/oauth.ts +90 -0
  42. package/src/be/db-queries/tracker.ts +182 -0
  43. package/src/be/db.ts +3327 -784
  44. package/src/be/embedding.ts +80 -0
  45. package/src/be/migrations/001_initial.sql +409 -0
  46. package/src/be/migrations/002_one_time_schedules.sql +59 -0
  47. package/src/be/migrations/003_workflows.sql +51 -0
  48. package/src/be/migrations/004_workflow_source.sql +81 -0
  49. package/src/be/migrations/005_epic_next_steps.sql +2 -0
  50. package/src/be/migrations/006_vcs_provider.sql +94 -0
  51. package/src/be/migrations/007_task_dir.sql +2 -0
  52. package/src/be/migrations/008_workflow_redesign.sql +85 -0
  53. package/src/be/migrations/009_tracker_integration.sql +144 -0
  54. package/src/be/migrations/010_step_diagnostics.sql +1 -0
  55. package/src/be/migrations/011_step_next_port.sql +1 -0
  56. package/src/be/migrations/012_trigger_schema.sql +1 -0
  57. package/src/be/migrations/013_task_output_schema.sql +2 -0
  58. package/src/be/migrations/014_prompt_templates.sql +33 -0
  59. package/src/be/migrations/015_workflow_workspace.sql +3 -0
  60. package/src/be/migrations/016_active_session_runner_session.sql +4 -0
  61. package/src/be/migrations/017_channel_activity_cursors.sql +6 -0
  62. package/src/be/migrations/018_fix_seed_double_version.sql +30 -0
  63. package/src/be/migrations/runner.ts +188 -0
  64. package/src/be/seed.ts +62 -0
  65. package/src/cli.tsx +231 -299
  66. package/src/commands/artifact.ts +241 -0
  67. package/src/commands/onboard/compose-generator.ts +169 -0
  68. package/src/commands/onboard/env-generator.ts +79 -0
  69. package/src/commands/onboard/manifest.ts +37 -0
  70. package/src/commands/onboard/presets.ts +85 -0
  71. package/src/commands/onboard/service-names.ts +47 -0
  72. package/src/commands/onboard/steps/core-credentials.tsx +111 -0
  73. package/src/commands/onboard/steps/custom-templates.tsx +168 -0
  74. package/src/commands/onboard/steps/generate.tsx +154 -0
  75. package/src/commands/onboard/steps/harness-credentials.tsx +195 -0
  76. package/src/commands/onboard/steps/harness.tsx +21 -0
  77. package/src/commands/onboard/steps/health-check.tsx +171 -0
  78. package/src/commands/onboard/steps/integration-github.tsx +105 -0
  79. package/src/commands/onboard/steps/integration-gitlab.tsx +79 -0
  80. package/src/commands/onboard/steps/integration-menu.tsx +58 -0
  81. package/src/commands/onboard/steps/integration-sentry.tsx +79 -0
  82. package/src/commands/onboard/steps/integration-slack.tsx +165 -0
  83. package/src/commands/onboard/steps/post-connect.tsx +145 -0
  84. package/src/commands/onboard/steps/post-dashboard.tsx +34 -0
  85. package/src/commands/onboard/steps/post-task.tsx +103 -0
  86. package/src/commands/onboard/steps/prereq-check.tsx +178 -0
  87. package/src/commands/onboard/steps/review.tsx +82 -0
  88. package/src/commands/onboard/steps/start.tsx +97 -0
  89. package/src/commands/onboard/templates.ts +34 -0
  90. package/src/commands/onboard/types.ts +259 -0
  91. package/src/commands/onboard.tsx +425 -0
  92. package/src/commands/runner.ts +1540 -630
  93. package/src/commands/setup.tsx +23 -38
  94. package/src/commands/shared/client-config.ts +41 -0
  95. package/src/commands/templates.ts +172 -0
  96. package/src/github/app.ts +8 -0
  97. package/src/github/handlers.ts +384 -151
  98. package/src/github/index.ts +1 -0
  99. package/src/github/mentions-aliases.test.ts +73 -0
  100. package/src/github/mentions.test.ts +3 -3
  101. package/src/github/mentions.ts +32 -6
  102. package/src/github/templates.ts +398 -0
  103. package/src/github/types.ts +1 -0
  104. package/src/gitlab/auth.ts +63 -0
  105. package/src/gitlab/handlers.ts +368 -0
  106. package/src/gitlab/index.ts +19 -0
  107. package/src/gitlab/reactions.ts +104 -0
  108. package/src/gitlab/templates.ts +140 -0
  109. package/src/gitlab/types.ts +130 -0
  110. package/src/heartbeat/heartbeat.ts +434 -0
  111. package/src/heartbeat/index.ts +1 -0
  112. package/src/heartbeat/templates.ts +30 -0
  113. package/src/hooks/hook.ts +555 -4
  114. package/src/hooks/tool-loop-detection.test.ts +158 -0
  115. package/src/hooks/tool-loop-detection.ts +167 -0
  116. package/src/http/active-sessions.ts +199 -0
  117. package/src/http/agents.ts +328 -0
  118. package/src/http/config.ts +191 -0
  119. package/src/http/core.ts +309 -0
  120. package/src/http/db-query.ts +91 -0
  121. package/src/http/ecosystem.ts +63 -0
  122. package/src/http/epics.ts +460 -0
  123. package/src/http/index.ts +216 -0
  124. package/src/http/mcp.ts +77 -0
  125. package/src/http/memory.ts +168 -0
  126. package/src/http/openapi.ts +109 -0
  127. package/src/http/poll.ts +299 -0
  128. package/src/http/prompt-templates.ts +412 -0
  129. package/src/http/repos.ts +195 -0
  130. package/src/http/route-def.ts +123 -0
  131. package/src/http/schedules.ts +426 -0
  132. package/src/http/session-data.ts +241 -0
  133. package/src/http/stats.ts +174 -0
  134. package/src/http/tasks.ts +468 -0
  135. package/src/http/trackers/index.ts +10 -0
  136. package/src/http/trackers/linear.ts +187 -0
  137. package/src/http/types.ts +12 -0
  138. package/src/http/utils.ts +87 -0
  139. package/src/http/webhooks.ts +432 -0
  140. package/src/http/workflows.ts +530 -0
  141. package/src/http.ts +1 -1890
  142. package/src/linear/README.md +65 -0
  143. package/src/linear/app.ts +48 -0
  144. package/src/linear/client.ts +18 -0
  145. package/src/linear/index.ts +1 -0
  146. package/src/linear/oauth.ts +35 -0
  147. package/src/linear/outbound.ts +212 -0
  148. package/src/linear/sync.ts +567 -0
  149. package/src/linear/templates.ts +47 -0
  150. package/src/linear/types.ts +7 -0
  151. package/src/linear/webhook.ts +104 -0
  152. package/src/oauth/README.md +66 -0
  153. package/src/oauth/index.ts +6 -0
  154. package/src/oauth/wrapper.ts +204 -0
  155. package/src/prompts/base-prompt.ts +150 -265
  156. package/src/prompts/defaults.ts +196 -0
  157. package/src/prompts/registry.ts +57 -0
  158. package/src/prompts/resolver.ts +296 -0
  159. package/src/prompts/session-templates.ts +604 -0
  160. package/src/providers/claude-adapter.ts +442 -0
  161. package/src/providers/index.ts +24 -0
  162. package/src/providers/pi-mono-adapter.ts +442 -0
  163. package/src/providers/pi-mono-extension.ts +624 -0
  164. package/src/providers/pi-mono-mcp-client.ts +124 -0
  165. package/src/providers/types.ts +75 -0
  166. package/src/scheduler/scheduler.test.ts +2 -0
  167. package/src/scheduler/scheduler.ts +231 -40
  168. package/src/server.ts +97 -6
  169. package/src/slack/HEURISTICS.md +105 -0
  170. package/src/slack/actions.ts +133 -0
  171. package/src/slack/app.ts +7 -0
  172. package/src/slack/assistant.ts +118 -0
  173. package/src/slack/blocks.ts +233 -0
  174. package/src/slack/channel-activity.ts +177 -0
  175. package/src/slack/commands.ts +31 -17
  176. package/src/slack/files.ts +1 -1
  177. package/src/slack/handlers.test.ts +114 -1
  178. package/src/slack/handlers.ts +230 -55
  179. package/src/slack/responses.ts +120 -67
  180. package/src/slack/router.ts +17 -99
  181. package/src/slack/templates.ts +55 -0
  182. package/src/slack/thread-buffer.ts +213 -0
  183. package/src/slack/watcher.ts +119 -4
  184. package/src/tests/agent-activity.test.ts +247 -0
  185. package/src/tests/agentmail-filters.test.ts +97 -0
  186. package/src/tests/artifact-sdk.test.ts +800 -0
  187. package/src/tests/base-prompt.test.ts +264 -0
  188. package/src/tests/build-pi-skills.test.ts +127 -0
  189. package/src/tests/channel-activity.test.ts +363 -0
  190. package/src/tests/claude-adapter.test.ts +126 -0
  191. package/src/tests/context-versioning.test.ts +425 -0
  192. package/src/tests/db-queries-oauth.test.ts +197 -0
  193. package/src/tests/db-queries-tracker.test.ts +230 -0
  194. package/src/tests/epics.test.ts +3 -3
  195. package/src/tests/error-tracker.test.ts +368 -0
  196. package/src/tests/fetch-resolved-env.test.ts +167 -0
  197. package/src/tests/generate-default-claude-md.test.ts +9 -1
  198. package/src/tests/generate-identity-templates.test.ts +124 -0
  199. package/src/tests/gitlab-auth.test.ts +109 -0
  200. package/src/tests/gitlab-handlers.test.ts +691 -0
  201. package/src/tests/gitlab-vcs-db.test.ts +177 -0
  202. package/src/tests/heartbeat.test.ts +364 -0
  203. package/src/tests/http-api-integration.test.ts +1698 -0
  204. package/src/tests/linear-outbound-sync.test.ts +200 -0
  205. package/src/tests/linear-webhook.test.ts +406 -0
  206. package/src/tests/match-route.test.ts +187 -0
  207. package/src/tests/memory.test.ts +737 -0
  208. package/src/tests/migration-runner-regressions.test.ts +86 -0
  209. package/src/tests/model-control.test.ts +338 -0
  210. package/src/tests/oauth-wrapper.test.ts +147 -0
  211. package/src/tests/onboard-compose.test.ts +138 -0
  212. package/src/tests/onboard-env.test.ts +174 -0
  213. package/src/tests/onboard-manifest.test.ts +137 -0
  214. package/src/tests/pi-mono-adapter.test.ts +234 -0
  215. package/src/tests/pool-session-logs.test.ts +199 -0
  216. package/src/tests/progress-dedup.test.ts +98 -0
  217. package/src/tests/prompt-template-github.test.ts +682 -0
  218. package/src/tests/prompt-template-remaining.test.ts +504 -0
  219. package/src/tests/prompt-template-resolver.test.ts +621 -0
  220. package/src/tests/prompt-template-session.test.ts +363 -0
  221. package/src/tests/prompt-templates-db.test.ts +616 -0
  222. package/src/tests/provider-adapter.test.ts +122 -0
  223. package/src/tests/provider-command-format.test.ts +98 -0
  224. package/src/tests/reload-config.test.ts +170 -0
  225. package/src/tests/runner-polling-api.test.ts +25 -20
  226. package/src/tests/scheduled-tasks.test.ts +104 -0
  227. package/src/tests/scheduler-backoff.test.ts +166 -0
  228. package/src/tests/self-improvement.test.ts +541 -0
  229. package/src/tests/session-attach.test.ts +536 -0
  230. package/src/tests/session-costs.test.ts +267 -1
  231. package/src/tests/slack-actions.test.ts +133 -0
  232. package/src/tests/slack-assistant.test.ts +136 -0
  233. package/src/tests/slack-blocks.test.ts +246 -0
  234. package/src/tests/slack-metadata-inheritance.test.ts +243 -0
  235. package/src/tests/slack-queue-offline.test.ts +174 -0
  236. package/src/tests/slack-router.test.ts +181 -0
  237. package/src/tests/slack-thread-buffer.test.ts +305 -0
  238. package/src/tests/slack-thread-followups.test.ts +298 -0
  239. package/src/tests/slack-watcher.test.ts +101 -0
  240. package/src/tests/structured-output.test.ts +307 -0
  241. package/src/tests/swarm-repos.test.ts +198 -0
  242. package/src/tests/task-cancellation.test.ts +6 -4
  243. package/src/tests/task-working-dir.test.ts +176 -0
  244. package/src/tests/template-fetch.test.ts +490 -0
  245. package/src/tests/tool-annotations.test.ts +371 -0
  246. package/src/tests/tracker-tools.test.ts +184 -0
  247. package/src/tests/update-profile-agentid.test.ts +248 -0
  248. package/src/tests/update-profile-api.test.ts +143 -3
  249. package/src/tests/update-profile-auth.test.ts +195 -0
  250. package/src/tests/validation-adapters.test.ts +86 -0
  251. package/src/tests/vcs-provider.test.ts +27 -0
  252. package/src/tests/workflow-agent-task.test.ts +196 -0
  253. package/src/tests/workflow-async-v2.test.ts +508 -0
  254. package/src/tests/workflow-convergence.test.ts +541 -0
  255. package/src/tests/workflow-definition-validation.test.ts +366 -0
  256. package/src/tests/workflow-engine-v2.test.ts +691 -0
  257. package/src/tests/workflow-executors.test.ts +736 -0
  258. package/src/tests/workflow-http-v2.test.ts +599 -0
  259. package/src/tests/workflow-integration-io.test.ts +902 -0
  260. package/src/tests/workflow-io-schemas.test.ts +624 -0
  261. package/src/tests/workflow-registry.test.ts +592 -0
  262. package/src/tests/workflow-retry-v2.test.ts +401 -0
  263. package/src/tests/workflow-retry-validation.test.ts +282 -0
  264. package/src/tests/workflow-schedule-trigger.test.ts +104 -0
  265. package/src/tests/workflow-template.test.ts +288 -0
  266. package/src/tests/workflow-trigger-schema.test.ts +359 -0
  267. package/src/tests/workflow-triggers-v2.test.ts +264 -0
  268. package/src/tests/workflow-versions.test.ts +208 -0
  269. package/src/tests/workflow-workspace.test.ts +272 -0
  270. package/src/tests/x402-client.test.ts +117 -0
  271. package/src/tests/x402-config.test.ts +182 -0
  272. package/src/tests/x402-spending-tracker.test.ts +185 -0
  273. package/src/tools/cancel-task.ts +2 -0
  274. package/src/tools/context-diff.ts +171 -0
  275. package/src/tools/context-history.ts +138 -0
  276. package/src/tools/create-channel.ts +1 -0
  277. package/src/tools/db-query.ts +78 -0
  278. package/src/tools/delete-channel.ts +132 -0
  279. package/src/tools/epics/assign-task-to-epic.ts +1 -0
  280. package/src/tools/epics/create-epic.ts +3 -2
  281. package/src/tools/epics/delete-epic.ts +2 -0
  282. package/src/tools/epics/get-epic-details.ts +2 -0
  283. package/src/tools/epics/list-epics.ts +2 -0
  284. package/src/tools/epics/unassign-task-from-epic.ts +1 -0
  285. package/src/tools/epics/update-epic.ts +7 -4
  286. package/src/tools/get-swarm.ts +2 -0
  287. package/src/tools/get-task-details.ts +2 -0
  288. package/src/tools/get-tasks.ts +27 -1
  289. package/src/tools/inject-learning.ts +106 -0
  290. package/src/tools/join-swarm.ts +17 -7
  291. package/src/tools/list-channels.ts +2 -0
  292. package/src/tools/list-services.ts +2 -0
  293. package/src/tools/memory-get.ts +56 -0
  294. package/src/tools/memory-search.ts +131 -0
  295. package/src/tools/my-agent-info.ts +2 -0
  296. package/src/tools/poll-task.ts +2 -20
  297. package/src/tools/post-message.ts +1 -0
  298. package/src/tools/prompt-templates/delete.ts +86 -0
  299. package/src/tools/prompt-templates/get.ts +89 -0
  300. package/src/tools/prompt-templates/index.ts +5 -0
  301. package/src/tools/prompt-templates/list.ts +95 -0
  302. package/src/tools/prompt-templates/preview.ts +84 -0
  303. package/src/tools/prompt-templates/set.ts +117 -0
  304. package/src/tools/read-messages.ts +2 -0
  305. package/src/tools/register-agentmail-inbox.ts +166 -0
  306. package/src/tools/register-service.ts +2 -0
  307. package/src/tools/schedules/create-schedule.ts +134 -24
  308. package/src/tools/schedules/delete-schedule.ts +2 -0
  309. package/src/tools/schedules/list-schedules.ts +20 -4
  310. package/src/tools/schedules/run-schedule-now.ts +1 -0
  311. package/src/tools/schedules/update-schedule.ts +49 -17
  312. package/src/tools/send-task.ts +132 -10
  313. package/src/tools/slack-download-file.ts +4 -2
  314. package/src/tools/slack-list-channels.ts +2 -0
  315. package/src/tools/slack-post.ts +2 -0
  316. package/src/tools/slack-read.ts +2 -0
  317. package/src/tools/slack-reply.ts +2 -0
  318. package/src/tools/slack-upload-file.ts +2 -0
  319. package/src/tools/store-progress.ts +205 -4
  320. package/src/tools/swarm-config/delete-config.ts +87 -0
  321. package/src/tools/swarm-config/get-config.ts +108 -0
  322. package/src/tools/swarm-config/index.ts +4 -0
  323. package/src/tools/swarm-config/list-config.ts +99 -0
  324. package/src/tools/swarm-config/set-config.ts +118 -0
  325. package/src/tools/task-action.ts +50 -5
  326. package/src/tools/task-dedup.ts +97 -0
  327. package/src/tools/templates.ts +53 -0
  328. package/src/tools/tool-config.ts +124 -0
  329. package/src/tools/tracker/index.ts +6 -0
  330. package/src/tools/tracker/tracker-link-epic.ts +64 -0
  331. package/src/tools/tracker/tracker-link-task.ts +64 -0
  332. package/src/tools/tracker/tracker-map-agent.ts +57 -0
  333. package/src/tools/tracker/tracker-status.ts +56 -0
  334. package/src/tools/tracker/tracker-sync-status.ts +42 -0
  335. package/src/tools/tracker/tracker-unlink.ts +41 -0
  336. package/src/tools/unregister-service.ts +2 -0
  337. package/src/tools/update-profile.ts +172 -17
  338. package/src/tools/update-service-status.ts +2 -0
  339. package/src/tools/utils.ts +10 -1
  340. package/src/tools/workflows/create-workflow.ts +129 -0
  341. package/src/tools/workflows/delete-workflow.ts +42 -0
  342. package/src/tools/workflows/get-workflow-run.ts +59 -0
  343. package/src/tools/workflows/get-workflow.ts +53 -0
  344. package/src/tools/workflows/index.ts +9 -0
  345. package/src/tools/workflows/list-workflow-runs.ts +48 -0
  346. package/src/tools/workflows/list-workflows.ts +42 -0
  347. package/src/tools/workflows/retry-workflow-run.ts +40 -0
  348. package/src/tools/workflows/trigger-workflow.ts +96 -0
  349. package/src/tools/workflows/update-workflow.ts +133 -0
  350. package/src/tracker/types.ts +51 -0
  351. package/src/types.ts +530 -14
  352. package/src/utils/credentials.test.ts +156 -0
  353. package/src/utils/credentials.ts +50 -0
  354. package/src/utils/error-tracker.ts +190 -0
  355. package/src/vcs/index.ts +15 -0
  356. package/src/vcs/types.ts +5 -0
  357. package/src/workflows/checkpoint.ts +121 -0
  358. package/src/workflows/cooldown.ts +28 -0
  359. package/src/workflows/definition.ts +235 -0
  360. package/src/workflows/engine.ts +580 -0
  361. package/src/workflows/event-bus.ts +29 -0
  362. package/src/workflows/executors/agent-task.ts +103 -0
  363. package/src/workflows/executors/base.ts +86 -0
  364. package/src/workflows/executors/code-match.ts +88 -0
  365. package/src/workflows/executors/index.ts +16 -0
  366. package/src/workflows/executors/notify.ts +93 -0
  367. package/src/workflows/executors/property-match.ts +104 -0
  368. package/src/workflows/executors/raw-llm.ts +83 -0
  369. package/src/workflows/executors/registry.ts +76 -0
  370. package/src/workflows/executors/script.ts +103 -0
  371. package/src/workflows/executors/validate.ts +215 -0
  372. package/src/workflows/executors/vcs.ts +58 -0
  373. package/src/workflows/index.ts +61 -0
  374. package/src/workflows/input.ts +46 -0
  375. package/src/workflows/json-schema-validator.ts +118 -0
  376. package/src/workflows/recovery.ts +139 -0
  377. package/src/workflows/resume.ts +229 -0
  378. package/src/workflows/retry-poller.ts +216 -0
  379. package/src/workflows/template.ts +74 -0
  380. package/src/workflows/templates.ts +86 -0
  381. package/src/workflows/triggers.ts +124 -0
  382. package/src/workflows/validation.ts +104 -0
  383. package/src/workflows/version.ts +44 -0
  384. package/src/x402/cli.ts +140 -0
  385. package/src/x402/client.ts +192 -0
  386. package/src/x402/config.ts +131 -0
  387. package/src/x402/index.ts +37 -0
  388. package/src/x402/openfort-signer.ts +83 -0
  389. package/src/x402/spending-tracker.ts +109 -0
  390. package/templates/official/coder/CLAUDE.md +49 -0
  391. package/templates/official/coder/IDENTITY.md +28 -0
  392. package/templates/official/coder/SOUL.md +43 -0
  393. package/templates/official/coder/TOOLS.md +40 -0
  394. package/templates/official/coder/config.json +23 -0
  395. package/templates/official/coder/start-up.sh +23 -0
  396. package/templates/official/content-reviewer/CLAUDE.md +68 -0
  397. package/templates/official/content-reviewer/IDENTITY.md +28 -0
  398. package/templates/official/content-reviewer/SOUL.md +44 -0
  399. package/templates/official/content-reviewer/TOOLS.md +37 -0
  400. package/templates/official/content-reviewer/config.json +23 -0
  401. package/templates/official/content-reviewer/start-up.sh +23 -0
  402. package/templates/official/content-strategist/CLAUDE.md +63 -0
  403. package/templates/official/content-strategist/IDENTITY.md +33 -0
  404. package/templates/official/content-strategist/SOUL.md +48 -0
  405. package/templates/official/content-strategist/TOOLS.md +47 -0
  406. package/templates/official/content-strategist/config.json +23 -0
  407. package/templates/official/content-strategist/start-up.sh +23 -0
  408. package/templates/official/content-writer/CLAUDE.md +72 -0
  409. package/templates/official/content-writer/IDENTITY.md +30 -0
  410. package/templates/official/content-writer/SOUL.md +46 -0
  411. package/templates/official/content-writer/TOOLS.md +44 -0
  412. package/templates/official/content-writer/config.json +23 -0
  413. package/templates/official/content-writer/start-up.sh +23 -0
  414. package/templates/official/forward-deployed-engineer/CLAUDE.md +54 -0
  415. package/templates/official/forward-deployed-engineer/IDENTITY.md +37 -0
  416. package/templates/official/forward-deployed-engineer/SOUL.md +55 -0
  417. package/templates/official/forward-deployed-engineer/config.json +21 -0
  418. package/templates/official/lead/CLAUDE.md +33 -0
  419. package/templates/official/lead/IDENTITY.md +36 -0
  420. package/templates/official/lead/SOUL.md +51 -0
  421. package/templates/official/lead/config.json +22 -0
  422. package/templates/official/researcher/CLAUDE.md +46 -0
  423. package/templates/official/researcher/IDENTITY.md +28 -0
  424. package/templates/official/researcher/SOUL.md +43 -0
  425. package/templates/official/researcher/config.json +21 -0
  426. package/templates/official/reviewer/CLAUDE.md +63 -0
  427. package/templates/official/reviewer/IDENTITY.md +28 -0
  428. package/templates/official/reviewer/SOUL.md +45 -0
  429. package/templates/official/reviewer/config.json +21 -0
  430. package/templates/official/tester/CLAUDE.md +53 -0
  431. package/templates/official/tester/IDENTITY.md +28 -0
  432. package/templates/official/tester/SOUL.md +55 -0
  433. package/templates/official/tester/config.json +21 -0
  434. package/templates/schema.ts +35 -0
  435. package/.claude/settings.local.json +0 -115
  436. package/.dockerignore +0 -61
  437. package/.editorconfig +0 -15
  438. package/.env.docker.example +0 -39
  439. package/.env.example +0 -40
  440. package/.github/workflows/ci.yml +0 -76
  441. package/.github/workflows/docker-and-deploy.yml +0 -117
  442. package/.wts-config.json +0 -4
  443. package/.wts-setup.ts +0 -102
  444. package/CLAUDE.md +0 -104
  445. package/CONTRIBUTING.md +0 -270
  446. package/DEPLOYMENT.md +0 -605
  447. package/Dockerfile +0 -57
  448. package/Dockerfile.worker +0 -157
  449. package/FAQ.md +0 -19
  450. package/MCP.md +0 -406
  451. package/UI.md +0 -40
  452. package/assets/agent-swarm-logo-orange.png +0 -0
  453. package/assets/agent-swarm-logo.png +0 -0
  454. package/assets/agent-swarm.mp4 +0 -0
  455. package/assets/agent-swarm.png +0 -0
  456. package/biome.json +0 -39
  457. package/deploy/DEPLOY.md +0 -60
  458. package/deploy/agent-swarm.service +0 -17
  459. package/deploy/docker-push.ts +0 -30
  460. package/deploy/install.ts +0 -85
  461. package/deploy/prod-db.ts +0 -42
  462. package/deploy/uninstall.ts +0 -12
  463. package/deploy/update.ts +0 -21
  464. package/docker-compose.example.yml +0 -159
  465. package/docker-entrypoint.sh +0 -352
  466. package/ecosystem.config.cjs +0 -66
  467. package/plugin/README.md +0 -1
  468. package/plugin/hooks/hooks.json +0 -71
  469. package/pyproject.toml +0 -9
  470. package/scripts/generate-mcp-docs.ts +0 -415
  471. package/slack-manifest.json +0 -71
  472. package/src/tests/get-inbox-message.test.ts +0 -145
  473. package/src/tools/get-inbox-message.ts +0 -89
  474. package/src/tools/inbox-delegate.ts +0 -113
  475. package/thoughts/shared/plans/2025-12-18-slack-integration.md +0 -1195
  476. package/thoughts/shared/plans/2025-12-19-agent-log-streaming.md +0 -732
  477. package/thoughts/shared/plans/2025-12-19-role-based-swarm-plugin.md +0 -361
  478. package/thoughts/shared/plans/2025-12-20-mobile-responsive-ui.md +0 -501
  479. package/thoughts/shared/plans/2025-12-20-startup-team-swarm.md +0 -560
  480. package/thoughts/shared/plans/2025-12-23-runner-level-polling.md +0 -934
  481. package/thoughts/shared/plans/2025-12-23-runner-session-logs.md +0 -1000
  482. package/thoughts/shared/plans/2025-12-23-worker-lead-spawn-triggers.md +0 -568
  483. package/thoughts/shared/plans/2026-01-09-inverse-teleport.md +0 -1516
  484. package/thoughts/shared/plans/2026-01-12-agent-rename-pm2-control.md +0 -1133
  485. package/thoughts/shared/plans/2026-01-12-github-app-integration.md +0 -380
  486. package/thoughts/shared/plans/2026-01-12-lead-inbox-model.md +0 -876
  487. package/thoughts/shared/plans/2026-01-12-ralph-wiggum-integration.md +0 -463
  488. package/thoughts/shared/plans/2026-01-13-agent-concurrency.md +0 -691
  489. package/thoughts/shared/plans/2026-01-13-github-assignment-handling.md +0 -690
  490. package/thoughts/shared/plans/2026-01-13-prevent-duplicate-trigger-processing.md +0 -1071
  491. package/thoughts/shared/plans/2026-01-14-fix-slack-thread-context.md +0 -507
  492. package/thoughts/shared/plans/2026-01-15-scheduled-tasks-implementation.md +0 -565
  493. package/thoughts/shared/plans/2026-01-15-usage-cost-tracking-ui.md +0 -1479
  494. package/thoughts/shared/plans/2026-01-16-epics-feature-implementation.md +0 -1230
  495. package/thoughts/shared/research/.gitkeep +0 -0
  496. package/thoughts/shared/research/2025-01-09-inverse-teleport-plan-review.md +0 -420
  497. package/thoughts/shared/research/2025-12-18-slack-integration.md +0 -442
  498. package/thoughts/shared/research/2025-12-19-agent-log-streaming.md +0 -339
  499. package/thoughts/shared/research/2025-12-19-agent-secrets-cli-research.md +0 -390
  500. package/thoughts/shared/research/2025-12-21-gemini-cli-integration.md +0 -376
  501. package/thoughts/shared/research/2025-12-22-runner-loop-architecture.md +0 -582
  502. package/thoughts/shared/research/2025-12-22-setup-experience-improvements.md +0 -264
  503. package/thoughts/shared/research/2026-01-13-lead-duplicate-trigger-processing.md +0 -223
  504. package/thoughts/shared/research/2026-01-14-lead-slack-thread-context.md +0 -277
  505. package/thoughts/shared/research/2026-01-15-ai-tracker-agent-swarm-integration.md +0 -376
  506. package/thoughts/shared/research/2026-01-15-auto-starting-processes-in-worker-containers.md +0 -787
  507. package/thoughts/shared/research/2026-01-15-scheduled-tasks.md +0 -390
  508. package/thoughts/shared/research/2026-01-16-epics-feature-research.md +0 -437
  509. package/thoughts/taras/plans/2026-01-22-agent-swarm-schemas.md +0 -98
  510. package/thoughts/taras/plans/2026-01-28-per-worker-claude-md.md +0 -617
  511. package/thoughts/taras/plans/2026-01-28-sentry-cli-integration.md +0 -214
  512. package/thoughts/taras/research/2026-01-22-vercel-cli-integration.md +0 -287
  513. package/thoughts/taras/research/2026-01-27-excessive-polling-issue.md +0 -311
  514. package/thoughts/taras/research/2026-01-28-per-worker-claude-md.md +0 -383
  515. package/thoughts/taras/research/2026-01-28-sentry-cli-integration.md +0 -240
  516. package/tsconfig.json +0 -37
  517. package/ui/CLAUDE.md +0 -49
  518. package/ui/bun.lock +0 -771
  519. package/ui/index.html +0 -22
  520. package/ui/package-lock.json +0 -5290
  521. package/ui/package.json +0 -33
  522. package/ui/pnpm-lock.yaml +0 -3341
  523. package/ui/postcss.config.js +0 -6
  524. package/ui/public/logo.png +0 -0
  525. package/ui/src/App.tsx +0 -63
  526. package/ui/src/components/ActivityFeed.tsx +0 -440
  527. package/ui/src/components/AgentDetailPanel.tsx +0 -733
  528. package/ui/src/components/AgentsPanel.tsx +0 -815
  529. package/ui/src/components/ChatPanel.tsx +0 -1920
  530. package/ui/src/components/ConfigModal.tsx +0 -253
  531. package/ui/src/components/Dashboard.tsx +0 -832
  532. package/ui/src/components/EditAgentProfileModal.tsx +0 -433
  533. package/ui/src/components/EpicDetailPage.tsx +0 -741
  534. package/ui/src/components/EpicsPanel.tsx +0 -566
  535. package/ui/src/components/Header.tsx +0 -160
  536. package/ui/src/components/JsonViewer.tsx +0 -171
  537. package/ui/src/components/ScheduledTaskDetailPanel.tsx +0 -517
  538. package/ui/src/components/ScheduledTasksPanel.tsx +0 -639
  539. package/ui/src/components/ServicesPanel.tsx +0 -622
  540. package/ui/src/components/SessionLogPanel.tsx +0 -1219
  541. package/ui/src/components/StatsBar.tsx +0 -321
  542. package/ui/src/components/StatusBadge.tsx +0 -168
  543. package/ui/src/components/TaskDetailPanel.tsx +0 -903
  544. package/ui/src/components/TasksPanel.tsx +0 -614
  545. package/ui/src/components/UsageCharts.tsx +0 -216
  546. package/ui/src/components/UsageTab.tsx +0 -394
  547. package/ui/src/hooks/queries.ts +0 -353
  548. package/ui/src/hooks/useAutoScroll.ts +0 -83
  549. package/ui/src/index.css +0 -257
  550. package/ui/src/lib/api.ts +0 -268
  551. package/ui/src/lib/config.ts +0 -35
  552. package/ui/src/lib/contentPreview.ts +0 -208
  553. package/ui/src/lib/theme.ts +0 -214
  554. package/ui/src/lib/utils.ts +0 -88
  555. package/ui/src/main.tsx +0 -28
  556. package/ui/src/types/api.ts +0 -323
  557. package/ui/src/vite-env.d.ts +0 -1
  558. package/ui/tailwind.config.js +0 -37
  559. package/ui/tsconfig.json +0 -31
  560. package/ui/vite.config.ts +0 -35
  561. /package/{thoughts/shared/plans → templates/community}/.gitkeep +0 -0
@@ -1,815 +0,0 @@
1
- import Box from "@mui/joy/Box";
2
- import Card from "@mui/joy/Card";
3
- import IconButton from "@mui/joy/IconButton";
4
- import Input from "@mui/joy/Input";
5
- import Option from "@mui/joy/Option";
6
- import Select from "@mui/joy/Select";
7
- import { useColorScheme } from "@mui/joy/styles";
8
- import Table from "@mui/joy/Table";
9
- import Typography from "@mui/joy/Typography";
10
- import { useMemo, useState } from "react";
11
- import { useAgents, useSessionCosts, useUpdateAgentName } from "../hooks/queries";
12
- import { formatCompactNumber, formatCurrency } from "../lib/utils";
13
- import type { AgentStatus, AgentWithTasks } from "../types/api";
14
- import StatusBadge from "./StatusBadge";
15
-
16
- function formatSmartTime(dateStr: string): string {
17
- const date = new Date(dateStr);
18
- const now = new Date();
19
- const diffMs = now.getTime() - date.getTime();
20
- const diffMins = Math.floor(diffMs / 60000);
21
- const diffHours = Math.floor(diffMins / 60);
22
-
23
- if (diffHours < 6) {
24
- if (diffMins < 1) return "just now";
25
- if (diffMins < 60) return `${diffMins}m ago`;
26
- return `${diffHours}h ago`;
27
- }
28
-
29
- const isToday = date.toDateString() === now.toDateString();
30
- if (isToday) {
31
- return date.toLocaleTimeString([], { hour: "2-digit", minute: "2-digit" });
32
- }
33
-
34
- return date.toLocaleDateString([], {
35
- month: "short",
36
- day: "numeric",
37
- hour: "2-digit",
38
- minute: "2-digit",
39
- });
40
- }
41
-
42
- interface AgentsRowProps {
43
- agent: AgentWithTasks;
44
- selected: boolean;
45
- onClick: () => void;
46
- isDark: boolean;
47
- usage?: { cost: number; tokens: number };
48
- }
49
-
50
- function AgentRow({ agent, selected, onClick, isDark, usage }: AgentsRowProps) {
51
- const [isEditing, setIsEditing] = useState(false);
52
- const [editName, setEditName] = useState(agent.name);
53
- const [error, setError] = useState<string | null>(null);
54
- const updateNameMutation = useUpdateAgentName();
55
-
56
- const activeTasks = agent.tasks.filter(
57
- (t) => t.status === "pending" || t.status === "in_progress",
58
- ).length;
59
-
60
- const isActive = agent.status === "busy";
61
-
62
- const colors = {
63
- amber: isDark ? "#F5A623" : "#D48806",
64
- gold: isDark ? "#D4A574" : "#8B6914",
65
- dormant: isDark ? "#6B5344" : "#A89A7C",
66
- selectedBg: isDark ? "rgba(245, 166, 35, 0.08)" : "rgba(212, 136, 6, 0.08)",
67
- amberGlow: isDark ? "0 0 10px rgba(245, 166, 35, 0.6)" : "0 0 8px rgba(212, 136, 6, 0.4)",
68
- amberSoftBg: isDark ? "rgba(245, 166, 35, 0.1)" : "rgba(212, 136, 6, 0.1)",
69
- amberBorder: isDark ? "rgba(245, 166, 35, 0.3)" : "rgba(212, 136, 6, 0.3)",
70
- amberTextShadow: isDark ? "0 0 10px rgba(245, 166, 35, 0.5)" : "none",
71
- };
72
-
73
- const handleSave = async () => {
74
- if (editName.trim() === agent.name) {
75
- setIsEditing(false);
76
- return;
77
- }
78
-
79
- try {
80
- await updateNameMutation.mutateAsync({ id: agent.id, name: editName.trim() });
81
- setIsEditing(false);
82
- setError(null);
83
- } catch (err) {
84
- setError((err as Error).message);
85
- }
86
- };
87
-
88
- const handleCancel = () => {
89
- setEditName(agent.name);
90
- setIsEditing(false);
91
- setError(null);
92
- };
93
-
94
- return (
95
- <tr
96
- onClick={onClick}
97
- style={{
98
- cursor: "pointer",
99
- backgroundColor: selected ? colors.selectedBg : undefined,
100
- }}
101
- className="row-hover"
102
- >
103
- <td>
104
- <Box sx={{ display: "flex", flexDirection: "column", gap: 0.5 }}>
105
- <Box sx={{ display: "flex", alignItems: "center", gap: 1.5 }}>
106
- {/* Agent status dot */}
107
- <Box
108
- sx={{
109
- width: 8,
110
- height: 8,
111
- borderRadius: "50%",
112
- bgcolor: isActive
113
- ? colors.amber
114
- : agent.status === "idle"
115
- ? colors.gold
116
- : colors.dormant,
117
- boxShadow: isActive ? colors.amberGlow : "none",
118
- animation: isActive ? "pulse-amber 2s ease-in-out infinite" : undefined,
119
- }}
120
- />
121
- {isEditing ? (
122
- <Input
123
- value={editName}
124
- onChange={(e) => setEditName(e.target.value)}
125
- onKeyDown={(e) => {
126
- if (e.key === "Enter") {
127
- e.stopPropagation();
128
- handleSave();
129
- }
130
- if (e.key === "Escape") {
131
- e.stopPropagation();
132
- handleCancel();
133
- }
134
- }}
135
- onBlur={handleSave}
136
- onClick={(e) => e.stopPropagation()}
137
- autoFocus
138
- size="sm"
139
- sx={{
140
- fontFamily: "code",
141
- fontSize: "0.875rem",
142
- fontWeight: 600,
143
- maxWidth: 150,
144
- bgcolor: "background.surface",
145
- borderColor: colors.amber,
146
- color: "text.primary",
147
- "&:focus-within": {
148
- borderColor: colors.amber,
149
- boxShadow: colors.amberGlow,
150
- },
151
- }}
152
- />
153
- ) : (
154
- <Box sx={{ display: "flex", alignItems: "center", gap: 0.5 }}>
155
- <Typography
156
- sx={{
157
- fontFamily: "code",
158
- fontWeight: 600,
159
- color: agent.isLead ? colors.amber : "text.primary",
160
- whiteSpace: "nowrap",
161
- }}
162
- >
163
- {agent.name}
164
- </Typography>
165
- <IconButton
166
- size="sm"
167
- onClick={(e) => {
168
- e.stopPropagation();
169
- setIsEditing(true);
170
- }}
171
- sx={{
172
- opacity: 0,
173
- ".row-hover:hover &": { opacity: 0.6 },
174
- "&:hover": {
175
- opacity: 1,
176
- bgcolor: isDark ? "rgba(245, 166, 35, 0.1)" : "rgba(212, 136, 6, 0.15)",
177
- },
178
- minHeight: 20,
179
- minWidth: 20,
180
- padding: 0.25,
181
- color: "text.secondary",
182
- }}
183
- >
184
-
185
- </IconButton>
186
- </Box>
187
- )}
188
- {agent.isLead && (
189
- <Typography
190
- sx={{
191
- fontFamily: "code",
192
- fontSize: "0.6rem",
193
- color: colors.amber,
194
- textShadow: colors.amberTextShadow,
195
- bgcolor: colors.amberSoftBg,
196
- px: 0.75,
197
- py: 0.25,
198
- borderRadius: 1,
199
- border: `1px solid ${colors.amberBorder}`,
200
- }}
201
- >
202
- LEAD
203
- </Typography>
204
- )}
205
- </Box>
206
- {error && (
207
- <Typography
208
- sx={{
209
- fontFamily: "code",
210
- fontSize: "0.65rem",
211
- color: "#d32f2f",
212
- pl: 3.5,
213
- }}
214
- >
215
- {error}
216
- </Typography>
217
- )}
218
- </Box>
219
- </td>
220
- <td>
221
- <Typography
222
- sx={{
223
- fontFamily: "code",
224
- fontSize: "0.75rem",
225
- color: agent.role ? "text.secondary" : "text.tertiary",
226
- whiteSpace: "nowrap",
227
- overflow: "hidden",
228
- textOverflow: "ellipsis",
229
- maxWidth: 120,
230
- }}
231
- >
232
- {agent.role || "—"}
233
- </Typography>
234
- </td>
235
- <td>
236
- <StatusBadge status={agent.status} />
237
- </td>
238
- <td>
239
- <Typography
240
- sx={{
241
- fontFamily: "code",
242
- fontSize: "0.8rem",
243
- color: agent.capacity
244
- ? agent.capacity.current > 0
245
- ? colors.amber
246
- : "text.tertiary"
247
- : activeTasks > 0
248
- ? colors.amber
249
- : "text.tertiary",
250
- whiteSpace: "nowrap",
251
- }}
252
- >
253
- {agent.capacity
254
- ? `${agent.capacity.current}/${agent.capacity.max}`
255
- : `${activeTasks}/${agent.tasks.length}`}
256
- </Typography>
257
- </td>
258
- <td>
259
- <Box sx={{ display: "flex", flexDirection: "column", gap: 0.25 }}>
260
- <Typography
261
- sx={{
262
- fontFamily: "code",
263
- fontSize: "0.7rem",
264
- color: usage && usage.cost > 0 ? colors.amber : "text.tertiary",
265
- }}
266
- >
267
- {usage ? formatCurrency(usage.cost) : "—"}
268
- </Typography>
269
- <Typography
270
- sx={{
271
- fontFamily: "code",
272
- fontSize: "0.6rem",
273
- color: "text.tertiary",
274
- }}
275
- >
276
- {usage ? `${formatCompactNumber(usage.tokens)} tok` : ""}
277
- </Typography>
278
- </Box>
279
- </td>
280
- <td>
281
- <Typography
282
- sx={{
283
- fontFamily: "code",
284
- fontSize: "0.7rem",
285
- color: "text.tertiary",
286
- }}
287
- >
288
- {formatSmartTime(agent.lastUpdatedAt)}
289
- </Typography>
290
- </td>
291
- </tr>
292
- );
293
- }
294
-
295
- // Mobile card component
296
- interface AgentCardProps {
297
- agent: AgentWithTasks;
298
- selected: boolean;
299
- onClick: () => void;
300
- isDark: boolean;
301
- }
302
-
303
- function AgentCard({ agent, selected, onClick, isDark }: AgentCardProps) {
304
- const [isEditing, setIsEditing] = useState(false);
305
- const [editName, setEditName] = useState(agent.name);
306
- const [error, setError] = useState<string | null>(null);
307
- const updateNameMutation = useUpdateAgentName();
308
-
309
- const activeTasks = agent.tasks.filter(
310
- (t) => t.status === "pending" || t.status === "in_progress",
311
- ).length;
312
-
313
- const isActive = agent.status === "busy";
314
-
315
- const colors = {
316
- amber: isDark ? "#F5A623" : "#D48806",
317
- gold: isDark ? "#D4A574" : "#8B6914",
318
- dormant: isDark ? "#6B5344" : "#A89A7C",
319
- selectedBorder: isDark ? "#F5A623" : "#D48806",
320
- amberGlow: isDark ? "0 0 10px rgba(245, 166, 35, 0.6)" : "0 0 8px rgba(212, 136, 6, 0.4)",
321
- amberSoftBg: isDark ? "rgba(245, 166, 35, 0.1)" : "rgba(212, 136, 6, 0.1)",
322
- amberBorder: isDark ? "rgba(245, 166, 35, 0.3)" : "rgba(212, 136, 6, 0.3)",
323
- amberTextShadow: isDark ? "0 0 10px rgba(245, 166, 35, 0.5)" : "none",
324
- };
325
-
326
- const handleSave = async () => {
327
- if (editName.trim() === agent.name) {
328
- setIsEditing(false);
329
- return;
330
- }
331
-
332
- try {
333
- await updateNameMutation.mutateAsync({ id: agent.id, name: editName.trim() });
334
- setIsEditing(false);
335
- setError(null);
336
- } catch (err) {
337
- setError((err as Error).message);
338
- }
339
- };
340
-
341
- const handleCancel = () => {
342
- setEditName(agent.name);
343
- setIsEditing(false);
344
- setError(null);
345
- };
346
-
347
- return (
348
- <Box
349
- onClick={onClick}
350
- sx={{
351
- p: 2,
352
- mb: 1,
353
- borderRadius: "8px",
354
- border: "1px solid",
355
- borderColor: selected ? colors.selectedBorder : "neutral.outlinedBorder",
356
- bgcolor: selected ? colors.amberSoftBg : "background.surface",
357
- cursor: "pointer",
358
- transition: "all 0.2s ease",
359
- "&:active": {
360
- bgcolor: colors.amberSoftBg,
361
- },
362
- }}
363
- >
364
- <Box
365
- sx={{ display: "flex", justifyContent: "space-between", alignItems: "flex-start", mb: 1 }}
366
- >
367
- <Box sx={{ display: "flex", alignItems: "center", gap: 1, flex: 1, minWidth: 0 }}>
368
- <Box
369
- sx={{
370
- width: 10,
371
- height: 10,
372
- borderRadius: "50%",
373
- bgcolor: isActive
374
- ? colors.amber
375
- : agent.status === "idle"
376
- ? colors.gold
377
- : colors.dormant,
378
- boxShadow: isActive ? colors.amberGlow : "none",
379
- flexShrink: 0,
380
- }}
381
- />
382
- {isEditing ? (
383
- <Input
384
- value={editName}
385
- onChange={(e) => setEditName(e.target.value)}
386
- onKeyDown={(e) => {
387
- if (e.key === "Enter") {
388
- e.stopPropagation();
389
- handleSave();
390
- }
391
- if (e.key === "Escape") {
392
- e.stopPropagation();
393
- handleCancel();
394
- }
395
- }}
396
- onBlur={handleSave}
397
- onClick={(e) => e.stopPropagation()}
398
- autoFocus
399
- size="sm"
400
- sx={{
401
- fontFamily: "code",
402
- fontSize: "0.875rem",
403
- fontWeight: 600,
404
- maxWidth: 150,
405
- bgcolor: "background.surface",
406
- borderColor: colors.amber,
407
- color: "text.primary",
408
- "&:focus-within": {
409
- borderColor: colors.amber,
410
- boxShadow: colors.amberGlow,
411
- },
412
- }}
413
- />
414
- ) : (
415
- <Box sx={{ display: "flex", alignItems: "center", gap: 0.5, flex: 1, minWidth: 0 }}>
416
- <Typography
417
- sx={{
418
- fontFamily: "code",
419
- fontWeight: 600,
420
- color: agent.isLead ? colors.amber : "text.primary",
421
- whiteSpace: "nowrap",
422
- overflow: "hidden",
423
- textOverflow: "ellipsis",
424
- }}
425
- >
426
- {agent.name}
427
- </Typography>
428
- <IconButton
429
- size="sm"
430
- onClick={(e) => {
431
- e.stopPropagation();
432
- setIsEditing(true);
433
- }}
434
- sx={{
435
- flexShrink: 0,
436
- minHeight: 20,
437
- minWidth: 20,
438
- padding: 0.25,
439
- color: "text.secondary",
440
- "&:hover": {
441
- bgcolor: isDark ? "rgba(245, 166, 35, 0.1)" : "rgba(212, 136, 6, 0.15)",
442
- },
443
- }}
444
- >
445
-
446
- </IconButton>
447
- </Box>
448
- )}
449
- {agent.isLead && (
450
- <Typography
451
- sx={{
452
- fontFamily: "code",
453
- fontSize: "0.55rem",
454
- color: colors.amber,
455
- textShadow: colors.amberTextShadow,
456
- bgcolor: colors.amberSoftBg,
457
- px: 0.5,
458
- py: 0.2,
459
- borderRadius: 0.5,
460
- border: `1px solid ${colors.amberBorder}`,
461
- flexShrink: 0,
462
- }}
463
- >
464
- LEAD
465
- </Typography>
466
- )}
467
- </Box>
468
- <StatusBadge status={agent.status} />
469
- </Box>
470
- {error && (
471
- <Typography
472
- sx={{
473
- fontFamily: "code",
474
- fontSize: "0.65rem",
475
- color: "#d32f2f",
476
- mb: 1,
477
- }}
478
- >
479
- {error}
480
- </Typography>
481
- )}
482
- <Typography
483
- sx={{
484
- fontFamily: "code",
485
- fontSize: "0.75rem",
486
- color: "text.tertiary",
487
- }}
488
- >
489
- {agent.role || "No role"} ·{" "}
490
- {agent.capacity
491
- ? `${agent.capacity.current}/${agent.capacity.max}`
492
- : `${activeTasks}/${agent.tasks.length}`}{" "}
493
- tasks
494
- </Typography>
495
- </Box>
496
- );
497
- }
498
-
499
- interface AgentsPanelProps {
500
- selectedAgentId: string | null;
501
- onSelectAgent: (agentId: string | null) => void;
502
- statusFilter?: AgentStatus | "all";
503
- onStatusFilterChange?: (status: AgentStatus | "all") => void;
504
- }
505
-
506
- export default function AgentsPanel({
507
- selectedAgentId,
508
- onSelectAgent,
509
- statusFilter: controlledStatusFilter,
510
- onStatusFilterChange,
511
- }: AgentsPanelProps) {
512
- const [searchQuery, setSearchQuery] = useState("");
513
- const [internalStatusFilter, setInternalStatusFilter] = useState<AgentStatus | "all">("all");
514
-
515
- // Use controlled or internal state
516
- const statusFilter = controlledStatusFilter ?? internalStatusFilter;
517
- const setStatusFilter = onStatusFilterChange ?? setInternalStatusFilter;
518
-
519
- const { data: agents, isLoading } = useAgents();
520
- const { data: allCosts } = useSessionCosts({ limit: 2000 });
521
- const { mode } = useColorScheme();
522
- const isDark = mode === "dark";
523
-
524
- // Create agent usage map (memoized) - MTD = Month to Date
525
- const agentUsageMap = useMemo(() => {
526
- if (!allCosts) return new Map<string, { cost: number; tokens: number }>();
527
-
528
- const now = new Date();
529
- const startOfMonth = new Date(now.getFullYear(), now.getMonth(), 1);
530
-
531
- const map = new Map<string, { cost: number; tokens: number }>();
532
-
533
- allCosts
534
- .filter((c) => new Date(c.createdAt) >= startOfMonth)
535
- .forEach((cost) => {
536
- const existing = map.get(cost.agentId) || { cost: 0, tokens: 0 };
537
- map.set(cost.agentId, {
538
- cost: existing.cost + cost.totalCostUsd,
539
- tokens: existing.tokens + cost.inputTokens + cost.outputTokens,
540
- });
541
- });
542
-
543
- return map;
544
- }, [allCosts]);
545
-
546
- const colors = {
547
- amber: isDark ? "#F5A623" : "#D48806",
548
- amberGlow: isDark ? "0 0 8px rgba(245, 166, 35, 0.5)" : "0 0 6px rgba(212, 136, 6, 0.3)",
549
- hoverBg: isDark ? "rgba(245, 166, 35, 0.05)" : "rgba(212, 136, 6, 0.05)",
550
- hoverBorder: isDark ? "#4A3A2F" : "#D1C5B4",
551
- amberInputGlow: isDark ? "0 0 10px rgba(245, 166, 35, 0.2)" : "0 0 8px rgba(212, 136, 6, 0.15)",
552
- };
553
-
554
- // Filter agents based on search and status
555
- const filteredAgents = useMemo(() => {
556
- if (!agents) return [];
557
-
558
- return agents.filter((agent) => {
559
- // Search filter
560
- if (searchQuery.trim()) {
561
- const query = searchQuery.toLowerCase();
562
- if (!agent.name.toLowerCase().includes(query) && !agent.id.toLowerCase().includes(query)) {
563
- return false;
564
- }
565
- }
566
-
567
- // Status filter
568
- if (statusFilter !== "all" && agent.status !== statusFilter) {
569
- return false;
570
- }
571
-
572
- return true;
573
- });
574
- }, [agents, searchQuery, statusFilter]);
575
-
576
- if (isLoading) {
577
- return (
578
- <Card
579
- variant="outlined"
580
- sx={{
581
- p: 2,
582
- height: "100%",
583
- bgcolor: "background.surface",
584
- borderColor: "neutral.outlinedBorder",
585
- }}
586
- >
587
- <Typography sx={{ fontFamily: "code", color: "text.tertiary" }}>
588
- Loading agents...
589
- </Typography>
590
- </Card>
591
- );
592
- }
593
-
594
- if (!agents || agents.length === 0) {
595
- return (
596
- <Card
597
- variant="outlined"
598
- sx={{
599
- p: 2,
600
- height: "100%",
601
- bgcolor: "background.surface",
602
- borderColor: "neutral.outlinedBorder",
603
- }}
604
- >
605
- <Typography sx={{ fontFamily: "code", color: "text.tertiary" }}>
606
- No agents in the swarm
607
- </Typography>
608
- </Card>
609
- );
610
- }
611
-
612
- return (
613
- <Card
614
- variant="outlined"
615
- className="card-hover"
616
- sx={{
617
- p: 0,
618
- overflow: "hidden",
619
- height: "100%",
620
- display: "flex",
621
- flexDirection: "column",
622
- bgcolor: "background.surface",
623
- borderColor: "neutral.outlinedBorder",
624
- }}
625
- >
626
- {/* Header */}
627
- <Box
628
- sx={{
629
- px: { xs: 1.5, md: 2 },
630
- py: 1.5,
631
- borderBottom: "1px solid",
632
- borderColor: "neutral.outlinedBorder",
633
- bgcolor: "background.level1",
634
- display: "flex",
635
- flexDirection: { xs: "column", sm: "row" },
636
- alignItems: { xs: "stretch", sm: "center" },
637
- justifyContent: "space-between",
638
- gap: 1.5,
639
- }}
640
- >
641
- <Box sx={{ display: "flex", alignItems: "center", gap: 1 }}>
642
- {/* Hex accent */}
643
- <Box
644
- sx={{
645
- width: 8,
646
- height: 10,
647
- clipPath: "polygon(50% 0%, 100% 25%, 100% 75%, 50% 100%, 0% 75%, 0% 25%)",
648
- bgcolor: colors.amber,
649
- boxShadow: colors.amberGlow,
650
- }}
651
- />
652
- <Typography
653
- level="title-md"
654
- sx={{
655
- fontFamily: "display",
656
- fontWeight: 600,
657
- color: colors.amber,
658
- letterSpacing: "0.03em",
659
- fontSize: { xs: "0.9rem", md: "1rem" },
660
- }}
661
- >
662
- AGENTS
663
- </Typography>
664
- <Typography
665
- sx={{
666
- fontFamily: "code",
667
- fontSize: "0.7rem",
668
- color: "text.tertiary",
669
- }}
670
- >
671
- ({filteredAgents.length}/{agents.length})
672
- </Typography>
673
- </Box>
674
-
675
- {/* Filters */}
676
- <Box
677
- sx={{
678
- display: "flex",
679
- flexDirection: { xs: "column", sm: "row" },
680
- alignItems: { xs: "stretch", sm: "center" },
681
- gap: 1,
682
- }}
683
- >
684
- {/* Search */}
685
- <Input
686
- placeholder="Search..."
687
- value={searchQuery}
688
- onChange={(e) => setSearchQuery(e.target.value)}
689
- size="sm"
690
- sx={{
691
- fontFamily: "code",
692
- fontSize: "0.75rem",
693
- minWidth: { xs: "100%", sm: 140 },
694
- bgcolor: "background.surface",
695
- borderColor: "neutral.outlinedBorder",
696
- color: "text.primary",
697
- "&:hover": {
698
- borderColor: colors.hoverBorder,
699
- },
700
- "&:focus-within": {
701
- borderColor: colors.amber,
702
- boxShadow: colors.amberInputGlow,
703
- },
704
- }}
705
- />
706
-
707
- {/* Status Filter */}
708
- <Select
709
- value={statusFilter}
710
- onChange={(_, value) => setStatusFilter(value as AgentStatus | "all")}
711
- size="sm"
712
- sx={{
713
- fontFamily: "code",
714
- fontSize: "0.75rem",
715
- minWidth: { xs: "100%", sm: 100 },
716
- bgcolor: "background.surface",
717
- borderColor: "neutral.outlinedBorder",
718
- color: "text.secondary",
719
- "&:hover": {
720
- borderColor: colors.amber,
721
- },
722
- "& .MuiSelect-indicator": {
723
- color: "text.tertiary",
724
- },
725
- }}
726
- >
727
- <Option value="all">ALL</Option>
728
- <Option value="idle">IDLE</Option>
729
- <Option value="busy">BUSY</Option>
730
- <Option value="offline">OFFLINE</Option>
731
- </Select>
732
- </Box>
733
- </Box>
734
-
735
- {/* Content */}
736
- <Box sx={{ flex: 1, overflow: "auto" }}>
737
- {filteredAgents.length === 0 ? (
738
- <Box sx={{ p: 3, textAlign: "center" }}>
739
- <Typography sx={{ fontFamily: "code", color: "text.tertiary" }}>
740
- No agents match your filters
741
- </Typography>
742
- </Box>
743
- ) : (
744
- <>
745
- {/* Desktop Table */}
746
- <Box sx={{ display: { xs: "none", md: "block" } }}>
747
- <Table
748
- size="sm"
749
- sx={{
750
- "--TableCell-paddingY": "10px",
751
- "--TableCell-paddingX": "12px",
752
- "--TableCell-borderColor": "var(--joy-palette-neutral-outlinedBorder)",
753
- "& thead th": {
754
- bgcolor: "background.surface",
755
- fontFamily: "code",
756
- fontSize: "0.7rem",
757
- letterSpacing: "0.05em",
758
- color: "text.tertiary",
759
- borderBottom: "1px solid",
760
- borderColor: "neutral.outlinedBorder",
761
- position: "sticky",
762
- top: 0,
763
- zIndex: 1,
764
- },
765
- "& tbody tr": {
766
- transition: "background-color 0.2s ease",
767
- },
768
- "& tbody tr:hover": {
769
- bgcolor: colors.hoverBg,
770
- },
771
- }}
772
- >
773
- <thead>
774
- <tr>
775
- <th>NAME</th>
776
- <th style={{ width: "130px" }}>ROLE</th>
777
- <th style={{ width: "90px" }}>STATUS</th>
778
- <th style={{ width: "100px" }}>CAPACITY</th>
779
- <th style={{ width: "100px" }}>MTD USAGE</th>
780
- <th style={{ width: "100px" }}>UPDATED</th>
781
- </tr>
782
- </thead>
783
- <tbody>
784
- {filteredAgents.map((agent) => (
785
- <AgentRow
786
- key={agent.id}
787
- agent={agent}
788
- selected={selectedAgentId === agent.id}
789
- onClick={() => onSelectAgent(selectedAgentId === agent.id ? null : agent.id)}
790
- isDark={isDark}
791
- usage={agentUsageMap.get(agent.id)}
792
- />
793
- ))}
794
- </tbody>
795
- </Table>
796
- </Box>
797
-
798
- {/* Mobile Cards */}
799
- <Box sx={{ display: { xs: "block", md: "none" }, p: 1.5 }}>
800
- {filteredAgents.map((agent) => (
801
- <AgentCard
802
- key={agent.id}
803
- agent={agent}
804
- selected={selectedAgentId === agent.id}
805
- onClick={() => onSelectAgent(selectedAgentId === agent.id ? null : agent.id)}
806
- isDark={isDark}
807
- />
808
- ))}
809
- </Box>
810
- </>
811
- )}
812
- </Box>
813
- </Card>
814
- );
815
- }