@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
@@ -0,0 +1,262 @@
1
+ import {
2
+ createTaskExtended,
3
+ findTaskByAgentMailThread,
4
+ getAgentById,
5
+ getAgentMailInboxMapping,
6
+ getAllAgents,
7
+ } from "../be/db";
8
+ import { resolveTemplate } from "../prompts/resolver";
9
+ import { workflowEventBus } from "../workflows/event-bus";
10
+ // Side-effect import: registers all AgentMail event templates in the in-memory registry
11
+ import "./templates";
12
+ import type { AgentMailMessage, AgentMailWebhookPayload } from "./types";
13
+
14
+ /**
15
+ * Check if an inbox domain is allowed by the filter.
16
+ * Returns true if no filter is set or the inbox domain matches.
17
+ */
18
+ export function isInboxAllowed(inboxId: string, filter: string | undefined): boolean {
19
+ if (!filter) return true;
20
+ const allowedDomains = filter.split(",").map((d) => d.trim().toLowerCase());
21
+ const inboxDomain = inboxId.split("@")[1]?.toLowerCase();
22
+ return !!inboxDomain && allowedDomains.includes(inboxDomain);
23
+ }
24
+
25
+ /**
26
+ * Check if a sender domain is allowed by the filter.
27
+ * Returns true if no filter is set or at least one sender domain matches.
28
+ */
29
+ export function isSenderAllowed(
30
+ from: AgentMailMessage["from_"],
31
+ filter: string | undefined,
32
+ ): boolean {
33
+ if (!filter) return true;
34
+ const allowedDomains = filter.split(",").map((d) => d.trim().toLowerCase());
35
+ const fromAddresses = Array.isArray(from) ? from : [from || ""];
36
+ return fromAddresses.some((addr) => {
37
+ const emailMatch = addr.match(/<([^>]+)>/);
38
+ const email = emailMatch?.[1] ?? addr;
39
+ const domain = email.split("@")[1]?.toLowerCase();
40
+ return !!domain && allowedDomains.includes(domain);
41
+ });
42
+ }
43
+
44
+ // Simple deduplication cache (60 second TTL)
45
+ const processedEvents = new Map<string, number>();
46
+ const EVENT_TTL = 60_000;
47
+
48
+ function isDuplicate(eventKey: string): boolean {
49
+ const now = Date.now();
50
+
51
+ // Clean old entries
52
+ for (const [key, timestamp] of processedEvents) {
53
+ if (now - timestamp > EVENT_TTL) {
54
+ processedEvents.delete(key);
55
+ }
56
+ }
57
+
58
+ if (processedEvents.has(eventKey)) {
59
+ return true;
60
+ }
61
+
62
+ processedEvents.set(eventKey, now);
63
+ return false;
64
+ }
65
+
66
+ /**
67
+ * Find the lead agent to receive AgentMail messages when no inbox mapping exists
68
+ */
69
+ function findLeadAgent() {
70
+ const agents = getAllAgents();
71
+ const onlineLead = agents.find((a) => a.isLead && a.status !== "offline");
72
+ if (onlineLead) return onlineLead;
73
+ return agents.find((a) => a.isLead) ?? null;
74
+ }
75
+
76
+ /**
77
+ * Handle message.received webhook event
78
+ */
79
+ export async function handleMessageReceived(
80
+ payload: AgentMailWebhookPayload,
81
+ ): Promise<{ created: boolean; taskId?: string }> {
82
+ const message = payload.message;
83
+ if (!message) {
84
+ console.log("[AgentMail] message.received event missing message payload");
85
+ return { created: false };
86
+ }
87
+
88
+ // Deduplicate using event_id
89
+ if (isDuplicate(`agentmail:${payload.event_id}`)) {
90
+ console.log(`[AgentMail] Duplicate event ${payload.event_id}, skipping`);
91
+ return { created: false };
92
+ }
93
+
94
+ const { inbox_id, thread_id, message_id } = message;
95
+ const from =
96
+ (Array.isArray(message.from_) ? message.from_.join(", ") : message.from_) || "unknown";
97
+ const subject = message.subject || "(no subject)";
98
+ const body = message.text || message.html || "";
99
+ const preview = body.length > 500 ? `${body.substring(0, 500)}...` : body;
100
+
101
+ // Emit workflow trigger event
102
+ workflowEventBus.emit("agentmail.message.received", {
103
+ inboxId: inbox_id,
104
+ from,
105
+ subject,
106
+ body: preview,
107
+ threadId: thread_id,
108
+ messageId: message_id,
109
+ });
110
+
111
+ // Check for thread continuity - find existing task for this thread
112
+ const existingTask = findTaskByAgentMailThread(thread_id);
113
+ if (existingTask) {
114
+ // Create a follow-up task with parentTaskId to continue the session
115
+ const followupResult = resolveTemplate("agentmail.email.followup", {
116
+ from,
117
+ subject,
118
+ inbox_id,
119
+ thread_id,
120
+ preview,
121
+ });
122
+
123
+ if (followupResult.skipped) {
124
+ return { created: false };
125
+ }
126
+
127
+ const task = createTaskExtended(followupResult.text, {
128
+ agentId: existingTask.agentId,
129
+ source: "agentmail",
130
+ taskType: "agentmail-reply",
131
+ agentmailInboxId: inbox_id,
132
+ agentmailMessageId: message_id,
133
+ agentmailThreadId: thread_id,
134
+ parentTaskId: existingTask.id,
135
+ });
136
+
137
+ console.log(
138
+ `[AgentMail] Created follow-up task ${task.id} for thread ${thread_id} (parent: ${existingTask.id})`,
139
+ );
140
+ return { created: true, taskId: task.id };
141
+ }
142
+
143
+ // Look up agent from inbox mapping
144
+ const mapping = getAgentMailInboxMapping(inbox_id);
145
+
146
+ if (mapping) {
147
+ const agent = getAgentById(mapping.agentId);
148
+ if (agent) {
149
+ if (agent.isLead) {
150
+ // Route to lead as task
151
+ const leadResult = resolveTemplate("agentmail.email.mapped_lead", {
152
+ from,
153
+ subject,
154
+ inbox_id,
155
+ thread_id,
156
+ message_id,
157
+ preview,
158
+ });
159
+
160
+ if (leadResult.skipped) {
161
+ return { created: false };
162
+ }
163
+
164
+ const task = createTaskExtended(leadResult.text, {
165
+ agentId: agent.id,
166
+ source: "agentmail",
167
+ taskType: "agentmail-message",
168
+ agentmailInboxId: inbox_id,
169
+ agentmailMessageId: message_id,
170
+ agentmailThreadId: thread_id,
171
+ });
172
+
173
+ console.log(
174
+ `[AgentMail] Created task ${task.id} for lead ${agent.name} (inbox: ${inbox_id})`,
175
+ );
176
+ return { created: true, taskId: task.id };
177
+ }
178
+
179
+ // Route to worker as task
180
+ const workerResult = resolveTemplate("agentmail.email.mapped_worker", {
181
+ from,
182
+ subject,
183
+ inbox_id,
184
+ thread_id,
185
+ preview,
186
+ });
187
+
188
+ if (workerResult.skipped) {
189
+ return { created: false };
190
+ }
191
+
192
+ const task = createTaskExtended(workerResult.text, {
193
+ agentId: agent.id,
194
+ source: "agentmail",
195
+ taskType: "agentmail-message",
196
+ agentmailInboxId: inbox_id,
197
+ agentmailMessageId: message_id,
198
+ agentmailThreadId: thread_id,
199
+ });
200
+
201
+ console.log(
202
+ `[AgentMail] Created task ${task.id} for worker ${agent.name} (inbox: ${inbox_id})`,
203
+ );
204
+ return { created: true, taskId: task.id };
205
+ }
206
+ }
207
+
208
+ // No mapping found - route to lead as task
209
+ const lead = findLeadAgent();
210
+ if (lead) {
211
+ const unmappedResult = resolveTemplate("agentmail.email.unmapped", {
212
+ from,
213
+ subject,
214
+ inbox_id,
215
+ thread_id,
216
+ message_id,
217
+ preview,
218
+ });
219
+
220
+ if (unmappedResult.skipped) {
221
+ return { created: false };
222
+ }
223
+
224
+ const task = createTaskExtended(unmappedResult.text, {
225
+ agentId: lead.id,
226
+ source: "agentmail",
227
+ taskType: "agentmail-message",
228
+ agentmailInboxId: inbox_id,
229
+ agentmailMessageId: message_id,
230
+ agentmailThreadId: thread_id,
231
+ });
232
+
233
+ console.log(
234
+ `[AgentMail] Created task ${task.id} for lead ${lead.name} (unmapped inbox: ${inbox_id})`,
235
+ );
236
+ return { created: true, taskId: task.id };
237
+ }
238
+
239
+ // No lead available - create unassigned task
240
+ const noAgentResult = resolveTemplate("agentmail.email.no_agent", {
241
+ from,
242
+ subject,
243
+ inbox_id,
244
+ thread_id,
245
+ preview,
246
+ });
247
+
248
+ if (noAgentResult.skipped) {
249
+ return { created: false };
250
+ }
251
+
252
+ const task = createTaskExtended(noAgentResult.text, {
253
+ source: "agentmail",
254
+ taskType: "agentmail-message",
255
+ agentmailInboxId: inbox_id,
256
+ agentmailMessageId: message_id,
257
+ agentmailThreadId: thread_id,
258
+ });
259
+
260
+ console.log(`[AgentMail] Created unassigned task ${task.id} (no lead or mapping available)`);
261
+ return { created: true, taskId: task.id };
262
+ }
@@ -0,0 +1,9 @@
1
+ // AgentMail Integration
2
+ export { initAgentMail, isAgentMailEnabled, resetAgentMail, verifyAgentMailWebhook } from "./app";
3
+ export { handleMessageReceived, isInboxAllowed, isSenderAllowed } from "./handlers";
4
+ export type {
5
+ AgentMailAttachment,
6
+ AgentMailEventType,
7
+ AgentMailMessage,
8
+ AgentMailWebhookPayload,
9
+ } from "./types";
@@ -0,0 +1,111 @@
1
+ /**
2
+ * AgentMail event prompt template definitions.
3
+ *
4
+ * Each template is registered at module load time via registerTemplate().
5
+ * Handlers import this module for the side-effect of registration.
6
+ */
7
+
8
+ import { registerTemplate } from "../prompts/registry";
9
+
10
+ // ============================================================================
11
+ // Email events
12
+ // ============================================================================
13
+
14
+ registerTemplate({
15
+ eventType: "agentmail.email.followup",
16
+ header: "[AgentMail] Follow-up email in thread",
17
+ defaultBody: `From: {{from}}
18
+ Subject: {{subject}}
19
+ Inbox: {{inbox_id}}
20
+ Thread: {{thread_id}}
21
+
22
+ {{preview}}`,
23
+ variables: [
24
+ { name: "from", description: "Sender address(es)" },
25
+ { name: "subject", description: "Email subject" },
26
+ { name: "inbox_id", description: "AgentMail inbox ID" },
27
+ { name: "thread_id", description: "AgentMail thread ID" },
28
+ { name: "preview", description: "Email body preview (truncated to 500 chars)" },
29
+ ],
30
+ category: "event",
31
+ });
32
+
33
+ registerTemplate({
34
+ eventType: "agentmail.email.mapped_lead",
35
+ header: "[AgentMail] New email received",
36
+ defaultBody: `From: {{from}}
37
+ Subject: {{subject}}
38
+ Inbox: {{inbox_id}}
39
+ Thread: {{thread_id}}
40
+ Message: {{message_id}}
41
+
42
+ {{preview}}`,
43
+ variables: [
44
+ { name: "from", description: "Sender address(es)" },
45
+ { name: "subject", description: "Email subject" },
46
+ { name: "inbox_id", description: "AgentMail inbox ID" },
47
+ { name: "thread_id", description: "AgentMail thread ID" },
48
+ { name: "message_id", description: "AgentMail message ID" },
49
+ { name: "preview", description: "Email body preview (truncated to 500 chars)" },
50
+ ],
51
+ category: "event",
52
+ });
53
+
54
+ registerTemplate({
55
+ eventType: "agentmail.email.mapped_worker",
56
+ header: "[AgentMail] New email received",
57
+ defaultBody: `From: {{from}}
58
+ Subject: {{subject}}
59
+ Inbox: {{inbox_id}}
60
+ Thread: {{thread_id}}
61
+
62
+ {{preview}}`,
63
+ variables: [
64
+ { name: "from", description: "Sender address(es)" },
65
+ { name: "subject", description: "Email subject" },
66
+ { name: "inbox_id", description: "AgentMail inbox ID" },
67
+ { name: "thread_id", description: "AgentMail thread ID" },
68
+ { name: "preview", description: "Email body preview (truncated to 500 chars)" },
69
+ ],
70
+ category: "event",
71
+ });
72
+
73
+ registerTemplate({
74
+ eventType: "agentmail.email.unmapped",
75
+ header: "[AgentMail] New email received (unmapped inbox)",
76
+ defaultBody: `From: {{from}}
77
+ Subject: {{subject}}
78
+ Inbox: {{inbox_id}}
79
+ Thread: {{thread_id}}
80
+ Message: {{message_id}}
81
+
82
+ {{preview}}`,
83
+ variables: [
84
+ { name: "from", description: "Sender address(es)" },
85
+ { name: "subject", description: "Email subject" },
86
+ { name: "inbox_id", description: "AgentMail inbox ID" },
87
+ { name: "thread_id", description: "AgentMail thread ID" },
88
+ { name: "message_id", description: "AgentMail message ID" },
89
+ { name: "preview", description: "Email body preview (truncated to 500 chars)" },
90
+ ],
91
+ category: "event",
92
+ });
93
+
94
+ registerTemplate({
95
+ eventType: "agentmail.email.no_agent",
96
+ header: "[AgentMail] New email received (no agent available)",
97
+ defaultBody: `From: {{from}}
98
+ Subject: {{subject}}
99
+ Inbox: {{inbox_id}}
100
+ Thread: {{thread_id}}
101
+
102
+ {{preview}}`,
103
+ variables: [
104
+ { name: "from", description: "Sender address(es)" },
105
+ { name: "subject", description: "Email subject" },
106
+ { name: "inbox_id", description: "AgentMail inbox ID" },
107
+ { name: "thread_id", description: "AgentMail thread ID" },
108
+ { name: "preview", description: "Email body preview (truncated to 500 chars)" },
109
+ ],
110
+ category: "event",
111
+ });
@@ -0,0 +1,51 @@
1
+ /**
2
+ * AgentMail webhook payload types
3
+ * Based on Svix delivery format used by AgentMail
4
+ */
5
+
6
+ export interface AgentMailAttachment {
7
+ attachment_id: string;
8
+ filename: string;
9
+ content_type: string;
10
+ size: number;
11
+ inline: boolean;
12
+ }
13
+
14
+ export interface AgentMailMessage {
15
+ message_id: string;
16
+ thread_id: string;
17
+ inbox_id: string;
18
+ organization_id: string;
19
+ from_: string | string[];
20
+ to: string[];
21
+ cc: string[];
22
+ bcc: string[];
23
+ reply_to: string[];
24
+ subject: string;
25
+ preview: string;
26
+ text: string | null;
27
+ html: string | null;
28
+ labels: string[];
29
+ attachments: AgentMailAttachment[];
30
+ in_reply_to: string | null;
31
+ references: string[];
32
+ timestamp: string;
33
+ created_at: string;
34
+ updated_at: string;
35
+ }
36
+
37
+ export interface AgentMailWebhookPayload {
38
+ type: "event";
39
+ event_type: AgentMailEventType;
40
+ event_id: string;
41
+ message?: AgentMailMessage;
42
+ }
43
+
44
+ export type AgentMailEventType =
45
+ | "message.received"
46
+ | "message.sent"
47
+ | "message.delivered"
48
+ | "message.bounced"
49
+ | "message.complained"
50
+ | "message.rejected"
51
+ | "domain.verified";
@@ -0,0 +1,30 @@
1
+ // This is a string template that gets served as JavaScript to the browser
2
+ export const BROWSER_SDK_JS = `
3
+ class SwarmSDK {
4
+ constructor() {
5
+ this._configPromise = fetch('/@swarm/config').then(r => r.json());
6
+ }
7
+
8
+ async createTask(opts) { return this._post('/@swarm/api/tasks', opts); }
9
+ async getTasks(filters) { return this._get('/@swarm/api/tasks?' + new URLSearchParams(filters)); }
10
+ async getTaskDetails(id) { return this._get('/@swarm/api/tasks/' + id); }
11
+ async storeProgress(taskId, data) { return this._post('/@swarm/api/tasks/' + taskId + '/progress', data); }
12
+ async postMessage(opts) { return this._post('/@swarm/api/messages', opts); }
13
+ async readMessages(opts) { return this._get('/@swarm/api/messages?' + new URLSearchParams(opts)); }
14
+ async getSwarm() { return this._get('/@swarm/api/agents'); }
15
+ async listServices() { return this._get('/@swarm/api/services'); }
16
+ async listEpics(opts) { return this._get('/@swarm/api/epics?' + new URLSearchParams(opts || {})); }
17
+ async slackReply(opts) { return this._post('/@swarm/api/slack/reply', opts); }
18
+
19
+ async _post(url, body) {
20
+ const res = await fetch(url, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(body) });
21
+ return res.json();
22
+ }
23
+ async _get(url) {
24
+ const res = await fetch(url);
25
+ return res.json();
26
+ }
27
+ }
28
+
29
+ window.SwarmSDK = SwarmSDK;
30
+ `;
@@ -0,0 +1,2 @@
1
+ export type { ArtifactServer, ArtifactServerOptions } from "./server";
2
+ export { createArtifactServer } from "./server";
@@ -0,0 +1,20 @@
1
+ declare module "@desplega.ai/localtunnel" {
2
+ interface TunnelOptions {
3
+ port: number;
4
+ subdomain?: string;
5
+ host?: string;
6
+ auth?: string;
7
+ username?: string;
8
+ local_host?: string;
9
+ }
10
+
11
+ interface Tunnel {
12
+ url: string;
13
+ close(): void;
14
+ on(event: "close", listener: () => void): void;
15
+ on(event: "error", listener: (err: Error) => void): void;
16
+ }
17
+
18
+ function localtunnel(opts: TunnelOptions): Promise<Tunnel>;
19
+ export default localtunnel;
20
+ }
@@ -0,0 +1,12 @@
1
+ import { createServer } from "node:net";
2
+
3
+ export async function getAvailablePort(): Promise<number> {
4
+ return new Promise((resolve, reject) => {
5
+ const server = createServer();
6
+ server.listen(0, () => {
7
+ const { port } = server.address() as { port: number };
8
+ server.close(() => resolve(port));
9
+ });
10
+ server.on("error", reject);
11
+ });
12
+ }
@@ -0,0 +1,156 @@
1
+ import { Hono } from "hono";
2
+ import { serveStatic } from "hono/bun";
3
+ import { BROWSER_SDK_JS } from "./browser-sdk";
4
+ import { getAvailablePort } from "./port";
5
+ import { createTunnel } from "./tunnel";
6
+
7
+ export interface ArtifactServerOptions {
8
+ name: string;
9
+ static?: string;
10
+ app?: Hono;
11
+ port?: number;
12
+ auth?: boolean;
13
+ subdomain?: string;
14
+ }
15
+
16
+ export interface ArtifactServer {
17
+ start(): Promise<void>;
18
+ stop(): Promise<void>;
19
+ url: string;
20
+ port: number;
21
+ tunnel: ReturnType<typeof createTunnel> extends Promise<infer T> ? T | null : never;
22
+ }
23
+
24
+ export function createArtifactServer(opts: ArtifactServerOptions): ArtifactServer {
25
+ const agentId = process.env.AGENT_ID || "unknown";
26
+ const apiKey = process.env.API_KEY || "";
27
+ const mcpBaseUrl = process.env.MCP_BASE_URL || `http://localhost:${process.env.PORT || "3013"}`;
28
+
29
+ const app = new Hono();
30
+
31
+ // Inject swarm middleware
32
+ app.get("/@swarm/sdk.js", (c) => {
33
+ c.header("Content-Type", "application/javascript");
34
+ return c.body(BROWSER_SDK_JS);
35
+ });
36
+
37
+ app.get("/@swarm/config", (c) => {
38
+ return c.json({ agentId, artifactName: opts.name });
39
+ });
40
+
41
+ // API proxy — forwards /@swarm/api/* to MCP server
42
+ app.all("/@swarm/api/*", async (c) => {
43
+ const path = c.req.path.replace("/@swarm/api", "/api");
44
+ const targetUrl = `${mcpBaseUrl}${path}`;
45
+ const headers: Record<string, string> = {
46
+ Authorization: `Bearer ${apiKey}`,
47
+ "X-Agent-ID": agentId,
48
+ };
49
+ if (c.req.method !== "GET") {
50
+ headers["Content-Type"] = "application/json";
51
+ }
52
+ try {
53
+ const res = await fetch(targetUrl, {
54
+ method: c.req.method,
55
+ headers,
56
+ body: c.req.method !== "GET" ? await c.req.text() : undefined,
57
+ });
58
+ return new Response(res.body, {
59
+ status: res.status,
60
+ headers: {
61
+ "Access-Control-Allow-Origin": "*",
62
+ "Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
63
+ "Access-Control-Allow-Headers": "Content-Type",
64
+ },
65
+ });
66
+ } catch (e) {
67
+ return c.json({ error: "Proxy error", message: String(e) }, 502);
68
+ }
69
+ });
70
+
71
+ // CORS preflight
72
+ app.options("/@swarm/api/*", (_c) => {
73
+ return new Response(null, {
74
+ status: 204,
75
+ headers: {
76
+ "Access-Control-Allow-Origin": "*",
77
+ "Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
78
+ "Access-Control-Allow-Headers": "Content-Type",
79
+ },
80
+ });
81
+ });
82
+
83
+ // User app or static serving
84
+ if (opts.app) {
85
+ app.route("/", opts.app);
86
+ } else if (opts.static) {
87
+ app.use("/*", serveStatic({ root: opts.static }));
88
+ }
89
+
90
+ let server: ReturnType<typeof Bun.serve> | null = null;
91
+ // biome-ignore lint/suspicious/noExplicitAny: localtunnel tunnel type
92
+ let tunnel: any = null;
93
+ let actualPort = 0;
94
+
95
+ const artifact: ArtifactServer = {
96
+ url: "",
97
+ port: 0,
98
+ tunnel: null,
99
+
100
+ async start() {
101
+ actualPort = opts.port || (await getAvailablePort());
102
+ server = Bun.serve({ port: actualPort, fetch: app.fetch });
103
+ artifact.port = actualPort;
104
+
105
+ const subdomain = opts.subdomain || `${agentId}-${opts.name}`;
106
+ const authPassword = opts.auth === false ? undefined : apiKey;
107
+
108
+ tunnel = await createTunnel({
109
+ port: actualPort,
110
+ subdomain,
111
+ auth: authPassword,
112
+ });
113
+
114
+ artifact.url = tunnel.url;
115
+ artifact.tunnel = tunnel;
116
+
117
+ // Register in service registry
118
+ try {
119
+ await fetch(`${mcpBaseUrl}/api/services`, {
120
+ method: "POST",
121
+ headers: {
122
+ Authorization: `Bearer ${apiKey}`,
123
+ "X-Agent-ID": agentId,
124
+ "Content-Type": "application/json",
125
+ },
126
+ body: JSON.stringify({
127
+ script: `artifact-${opts.name}`,
128
+ metadata: {
129
+ type: "artifact",
130
+ artifactName: opts.name,
131
+ port: actualPort,
132
+ publicUrl: tunnel.url,
133
+ },
134
+ }),
135
+ });
136
+ } catch (e) {
137
+ console.warn("Failed to register artifact in service registry:", e);
138
+ }
139
+
140
+ console.log(`Artifact "${opts.name}" live at ${tunnel.url} (port ${actualPort})`);
141
+ },
142
+
143
+ async stop() {
144
+ if (tunnel) {
145
+ tunnel.close();
146
+ tunnel = null;
147
+ }
148
+ if (server) {
149
+ server.stop();
150
+ server = null;
151
+ }
152
+ },
153
+ };
154
+
155
+ return artifact;
156
+ }
@@ -0,0 +1,19 @@
1
+ import localtunnel from "@desplega.ai/localtunnel";
2
+
3
+ interface TunnelOptions {
4
+ port: number;
5
+ subdomain: string;
6
+ auth?: string;
7
+ username?: string;
8
+ }
9
+
10
+ export async function createTunnel(opts: TunnelOptions) {
11
+ const tunnel = await localtunnel({
12
+ port: opts.port,
13
+ subdomain: opts.subdomain,
14
+ auth: opts.auth,
15
+ username: opts.username || "hi", // default "hi" for MVP (custom username deferred)
16
+ // host defaults to lt.desplega.ai in our fork
17
+ });
18
+ return tunnel;
19
+ }