@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,86 @@
1
+ import { Database } from "bun:sqlite";
2
+ import { afterEach, describe, expect, test } from "bun:test";
3
+ import { unlink } from "node:fs/promises";
4
+ import { closeDb, initDb } from "../be/db";
5
+
6
+ const INCOMPLETE_DB_PATH = "./test-migration-incomplete.sqlite";
7
+ const FRESH_DB_PATH = "./test-migration-fresh.sqlite";
8
+
9
+ async function removeDbFiles(dbPath: string): Promise<void> {
10
+ for (const suffix of ["", "-wal", "-shm"]) {
11
+ try {
12
+ await unlink(dbPath + suffix);
13
+ } catch (error) {
14
+ if ((error as NodeJS.ErrnoException).code !== "ENOENT") {
15
+ throw error;
16
+ }
17
+ }
18
+ }
19
+ }
20
+
21
+ afterEach(async () => {
22
+ closeDb();
23
+ await removeDbFiles(INCOMPLETE_DB_PATH);
24
+ await removeDbFiles(FRESH_DB_PATH);
25
+ });
26
+
27
+ describe("migration regressions", () => {
28
+ test("incomplete existing DB runs 001_initial instead of blind bootstrap", () => {
29
+ const now = new Date().toISOString();
30
+ const legacyDb = new Database(INCOMPLETE_DB_PATH, { create: true });
31
+ legacyDb.run(`
32
+ CREATE TABLE agents (
33
+ id TEXT PRIMARY KEY,
34
+ name TEXT NOT NULL,
35
+ isLead INTEGER NOT NULL DEFAULT 0,
36
+ status TEXT NOT NULL,
37
+ maxTasks INTEGER DEFAULT 1,
38
+ emptyPollCount INTEGER DEFAULT 0,
39
+ createdAt TEXT NOT NULL,
40
+ lastUpdatedAt TEXT NOT NULL
41
+ )
42
+ `);
43
+ legacyDb.run(
44
+ "INSERT INTO agents (id, name, isLead, status, createdAt, lastUpdatedAt) VALUES (?, ?, ?, ?, ?, ?)",
45
+ [crypto.randomUUID(), "legacy", 0, "idle", now, now],
46
+ );
47
+ legacyDb.close();
48
+
49
+ const database = initDb(INCOMPLETE_DB_PATH);
50
+
51
+ const channelsTable = database
52
+ .prepare<{ name: string }, []>(
53
+ "SELECT name FROM sqlite_master WHERE type='table' AND name='channels'",
54
+ )
55
+ .get();
56
+ expect(channelsTable?.name).toBe("channels");
57
+
58
+ const generalChannel = database
59
+ .prepare<{ id: string }, []>("SELECT id FROM channels WHERE name = 'general'")
60
+ .get();
61
+ expect(generalChannel?.id).toBe("00000000-0000-4000-8000-000000000001");
62
+
63
+ const columns = database
64
+ .prepare<{ name: string }, []>("PRAGMA table_info(agents)")
65
+ .all()
66
+ .map((column) => column.name);
67
+ expect(columns).toContain("soulMd");
68
+ expect(columns).toContain("identityMd");
69
+ expect(columns).toContain("toolsMd");
70
+ expect(columns).toContain("claudeMd");
71
+ expect(columns).toContain("setupScript");
72
+ });
73
+
74
+ test("fresh DB preserves source CHECK constraint on agent_tasks", () => {
75
+ const database = initDb(FRESH_DB_PATH);
76
+ const now = new Date().toISOString();
77
+
78
+ expect(() => {
79
+ database.run(
80
+ `INSERT INTO agent_tasks (id, task, status, source, createdAt, lastUpdatedAt)
81
+ VALUES (?, ?, ?, ?, ?, ?)`,
82
+ [crypto.randomUUID(), "invalid source", "pending", "not-valid", now, now],
83
+ );
84
+ }).toThrow();
85
+ });
86
+ });
@@ -0,0 +1,338 @@
1
+ import { afterAll, beforeAll, describe, expect, test } from "bun:test";
2
+ import { unlinkSync } from "node:fs";
3
+ import {
4
+ closeDb,
5
+ createAgent,
6
+ createScheduledTask,
7
+ createTaskExtended,
8
+ getResolvedConfig,
9
+ getScheduledTaskById,
10
+ getTaskById,
11
+ initDb,
12
+ updateScheduledTask,
13
+ upsertSwarmConfig,
14
+ } from "../be/db";
15
+ import { runScheduleNow } from "../scheduler";
16
+
17
+ const TEST_DB_PATH = "./test-model-control.sqlite";
18
+
19
+ beforeAll(() => {
20
+ initDb(TEST_DB_PATH);
21
+ });
22
+
23
+ afterAll(() => {
24
+ closeDb();
25
+ try {
26
+ unlinkSync(TEST_DB_PATH);
27
+ unlinkSync(`${TEST_DB_PATH}-wal`);
28
+ unlinkSync(`${TEST_DB_PATH}-shm`);
29
+ } catch {
30
+ // ignore if files don't exist
31
+ }
32
+ });
33
+
34
+ describe("Model Control - Task Creation", () => {
35
+ test("should store model when creating a task with model='sonnet'", () => {
36
+ const task = createTaskExtended("Test task with sonnet", { model: "sonnet" });
37
+ expect(task.model).toBe("sonnet");
38
+
39
+ const retrieved = getTaskById(task.id);
40
+ expect(retrieved?.model).toBe("sonnet");
41
+ });
42
+
43
+ test("should store model when creating a task with model='haiku'", () => {
44
+ const task = createTaskExtended("Test task with haiku", { model: "haiku" });
45
+ expect(task.model).toBe("haiku");
46
+ });
47
+
48
+ test("should store model when creating a task with model='opus'", () => {
49
+ const task = createTaskExtended("Test task with opus", { model: "opus" });
50
+ expect(task.model).toBe("opus");
51
+ });
52
+
53
+ test("should default model to undefined when not specified", () => {
54
+ const task = createTaskExtended("Test task without model");
55
+ expect(task.model).toBeUndefined();
56
+ });
57
+
58
+ test("should preserve model alongside other task options", () => {
59
+ const agent = createAgent({ name: "model-test-agent", isLead: false, status: "idle" });
60
+
61
+ const task = createTaskExtended("Task with model and options", {
62
+ model: "haiku",
63
+ agentId: agent.id,
64
+ priority: 80,
65
+ taskType: "test",
66
+ tags: ["model-test"],
67
+ });
68
+
69
+ expect(task.model).toBe("haiku");
70
+ expect(task.agentId).toBe(agent.id);
71
+ expect(task.priority).toBe(80);
72
+ expect(task.taskType).toBe("test");
73
+ expect(task.tags).toContain("model-test");
74
+ });
75
+
76
+ test("should store model on offered tasks", () => {
77
+ const agent = createAgent({ name: "offer-model-agent", isLead: false, status: "idle" });
78
+
79
+ const task = createTaskExtended("Offered task with model", {
80
+ model: "sonnet",
81
+ offeredTo: agent.id,
82
+ });
83
+
84
+ expect(task.model).toBe("sonnet");
85
+ expect(task.status).toBe("offered");
86
+ });
87
+ });
88
+
89
+ describe("Model Control - Schedule Creation", () => {
90
+ test("should store model on scheduled task creation", () => {
91
+ const schedule = createScheduledTask({
92
+ name: "model-schedule-sonnet",
93
+ intervalMs: 60000,
94
+ taskTemplate: "Scheduled with sonnet",
95
+ model: "sonnet",
96
+ });
97
+
98
+ expect(schedule.model).toBe("sonnet");
99
+
100
+ const retrieved = getScheduledTaskById(schedule.id);
101
+ expect(retrieved?.model).toBe("sonnet");
102
+ });
103
+
104
+ test("should store all valid model values on schedules", () => {
105
+ for (const model of ["haiku", "sonnet", "opus"] as const) {
106
+ const schedule = createScheduledTask({
107
+ name: `model-schedule-all-${model}-${Date.now()}`,
108
+ intervalMs: 60000,
109
+ taskTemplate: `Scheduled with ${model}`,
110
+ model,
111
+ });
112
+
113
+ expect(schedule.model).toBe(model);
114
+ }
115
+ });
116
+
117
+ test("should default model to undefined when not specified on schedule", () => {
118
+ const schedule = createScheduledTask({
119
+ name: "model-schedule-default",
120
+ intervalMs: 60000,
121
+ taskTemplate: "Scheduled without model",
122
+ });
123
+
124
+ expect(schedule.model).toBeUndefined();
125
+ });
126
+ });
127
+
128
+ describe("Model Control - Schedule Update", () => {
129
+ test("should update model on existing schedule", () => {
130
+ const schedule = createScheduledTask({
131
+ name: "model-update-test",
132
+ intervalMs: 60000,
133
+ taskTemplate: "Update model test",
134
+ model: "opus",
135
+ });
136
+
137
+ expect(schedule.model).toBe("opus");
138
+
139
+ const updated = updateScheduledTask(schedule.id, { model: "haiku" });
140
+ expect(updated?.model).toBe("haiku");
141
+
142
+ const retrieved = getScheduledTaskById(schedule.id);
143
+ expect(retrieved?.model).toBe("haiku");
144
+ });
145
+
146
+ test("should clear model by setting to null", () => {
147
+ const schedule = createScheduledTask({
148
+ name: "model-clear-test",
149
+ intervalMs: 60000,
150
+ taskTemplate: "Clear model test",
151
+ model: "sonnet",
152
+ });
153
+
154
+ expect(schedule.model).toBe("sonnet");
155
+
156
+ const updated = updateScheduledTask(schedule.id, { model: null });
157
+ expect(updated?.model).toBeUndefined();
158
+ });
159
+
160
+ test("should preserve model when updating other fields", () => {
161
+ const schedule = createScheduledTask({
162
+ name: "model-preserve-test",
163
+ intervalMs: 60000,
164
+ taskTemplate: "Preserve model test",
165
+ model: "haiku",
166
+ });
167
+
168
+ const updated = updateScheduledTask(schedule.id, { priority: 90 });
169
+ expect(updated?.model).toBe("haiku");
170
+ expect(updated?.priority).toBe(90);
171
+ });
172
+ });
173
+
174
+ describe("Model Control - Schedule to Task Propagation", () => {
175
+ test("should propagate model from schedule to task on manual run", async () => {
176
+ const schedule = createScheduledTask({
177
+ name: "model-propagate-manual",
178
+ intervalMs: 60000,
179
+ taskTemplate: "Propagated model task (manual)",
180
+ model: "haiku",
181
+ enabled: true,
182
+ });
183
+
184
+ await runScheduleNow(schedule.id);
185
+
186
+ // Find the created task by its template text
187
+ const { getDb } = await import("../be/db");
188
+ const row = getDb()
189
+ .query("SELECT id FROM agent_tasks WHERE task = ? ORDER BY createdAt DESC LIMIT 1")
190
+ .get("Propagated model task (manual)") as { id: string } | null;
191
+
192
+ expect(row).not.toBeNull();
193
+ const task = getTaskById(row!.id);
194
+ expect(task?.model).toBe("haiku");
195
+ });
196
+
197
+ test("should create task without model when schedule has no model", async () => {
198
+ const schedule = createScheduledTask({
199
+ name: "model-propagate-none",
200
+ intervalMs: 60000,
201
+ taskTemplate: "Propagated no-model task",
202
+ enabled: true,
203
+ });
204
+
205
+ await runScheduleNow(schedule.id);
206
+
207
+ const { getDb } = await import("../be/db");
208
+ const row = getDb()
209
+ .query("SELECT id FROM agent_tasks WHERE task = ? ORDER BY createdAt DESC LIMIT 1")
210
+ .get("Propagated no-model task") as { id: string } | null;
211
+
212
+ expect(row).not.toBeNull();
213
+ const task = getTaskById(row!.id);
214
+ expect(task?.model).toBeUndefined();
215
+ });
216
+ });
217
+
218
+ describe("Model Control - Config MODEL_OVERRIDE Resolution", () => {
219
+ test("should resolve global MODEL_OVERRIDE config", () => {
220
+ upsertSwarmConfig({
221
+ scope: "global",
222
+ key: "MODEL_OVERRIDE",
223
+ value: "sonnet",
224
+ });
225
+
226
+ const configs = getResolvedConfig();
227
+ const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
228
+ expect(modelOverride).toBeDefined();
229
+ expect(modelOverride?.value).toBe("sonnet");
230
+ });
231
+
232
+ test("agent-scoped MODEL_OVERRIDE should override global", () => {
233
+ const agent = createAgent({ name: "config-agent", isLead: false, status: "idle" });
234
+
235
+ upsertSwarmConfig({
236
+ scope: "global",
237
+ key: "MODEL_OVERRIDE",
238
+ value: "opus",
239
+ });
240
+
241
+ upsertSwarmConfig({
242
+ scope: "agent",
243
+ scopeId: agent.id,
244
+ key: "MODEL_OVERRIDE",
245
+ value: "haiku",
246
+ });
247
+
248
+ const configs = getResolvedConfig(agent.id);
249
+ const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
250
+ expect(modelOverride?.value).toBe("haiku");
251
+ expect(modelOverride?.scope).toBe("agent");
252
+ });
253
+
254
+ test("should fallback to global when no agent-scoped config exists", () => {
255
+ const agent = createAgent({ name: "fallback-agent", isLead: false, status: "idle" });
256
+
257
+ upsertSwarmConfig({
258
+ scope: "global",
259
+ key: "MODEL_OVERRIDE",
260
+ value: "sonnet",
261
+ });
262
+
263
+ const configs = getResolvedConfig(agent.id);
264
+ const modelOverride = configs.find((c) => c.key === "MODEL_OVERRIDE");
265
+ expect(modelOverride?.value).toBe("sonnet");
266
+ expect(modelOverride?.scope).toBe("global");
267
+ });
268
+ });
269
+
270
+ describe("Model Control - Priority Resolution Logic", () => {
271
+ // The runner resolves model as: task.model || freshEnv.MODEL_OVERRIDE || "opus"
272
+ // We test the same logic pattern here to ensure correctness
273
+
274
+ function resolveModel(taskModel?: string, configOverride?: string): string {
275
+ return taskModel || configOverride || "opus";
276
+ }
277
+
278
+ test("task.model takes highest priority", () => {
279
+ expect(resolveModel("haiku", "sonnet")).toBe("haiku");
280
+ });
281
+
282
+ test("config MODEL_OVERRIDE is used when task has no model", () => {
283
+ expect(resolveModel(undefined, "sonnet")).toBe("sonnet");
284
+ });
285
+
286
+ test("defaults to 'opus' when no task model and no config override", () => {
287
+ expect(resolveModel(undefined, undefined)).toBe("opus");
288
+ });
289
+
290
+ test("empty string task model falls through to config", () => {
291
+ expect(resolveModel("", "sonnet")).toBe("sonnet");
292
+ });
293
+
294
+ test("empty string config override falls through to default", () => {
295
+ expect(resolveModel(undefined, "")).toBe("opus");
296
+ });
297
+
298
+ test("all three levels specified — task wins", () => {
299
+ expect(resolveModel("haiku", "sonnet")).toBe("haiku");
300
+ // "opus" is the hardcoded default, tested implicitly
301
+ });
302
+ });
303
+
304
+ describe("Model Control - Zod Validation Schema", () => {
305
+ // The MCP tools use z.enum(["haiku", "sonnet", "opus"]) for validation.
306
+ // We test the schema directly to ensure only valid values are accepted.
307
+
308
+ test("should accept valid model values", async () => {
309
+ const { z } = await import("zod");
310
+ const modelSchema = z.enum(["haiku", "sonnet", "opus"]).optional();
311
+
312
+ expect(modelSchema.parse("haiku")).toBe("haiku");
313
+ expect(modelSchema.parse("sonnet")).toBe("sonnet");
314
+ expect(modelSchema.parse("opus")).toBe("opus");
315
+ expect(modelSchema.parse(undefined)).toBeUndefined();
316
+ });
317
+
318
+ test("should reject invalid model values", async () => {
319
+ const { z } = await import("zod");
320
+ const modelSchema = z.enum(["haiku", "sonnet", "opus"]).optional();
321
+
322
+ expect(() => modelSchema.parse("gpt-4")).toThrow();
323
+ expect(() => modelSchema.parse("claude")).toThrow();
324
+ expect(() => modelSchema.parse("turbo")).toThrow();
325
+ expect(() => modelSchema.parse("")).toThrow();
326
+ expect(() => modelSchema.parse(123)).toThrow();
327
+ expect(() => modelSchema.parse(null)).toThrow();
328
+ });
329
+
330
+ test("nullable model schema (update-schedule) should accept null", async () => {
331
+ const { z } = await import("zod");
332
+ const modelSchema = z.enum(["haiku", "sonnet", "opus"]).nullable().optional();
333
+
334
+ expect(modelSchema.parse(null)).toBeNull();
335
+ expect(modelSchema.parse("haiku")).toBe("haiku");
336
+ expect(modelSchema.parse(undefined)).toBeUndefined();
337
+ });
338
+ });
@@ -0,0 +1,147 @@
1
+ import { afterAll, beforeAll, beforeEach, describe, expect, test } from "bun:test";
2
+ import { unlink } from "node:fs/promises";
3
+ import { closeDb, initDb } from "../be/db";
4
+ import { upsertOAuthApp } from "../be/db-queries/oauth";
5
+ import {
6
+ _clearPendingStates,
7
+ _getPendingState,
8
+ buildAuthorizationUrl,
9
+ exchangeCode,
10
+ type OAuthProviderConfig,
11
+ } from "../oauth/wrapper";
12
+
13
+ const TEST_DB_PATH = "./test-oauth-wrapper.sqlite";
14
+
15
+ const testConfig: OAuthProviderConfig = {
16
+ provider: "test-provider",
17
+ clientId: "test-client-id",
18
+ clientSecret: "test-client-secret",
19
+ authorizeUrl: "https://example.com/oauth/authorize",
20
+ tokenUrl: "https://example.com/oauth/token",
21
+ redirectUri: "http://localhost:3013/callback",
22
+ scopes: ["read", "write"],
23
+ extraParams: { actor: "app" },
24
+ };
25
+
26
+ beforeAll(() => {
27
+ initDb(TEST_DB_PATH);
28
+ // Create an oauth_app row so token storage works (FK constraint)
29
+ upsertOAuthApp("test-provider", {
30
+ clientId: testConfig.clientId,
31
+ clientSecret: testConfig.clientSecret,
32
+ authorizeUrl: testConfig.authorizeUrl,
33
+ tokenUrl: testConfig.tokenUrl,
34
+ redirectUri: testConfig.redirectUri,
35
+ scopes: testConfig.scopes.join(","),
36
+ });
37
+ });
38
+
39
+ beforeEach(() => {
40
+ _clearPendingStates();
41
+ });
42
+
43
+ afterAll(async () => {
44
+ closeDb();
45
+ await unlink(TEST_DB_PATH).catch(() => {});
46
+ await unlink(`${TEST_DB_PATH}-wal`).catch(() => {});
47
+ await unlink(`${TEST_DB_PATH}-shm`).catch(() => {});
48
+ });
49
+
50
+ describe("buildAuthorizationUrl", () => {
51
+ test("generates a valid URL with PKCE params", async () => {
52
+ const result = await buildAuthorizationUrl(testConfig);
53
+
54
+ expect(result.url).toBeTruthy();
55
+ expect(result.state).toBeTruthy();
56
+ expect(result.codeVerifier).toBeTruthy();
57
+
58
+ const url = new URL(result.url);
59
+ expect(url.origin + url.pathname).toBe("https://example.com/oauth/authorize");
60
+ expect(url.searchParams.get("client_id")).toBe("test-client-id");
61
+ expect(url.searchParams.get("redirect_uri")).toBe("http://localhost:3013/callback");
62
+ expect(url.searchParams.get("response_type")).toBe("code");
63
+ expect(url.searchParams.get("scope")).toBe("read,write");
64
+ expect(url.searchParams.get("state")).toBe(result.state);
65
+ expect(url.searchParams.get("code_challenge")).toBeTruthy();
66
+ expect(url.searchParams.get("code_challenge_method")).toBe("S256");
67
+ });
68
+
69
+ test("includes extra params in the URL", async () => {
70
+ const result = await buildAuthorizationUrl(testConfig);
71
+ const url = new URL(result.url);
72
+ expect(url.searchParams.get("actor")).toBe("app");
73
+ });
74
+
75
+ test("stores pending state with code verifier", async () => {
76
+ const result = await buildAuthorizationUrl(testConfig);
77
+ const pending = _getPendingState(result.state);
78
+
79
+ expect(pending).toBeTruthy();
80
+ expect(pending!.codeVerifier).toBe(result.codeVerifier);
81
+ expect(pending!.config.provider).toBe("test-provider");
82
+ expect(pending!.createdAt).toBeGreaterThan(0);
83
+ });
84
+
85
+ test("generates unique state for each call", async () => {
86
+ const result1 = await buildAuthorizationUrl(testConfig);
87
+ const result2 = await buildAuthorizationUrl(testConfig);
88
+
89
+ expect(result1.state).not.toBe(result2.state);
90
+ expect(result1.codeVerifier).not.toBe(result2.codeVerifier);
91
+ });
92
+
93
+ test("works without extra params", async () => {
94
+ const configNoExtras: OAuthProviderConfig = {
95
+ ...testConfig,
96
+ extraParams: undefined,
97
+ };
98
+
99
+ const result = await buildAuthorizationUrl(configNoExtras);
100
+ const url = new URL(result.url);
101
+ expect(url.searchParams.get("actor")).toBeNull();
102
+ });
103
+ });
104
+
105
+ describe("exchangeCode", () => {
106
+ test("rejects invalid state", async () => {
107
+ await expect(exchangeCode(testConfig, "some-code", "invalid-state")).rejects.toThrow(
108
+ "Invalid or expired OAuth state",
109
+ );
110
+ });
111
+
112
+ test("rejects already-consumed state", async () => {
113
+ const result = await buildAuthorizationUrl(testConfig);
114
+
115
+ // First exchange attempt will fail because there's no real token server,
116
+ // but it should consume the state
117
+ try {
118
+ await exchangeCode(testConfig, "some-code", result.state);
119
+ } catch {
120
+ // Expected: fetch to token URL fails
121
+ }
122
+
123
+ // Second attempt with the same state should fail with "Invalid or expired"
124
+ await expect(exchangeCode(testConfig, "some-code", result.state)).rejects.toThrow(
125
+ "Invalid or expired OAuth state",
126
+ );
127
+ });
128
+ });
129
+
130
+ describe("state TTL cleanup", () => {
131
+ test("expired states are cleaned up on next buildAuthorizationUrl call", async () => {
132
+ // Manually insert an "expired" entry by backdating createdAt
133
+ const result = await buildAuthorizationUrl(testConfig);
134
+ const pending = _getPendingState(result.state);
135
+ expect(pending).toBeTruthy();
136
+
137
+ // Backdate to 11 minutes ago (past the 10-minute TTL)
138
+ pending!.createdAt = Date.now() - 11 * 60 * 1000;
139
+
140
+ // Building a new URL triggers cleanup
141
+ await buildAuthorizationUrl(testConfig);
142
+
143
+ // The expired state should be gone
144
+ const expired = _getPendingState(result.state);
145
+ expect(expired).toBeUndefined();
146
+ });
147
+ });