@swarmclawai/swarmclaw 0.8.4 → 0.8.7

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 (394) hide show
  1. package/README.md +9 -9
  2. package/bin/swarmclaw.js +5 -1
  3. package/bin/worker-cmd.js +73 -0
  4. package/package.json +2 -1
  5. package/src/app/api/agents/[id]/route.ts +17 -7
  6. package/src/app/api/agents/route.ts +21 -8
  7. package/src/app/api/approvals/route.test.ts +6 -6
  8. package/src/app/api/approvals/route.ts +2 -1
  9. package/src/app/api/auth/route.ts +2 -3
  10. package/src/app/api/chatrooms/[id]/chat/route.test.ts +299 -0
  11. package/src/app/api/chatrooms/[id]/chat/route.ts +3 -2
  12. package/src/app/api/chatrooms/[id]/route.ts +7 -6
  13. package/src/app/api/chats/[id]/chat/route.test.ts +496 -0
  14. package/src/app/api/chats/[id]/chat/route.ts +7 -3
  15. package/src/app/api/chats/[id]/clear/route.ts +9 -9
  16. package/src/app/api/chats/[id]/devserver/route.ts +2 -1
  17. package/src/app/api/chats/[id]/edit-resend/route.ts +3 -4
  18. package/src/app/api/chats/[id]/fork/route.ts +3 -5
  19. package/src/app/api/chats/[id]/restore/route.ts +6 -7
  20. package/src/app/api/chats/[id]/retry/route.ts +3 -4
  21. package/src/app/api/chats/[id]/route.ts +61 -62
  22. package/src/app/api/chats/route.ts +7 -1
  23. package/src/app/api/connectors/[id]/route.ts +7 -8
  24. package/src/app/api/connectors/route.ts +5 -4
  25. package/src/app/api/eval/run/route.ts +2 -1
  26. package/src/app/api/eval/suite/route.ts +2 -1
  27. package/src/app/api/external-agents/route.test.ts +1 -1
  28. package/src/app/api/external-agents/route.ts +2 -2
  29. package/src/app/api/files/serve/route.ts +1 -1
  30. package/src/app/api/gateways/[id]/route.ts +7 -5
  31. package/src/app/api/gateways/route.ts +1 -1
  32. package/src/app/api/knowledge/upload/route.ts +1 -1
  33. package/src/app/api/logs/route.ts +5 -7
  34. package/src/app/api/memory-images/[filename]/route.ts +2 -3
  35. package/src/app/api/openclaw/agent-files/route.ts +4 -3
  36. package/src/app/api/openclaw/approvals/route.ts +3 -4
  37. package/src/app/api/openclaw/config-sync/route.ts +3 -2
  38. package/src/app/api/openclaw/cron/route.ts +3 -2
  39. package/src/app/api/openclaw/dotenv-keys/route.ts +2 -1
  40. package/src/app/api/openclaw/exec-config/route.ts +3 -2
  41. package/src/app/api/openclaw/gateway/route.ts +5 -4
  42. package/src/app/api/openclaw/history/route.ts +3 -2
  43. package/src/app/api/openclaw/media/route.ts +2 -1
  44. package/src/app/api/openclaw/permissions/route.ts +3 -2
  45. package/src/app/api/openclaw/sandbox-env/route.ts +3 -2
  46. package/src/app/api/openclaw/skills/install/route.ts +2 -1
  47. package/src/app/api/openclaw/skills/remove/route.ts +2 -1
  48. package/src/app/api/openclaw/skills/route.ts +3 -2
  49. package/src/app/api/orchestrator/run/route.ts +5 -14
  50. package/src/app/api/perf/route.ts +43 -0
  51. package/src/app/api/plugins/dependencies/route.ts +2 -1
  52. package/src/app/api/plugins/install/route.ts +2 -1
  53. package/src/app/api/plugins/marketplace/route.ts +3 -2
  54. package/src/app/api/plugins/settings/route.ts +2 -1
  55. package/src/app/api/preview-server/route.ts +11 -10
  56. package/src/app/api/projects/[id]/route.ts +1 -1
  57. package/src/app/api/schedules/[id]/route.test.ts +128 -0
  58. package/src/app/api/schedules/[id]/route.ts +43 -43
  59. package/src/app/api/schedules/[id]/run/route.ts +11 -62
  60. package/src/app/api/schedules/route.ts +21 -87
  61. package/src/app/api/settings/route.ts +2 -0
  62. package/src/app/api/setup/doctor/route.ts +9 -8
  63. package/src/app/api/tasks/[id]/approve/route.ts +33 -30
  64. package/src/app/api/tasks/[id]/route.ts +12 -35
  65. package/src/app/api/tasks/import/github/route.ts +2 -1
  66. package/src/app/api/tasks/route.ts +79 -91
  67. package/src/app/api/wallets/[id]/approve/route.ts +2 -1
  68. package/src/app/api/wallets/[id]/route.ts +13 -19
  69. package/src/app/api/wallets/[id]/send/route.ts +2 -1
  70. package/src/app/api/wallets/route.ts +2 -1
  71. package/src/app/api/webhooks/[id]/route.ts +2 -1
  72. package/src/app/api/webhooks/route.test.ts +3 -1
  73. package/src/app/page.tsx +23 -331
  74. package/src/cli/index.js +19 -0
  75. package/src/cli/index.ts +38 -7
  76. package/src/cli/spec.js +9 -0
  77. package/src/components/activity/activity-feed.tsx +7 -4
  78. package/src/components/agents/agent-card.tsx +32 -6
  79. package/src/components/agents/agent-chat-list.tsx +55 -22
  80. package/src/components/agents/agent-files-editor.tsx +3 -2
  81. package/src/components/agents/agent-sheet.tsx +123 -22
  82. package/src/components/agents/inspector-panel.tsx +1 -1
  83. package/src/components/agents/openclaw-skills-panel.tsx +2 -1
  84. package/src/components/agents/trash-list.tsx +1 -1
  85. package/src/components/auth/access-key-gate.tsx +8 -2
  86. package/src/components/auth/setup-wizard.tsx +10 -9
  87. package/src/components/auth/user-picker.tsx +3 -2
  88. package/src/components/chat/chat-area.tsx +20 -1
  89. package/src/components/chat/chat-card.tsx +18 -3
  90. package/src/components/chat/chat-header.tsx +24 -4
  91. package/src/components/chat/chat-list.tsx +2 -11
  92. package/src/components/chat/heartbeat-history-panel.tsx +2 -1
  93. package/src/components/chat/message-bubble.tsx +45 -6
  94. package/src/components/chat/message-list.tsx +280 -145
  95. package/src/components/chat/streaming-bubble.tsx +217 -60
  96. package/src/components/chat/swarm-panel.test.ts +274 -0
  97. package/src/components/chat/swarm-panel.tsx +410 -0
  98. package/src/components/chat/swarm-status-card.tsx +346 -0
  99. package/src/components/chat/tool-call-bubble.tsx +48 -23
  100. package/src/components/chatrooms/chatroom-list.tsx +8 -5
  101. package/src/components/chatrooms/chatroom-message.tsx +10 -7
  102. package/src/components/chatrooms/chatroom-view.tsx +12 -9
  103. package/src/components/connectors/connector-health.tsx +6 -4
  104. package/src/components/connectors/connector-list.tsx +16 -11
  105. package/src/components/connectors/connector-sheet.tsx +12 -6
  106. package/src/components/home/home-view.tsx +38 -24
  107. package/src/components/input/chat-input.tsx +10 -1
  108. package/src/components/layout/app-layout.tsx +2 -38
  109. package/src/components/layout/sheet-layer.tsx +50 -0
  110. package/src/components/mcp-servers/mcp-server-list.tsx +37 -5
  111. package/src/components/mcp-servers/mcp-server-sheet.tsx +12 -2
  112. package/src/components/plugins/plugin-list.tsx +8 -4
  113. package/src/components/plugins/plugin-sheet.tsx +2 -1
  114. package/src/components/providers/provider-list.tsx +3 -2
  115. package/src/components/providers/provider-sheet.tsx +2 -1
  116. package/src/components/runs/run-list.tsx +11 -7
  117. package/src/components/schedules/schedule-card.tsx +5 -3
  118. package/src/components/shared/agent-switch-dialog.tsx +1 -1
  119. package/src/components/shared/attachment-chip.tsx +19 -3
  120. package/src/components/shared/notification-center.tsx +6 -3
  121. package/src/components/shared/settings/plugin-manager.tsx +3 -2
  122. package/src/components/shared/settings/section-embedding.tsx +2 -1
  123. package/src/components/shared/settings/section-orchestrator.tsx +2 -1
  124. package/src/components/shared/settings/section-user-preferences.tsx +107 -0
  125. package/src/components/shared/settings/settings-page.tsx +13 -9
  126. package/src/components/skills/clawhub-browser.tsx +15 -4
  127. package/src/components/skills/skill-list.tsx +15 -4
  128. package/src/components/tasks/approvals-panel.tsx +2 -1
  129. package/src/components/tasks/task-board.tsx +35 -37
  130. package/src/components/tasks/task-sheet.tsx +4 -3
  131. package/src/components/ui/full-screen-loader.tsx +164 -0
  132. package/src/components/wallets/wallet-approval-dialog.tsx +2 -1
  133. package/src/components/wallets/wallet-panel.tsx +6 -5
  134. package/src/components/wallets/wallet-section.tsx +3 -2
  135. package/src/components/webhooks/webhook-list.tsx +4 -5
  136. package/src/components/webhooks/webhook-sheet.tsx +6 -6
  137. package/src/hooks/use-app-bootstrap.ts +202 -0
  138. package/src/hooks/use-mounted-ref.ts +14 -0
  139. package/src/hooks/use-now.ts +31 -0
  140. package/src/hooks/use-openclaw-gateway.ts +2 -1
  141. package/src/instrumentation.ts +20 -8
  142. package/src/lib/agent-default-tools.test.ts +52 -0
  143. package/src/lib/agent-default-tools.ts +40 -0
  144. package/src/lib/api-client.test.ts +21 -0
  145. package/src/lib/api-client.ts +6 -11
  146. package/src/lib/canvas-content.test.ts +360 -0
  147. package/src/lib/chat-streaming-state.test.ts +49 -2
  148. package/src/lib/chat-streaming-state.ts +26 -10
  149. package/src/lib/fetch-timeout.test.ts +54 -0
  150. package/src/lib/fetch-timeout.ts +60 -3
  151. package/src/lib/live-tool-events.test.ts +77 -0
  152. package/src/lib/live-tool-events.ts +73 -0
  153. package/src/lib/local-observability.test.ts +2 -2
  154. package/src/lib/openclaw-endpoint.test.ts +1 -1
  155. package/src/lib/providers/anthropic.ts +12 -16
  156. package/src/lib/providers/index.ts +4 -2
  157. package/src/lib/providers/ollama.ts +9 -6
  158. package/src/lib/providers/openai.ts +11 -14
  159. package/src/lib/runtime-env.test.ts +8 -8
  160. package/src/lib/schedule-dedupe-advanced.test.ts +2 -2
  161. package/src/lib/schedule-dedupe.test.ts +1 -1
  162. package/src/lib/schedule-dedupe.ts +3 -2
  163. package/src/lib/server/agent-thread-session.test.ts +6 -6
  164. package/src/lib/server/agent-thread-session.ts +6 -9
  165. package/src/lib/server/alert-dispatch.ts +2 -1
  166. package/src/lib/server/api-routes.test.ts +6 -6
  167. package/src/lib/server/approval-connector-notify.test.ts +4 -4
  168. package/src/lib/server/approvals-auto-approve.test.ts +29 -29
  169. package/src/lib/server/approvals.test.ts +317 -0
  170. package/src/lib/server/approvals.ts +5 -4
  171. package/src/lib/server/autonomy-runtime.test.ts +11 -11
  172. package/src/lib/server/browser-state.ts +2 -2
  173. package/src/lib/server/capability-router.test.ts +1 -1
  174. package/src/lib/server/capability-router.ts +3 -2
  175. package/src/lib/server/chat-execution-advanced.test.ts +15 -2
  176. package/src/lib/server/chat-execution-connector-delivery.ts +67 -0
  177. package/src/lib/server/chat-execution-disabled.test.ts +3 -3
  178. package/src/lib/server/chat-execution-eval-history.test.ts +3 -3
  179. package/src/lib/server/chat-execution-heartbeat.test.ts +42 -1
  180. package/src/lib/server/chat-execution-session-sync.test.ts +119 -0
  181. package/src/lib/server/chat-execution-tool-events.ts +116 -0
  182. package/src/lib/server/chat-execution-utils.test.ts +479 -0
  183. package/src/lib/server/chat-execution-utils.ts +533 -0
  184. package/src/lib/server/chat-execution.ts +153 -748
  185. package/src/lib/server/chat-streaming-utils.ts +174 -0
  186. package/src/lib/server/chat-turn-tool-routing.ts +310 -0
  187. package/src/lib/server/chatroom-session-persistence.test.ts +2 -2
  188. package/src/lib/server/clawhub-client.ts +2 -1
  189. package/src/lib/server/collection-helpers.test.ts +92 -0
  190. package/src/lib/server/collection-helpers.ts +25 -3
  191. package/src/lib/server/connectors/access.ts +146 -0
  192. package/src/lib/server/connectors/bluebubbles.test.ts +1 -1
  193. package/src/lib/server/connectors/bluebubbles.ts +4 -4
  194. package/src/lib/server/connectors/commands.ts +367 -0
  195. package/src/lib/server/connectors/connector-routing.test.ts +4 -4
  196. package/src/lib/server/connectors/delivery.ts +142 -0
  197. package/src/lib/server/connectors/discord.ts +37 -40
  198. package/src/lib/server/connectors/email.ts +11 -10
  199. package/src/lib/server/connectors/googlechat.ts +4 -4
  200. package/src/lib/server/connectors/inbound-audio-transcription.ts +2 -1
  201. package/src/lib/server/connectors/ingress-delivery.ts +23 -0
  202. package/src/lib/server/connectors/manager-roundtrip.test.ts +300 -0
  203. package/src/lib/server/connectors/manager.test.ts +352 -77
  204. package/src/lib/server/connectors/manager.ts +134 -673
  205. package/src/lib/server/connectors/matrix.ts +4 -4
  206. package/src/lib/server/connectors/message-sentinel.ts +7 -0
  207. package/src/lib/server/connectors/openclaw.test.ts +1 -1
  208. package/src/lib/server/connectors/openclaw.ts +8 -10
  209. package/src/lib/server/connectors/outbox.test.ts +192 -0
  210. package/src/lib/server/connectors/outbox.ts +369 -0
  211. package/src/lib/server/connectors/pairing.test.ts +18 -1
  212. package/src/lib/server/connectors/pairing.ts +49 -4
  213. package/src/lib/server/connectors/policy.ts +9 -3
  214. package/src/lib/server/connectors/reconnect-state.ts +71 -0
  215. package/src/lib/server/connectors/response-media.ts +256 -0
  216. package/src/lib/server/connectors/runtime-state.ts +67 -0
  217. package/src/lib/server/connectors/session.test.ts +357 -0
  218. package/src/lib/server/connectors/session.ts +422 -0
  219. package/src/lib/server/connectors/signal.ts +7 -7
  220. package/src/lib/server/connectors/slack.ts +43 -43
  221. package/src/lib/server/connectors/teams.ts +4 -4
  222. package/src/lib/server/connectors/telegram.ts +37 -43
  223. package/src/lib/server/connectors/types.ts +31 -1
  224. package/src/lib/server/connectors/whatsapp.test.ts +108 -0
  225. package/src/lib/server/connectors/whatsapp.ts +106 -34
  226. package/src/lib/server/context-manager.test.ts +409 -0
  227. package/src/lib/server/cost.test.ts +1 -1
  228. package/src/lib/server/daemon-policy.ts +78 -0
  229. package/src/lib/server/daemon-state-connectors.test.ts +167 -0
  230. package/src/lib/server/daemon-state.test.ts +283 -55
  231. package/src/lib/server/daemon-state.ts +106 -109
  232. package/src/lib/server/data-dir.test.ts +5 -5
  233. package/src/lib/server/data-dir.ts +4 -0
  234. package/src/lib/server/delegation-jobs-advanced.test.ts +1 -1
  235. package/src/lib/server/delegation-jobs.test.ts +87 -0
  236. package/src/lib/server/delegation-jobs.ts +42 -48
  237. package/src/lib/server/devserver-launch.ts +1 -1
  238. package/src/lib/server/document-utils.ts +7 -9
  239. package/src/lib/server/elevenlabs.ts +2 -1
  240. package/src/lib/server/embeddings.test.ts +105 -0
  241. package/src/lib/server/ethereum.ts +3 -2
  242. package/src/lib/server/eval/agent-regression.ts +3 -2
  243. package/src/lib/server/eval/runner.ts +2 -1
  244. package/src/lib/server/eval/scorer.ts +2 -1
  245. package/src/lib/server/evm-swap.ts +2 -1
  246. package/src/lib/server/gateway/protocol.test.ts +1 -1
  247. package/src/lib/server/guardian.ts +2 -1
  248. package/src/lib/server/heartbeat-blocked-suppression.test.ts +151 -0
  249. package/src/lib/server/heartbeat-service-timer.test.ts +6 -6
  250. package/src/lib/server/heartbeat-service.test.ts +406 -0
  251. package/src/lib/server/heartbeat-service.ts +54 -7
  252. package/src/lib/server/heartbeat-wake.test.ts +19 -0
  253. package/src/lib/server/heartbeat-wake.ts +17 -16
  254. package/src/lib/server/integrity-monitor.test.ts +149 -0
  255. package/src/lib/server/json-utils.ts +22 -0
  256. package/src/lib/server/knowledge-db.test.ts +13 -13
  257. package/src/lib/server/link-understanding.ts +2 -1
  258. package/src/lib/server/llm-response-cache.test.ts +1 -1
  259. package/src/lib/server/main-agent-loop-advanced.test.ts +65 -3
  260. package/src/lib/server/main-agent-loop.test.ts +6 -6
  261. package/src/lib/server/main-agent-loop.ts +21 -7
  262. package/src/lib/server/mcp-client.test.ts +1 -1
  263. package/src/lib/server/mcp-conformance.test.ts +1 -1
  264. package/src/lib/server/mcp-conformance.ts +3 -2
  265. package/src/lib/server/memory-consolidation.ts +2 -1
  266. package/src/lib/server/memory-db.test.ts +485 -0
  267. package/src/lib/server/memory-db.ts +39 -26
  268. package/src/lib/server/memory-graph.test.ts +2 -2
  269. package/src/lib/server/memory-policy.test.ts +7 -7
  270. package/src/lib/server/memory-retrieval.test.ts +1 -1
  271. package/src/lib/server/openclaw-config-sync.ts +2 -1
  272. package/src/lib/server/openclaw-deploy.test.ts +1 -1
  273. package/src/lib/server/openclaw-deploy.ts +8 -12
  274. package/src/lib/server/openclaw-exec-config.ts +2 -1
  275. package/src/lib/server/openclaw-gateway.ts +6 -7
  276. package/src/lib/server/openclaw-skills-normalize.ts +2 -1
  277. package/src/lib/server/openclaw-sync.ts +7 -5
  278. package/src/lib/server/orchestrator-lg-structure.test.ts +17 -0
  279. package/src/lib/server/orchestrator-lg.ts +199 -327
  280. package/src/lib/server/path-utils.ts +31 -0
  281. package/src/lib/server/perf.ts +161 -0
  282. package/src/lib/server/plugins-approval-guidance.ts +115 -0
  283. package/src/lib/server/plugins.test.ts +1 -1
  284. package/src/lib/server/plugins.ts +22 -132
  285. package/src/lib/server/process-manager.ts +5 -8
  286. package/src/lib/server/provider-health.test.ts +137 -0
  287. package/src/lib/server/provider-health.ts +3 -3
  288. package/src/lib/server/provider-model-discovery.ts +3 -12
  289. package/src/lib/server/queue-followups.test.ts +9 -9
  290. package/src/lib/server/queue-reconcile.test.ts +2 -2
  291. package/src/lib/server/queue-recovery.test.ts +269 -0
  292. package/src/lib/server/queue.test.ts +570 -0
  293. package/src/lib/server/queue.ts +62 -455
  294. package/src/lib/server/resolve-image.ts +30 -0
  295. package/src/lib/server/runtime-settings.test.ts +4 -4
  296. package/src/lib/server/runtime-storage-write-paths.test.ts +60 -0
  297. package/src/lib/server/schedule-normalization.test.ts +279 -0
  298. package/src/lib/server/schedule-service.ts +263 -0
  299. package/src/lib/server/scheduler.ts +17 -74
  300. package/src/lib/server/session-mailbox.test.ts +191 -0
  301. package/src/lib/server/session-run-manager.test.ts +640 -0
  302. package/src/lib/server/session-run-manager.ts +59 -15
  303. package/src/lib/server/session-tools/autonomy-tools.test.ts +20 -20
  304. package/src/lib/server/session-tools/calendar.ts +2 -1
  305. package/src/lib/server/session-tools/canvas.ts +2 -1
  306. package/src/lib/server/session-tools/chatroom.ts +2 -1
  307. package/src/lib/server/session-tools/connector.ts +26 -28
  308. package/src/lib/server/session-tools/context-mgmt.ts +3 -2
  309. package/src/lib/server/session-tools/crawl.ts +4 -3
  310. package/src/lib/server/session-tools/crud.ts +105 -324
  311. package/src/lib/server/session-tools/delegate-fallback.test.ts +9 -9
  312. package/src/lib/server/session-tools/delegate.ts +6 -8
  313. package/src/lib/server/session-tools/discovery-approvals.test.ts +15 -15
  314. package/src/lib/server/session-tools/discovery.ts +4 -3
  315. package/src/lib/server/session-tools/document.ts +2 -1
  316. package/src/lib/server/session-tools/email.ts +2 -1
  317. package/src/lib/server/session-tools/extract.ts +2 -1
  318. package/src/lib/server/session-tools/file.ts +4 -3
  319. package/src/lib/server/session-tools/http.ts +2 -1
  320. package/src/lib/server/session-tools/human-loop.ts +2 -1
  321. package/src/lib/server/session-tools/image-gen.ts +4 -3
  322. package/src/lib/server/session-tools/index.ts +26 -30
  323. package/src/lib/server/session-tools/mailbox.ts +2 -1
  324. package/src/lib/server/session-tools/manage-connectors.test.ts +4 -4
  325. package/src/lib/server/session-tools/manage-schedules.test.ts +12 -12
  326. package/src/lib/server/session-tools/manage-tasks-advanced.test.ts +5 -5
  327. package/src/lib/server/session-tools/manage-tasks.test.ts +2 -2
  328. package/src/lib/server/session-tools/monitor.ts +2 -1
  329. package/src/lib/server/session-tools/platform.ts +2 -1
  330. package/src/lib/server/session-tools/plugin-creator.ts +2 -1
  331. package/src/lib/server/session-tools/replicate.ts +3 -2
  332. package/src/lib/server/session-tools/session-tools-wiring.test.ts +6 -6
  333. package/src/lib/server/session-tools/shell.ts +4 -9
  334. package/src/lib/server/session-tools/subagent.ts +322 -170
  335. package/src/lib/server/session-tools/table.ts +6 -5
  336. package/src/lib/server/session-tools/wallet-tool.test.ts +3 -3
  337. package/src/lib/server/session-tools/wallet.ts +7 -6
  338. package/src/lib/server/session-tools/web-browser-config.test.ts +1 -0
  339. package/src/lib/server/session-tools/web-utils.ts +317 -0
  340. package/src/lib/server/session-tools/web.ts +62 -328
  341. package/src/lib/server/skill-prompt-budget.test.ts +1 -1
  342. package/src/lib/server/skills-normalize.ts +2 -1
  343. package/src/lib/server/storage-item-access.test.ts +302 -0
  344. package/src/lib/server/storage.ts +366 -314
  345. package/src/lib/server/stream-agent-chat.test.ts +82 -3
  346. package/src/lib/server/stream-agent-chat.ts +146 -510
  347. package/src/lib/server/stream-continuation.ts +412 -0
  348. package/src/lib/server/subagent-lineage.test.ts +647 -0
  349. package/src/lib/server/subagent-lineage.ts +435 -0
  350. package/src/lib/server/subagent-runtime.test.ts +484 -0
  351. package/src/lib/server/subagent-runtime.ts +419 -0
  352. package/src/lib/server/subagent-swarm.test.ts +391 -0
  353. package/src/lib/server/subagent-swarm.ts +564 -0
  354. package/src/lib/server/system-events.ts +3 -3
  355. package/src/lib/server/task-followups.test.ts +491 -0
  356. package/src/lib/server/task-followups.ts +391 -0
  357. package/src/lib/server/task-lifecycle.test.ts +205 -0
  358. package/src/lib/server/task-lifecycle.ts +200 -0
  359. package/src/lib/server/task-quality-gate.test.ts +1 -1
  360. package/src/lib/server/task-resume.ts +208 -0
  361. package/src/lib/server/task-service.test.ts +108 -0
  362. package/src/lib/server/task-service.ts +264 -0
  363. package/src/lib/server/task-validation.test.ts +1 -1
  364. package/src/lib/server/test-utils/run-with-temp-data-dir.ts +42 -0
  365. package/src/lib/server/tool-capability-policy.test.ts +2 -2
  366. package/src/lib/server/tool-capability-policy.ts +3 -2
  367. package/src/lib/server/tool-planning.ts +2 -1
  368. package/src/lib/server/tool-retry.ts +2 -3
  369. package/src/lib/server/wake-dispatcher.test.ts +303 -0
  370. package/src/lib/server/wake-dispatcher.ts +318 -0
  371. package/src/lib/server/wake-mode.test.ts +161 -0
  372. package/src/lib/server/wake-mode.ts +174 -0
  373. package/src/lib/server/wallet-service.ts +8 -9
  374. package/src/lib/server/watch-jobs.ts +2 -1
  375. package/src/lib/server/workspace-context.ts +2 -2
  376. package/src/lib/shared-utils.test.ts +142 -0
  377. package/src/lib/shared-utils.ts +62 -0
  378. package/src/lib/tool-event-summary.ts +2 -1
  379. package/src/lib/view-routes.test.ts +100 -0
  380. package/src/lib/wallet.test.ts +322 -6
  381. package/src/proxy.test.ts +4 -4
  382. package/src/proxy.ts +2 -3
  383. package/src/stores/set-if-changed.ts +40 -0
  384. package/src/stores/slices/agent-slice.ts +111 -0
  385. package/src/stores/slices/auth-slice.ts +25 -0
  386. package/src/stores/slices/data-slice.ts +301 -0
  387. package/src/stores/slices/index.ts +7 -0
  388. package/src/stores/slices/session-slice.ts +112 -0
  389. package/src/stores/slices/task-slice.ts +63 -0
  390. package/src/stores/slices/ui-slice.ts +192 -0
  391. package/src/stores/use-app-store.ts +17 -822
  392. package/src/stores/use-approval-store.ts +2 -1
  393. package/src/stores/use-chat-store.ts +8 -1
  394. package/src/types/index.ts +10 -0
@@ -1,21 +1,22 @@
1
1
  import { NextResponse } from 'next/server'
2
2
  import { genId } from '@/lib/id'
3
+ import { perf } from '@/lib/server/perf'
3
4
  import { deleteTask, loadAgents, loadSettings, loadTasks, logActivity, upsertTask } from '@/lib/server/storage'
4
5
  import { TaskCreateSchema, formatZodError } from '@/lib/validation/schemas'
5
6
  import { z } from 'zod'
6
7
  import { enqueueTask, recoverStalledRunningTasks, validateCompletedTasksQueue } from '@/lib/server/queue'
7
- import { ensureTaskCompletionReport } from '@/lib/server/task-reports'
8
- import { formatValidationFailure, validateTaskCompletion } from '@/lib/server/task-validation'
9
8
  import { pushMainLoopEventToMainSessions } from '@/lib/server/main-agent-loop'
10
9
  import { notify } from '@/lib/server/ws-hub'
11
- import { computeTaskFingerprint, findDuplicateTask } from '@/lib/task-dedupe'
12
10
  import { resolveTaskAgentFromDescription } from '@/lib/server/task-mention'
13
11
  import { validateDag } from '@/lib/server/dag-validation'
14
12
  import { getPluginManager } from '@/lib/server/plugins'
15
- import { normalizeTaskQualityGate } from '@/lib/server/task-quality-gate'
13
+ import {
14
+ prepareTaskCreation,
15
+ } from '@/lib/server/task-service'
16
16
  import '@/lib/server/builtin-plugins'
17
17
 
18
18
  export async function GET(req: Request) {
19
+ const endPerf = perf.start('api', 'GET /api/tasks')
19
20
  // Keep completed queue integrity even if daemon is not running.
20
21
  validateCompletedTasksQueue()
21
22
  recoverStalledRunningTasks()
@@ -25,6 +26,7 @@ export async function GET(req: Request) {
25
26
  const allTasks = loadTasks()
26
27
 
27
28
  if (includeArchived) {
29
+ endPerf({ count: Object.keys(allTasks).length })
28
30
  return NextResponse.json(allTasks)
29
31
  }
30
32
 
@@ -35,6 +37,7 @@ export async function GET(req: Request) {
35
37
  filtered[id] = task
36
38
  }
37
39
  }
40
+ endPerf({ count: Object.keys(filtered).length })
38
41
  return NextResponse.json(filtered)
39
42
  }
40
43
 
@@ -71,9 +74,6 @@ export async function POST(req: Request) {
71
74
  const now = Date.now()
72
75
  const tasks = loadTasks()
73
76
  const settings = loadSettings()
74
- const normalizedQualityGate = body.qualityGate
75
- ? normalizeTaskQualityGate(body.qualityGate, settings)
76
- : null
77
77
  const maxAttempts = Number.isFinite(Number(body.maxAttempts))
78
78
  ? Math.max(1, Math.min(20, Math.trunc(Number(body.maxAttempts))))
79
79
  : Math.max(1, Math.min(20, Math.trunc(Number(settings.defaultTaskMaxAttempts ?? 3))))
@@ -96,101 +96,89 @@ export async function POST(req: Request) {
96
96
  ? resolveTaskAgentFromDescription(body.description, body.agentId || '', loadAgents())
97
97
  : (body.agentId || '')
98
98
 
99
- tasks[id] = {
99
+ const prepared = prepareTaskCreation({
100
100
  id,
101
- title: body.title || 'Untitled Task',
102
- description: body.description || '',
103
- status: body.status || 'backlog',
104
- agentId: resolvedAgentId,
105
- projectId: typeof body.projectId === 'string' && body.projectId ? body.projectId : null,
106
- goalContract: body.goalContract || null,
107
- cwd: typeof body.cwd === 'string' ? body.cwd : null,
108
- file: typeof body.file === 'string' ? body.file : null,
109
- sessionId: typeof body.sessionId === 'string' ? body.sessionId : null,
110
- result: typeof body.result === 'string' ? body.result : null,
111
- error: typeof body.error === 'string' ? body.error : null,
112
- outputFiles: Array.isArray(body.outputFiles)
113
- ? body.outputFiles.filter((entry: unknown) => typeof entry === 'string').slice(0, 24)
114
- : [],
115
- artifacts: Array.isArray(body.artifacts)
116
- ? body.artifacts
117
- .filter((artifact: unknown) => artifact && typeof artifact === 'object')
118
- .map((artifact: unknown) => {
119
- const row = artifact as {
120
- url?: unknown
121
- type?: unknown
122
- filename?: unknown
123
- }
124
- const normalizedType = String(row.type || '')
125
- return {
126
- url: String(row.url || ''),
127
- type: ['image', 'video', 'pdf', 'file'].includes(normalizedType)
128
- ? (normalizedType as 'image' | 'video' | 'pdf' | 'file')
129
- : 'file',
130
- filename: String(row.filename || ''),
131
- }
132
- })
133
- .filter((artifact: { url: string; filename: string }) => artifact.url && artifact.filename)
134
- .slice(0, 24)
135
- : [],
136
- createdAt: now,
137
- updatedAt: now,
138
- queuedAt: null,
139
- startedAt: null,
140
- completedAt: null,
141
- archivedAt: null,
142
- attempts: 0,
143
- maxAttempts,
144
- retryBackoffSec,
145
- retryScheduledAt: null,
146
- deadLetteredAt: null,
147
- checkpoint: null,
148
- blockedBy: Array.isArray(body.blockedBy) ? body.blockedBy.filter((s: unknown) => typeof s === 'string') : [],
149
- blocks: Array.isArray(body.blocks) ? body.blocks.filter((s: unknown) => typeof s === 'string') : [],
150
- tags: Array.isArray(body.tags) ? body.tags.filter((s: unknown) => typeof s === 'string') : [],
151
- dueAt: typeof body.dueAt === 'number' ? body.dueAt : null,
152
- customFields: body.customFields && typeof body.customFields === 'object' ? body.customFields : undefined,
153
- priority: ['low', 'medium', 'high', 'critical'].includes(body.priority) ? body.priority : undefined,
154
- fingerprint: computeTaskFingerprint(body.title || 'Untitled Task', body.agentId || ''),
155
- qualityGate: normalizedQualityGate,
101
+ input: {
102
+ ...body,
103
+ agentId: resolvedAgentId,
104
+ },
105
+ tasks,
106
+ now,
107
+ settings,
108
+ seed: {
109
+ projectId: typeof body.projectId === 'string' && body.projectId ? body.projectId : null,
110
+ goalContract: body.goalContract || null,
111
+ cwd: typeof body.cwd === 'string' ? body.cwd : null,
112
+ file: typeof body.file === 'string' ? body.file : null,
113
+ sessionId: typeof body.sessionId === 'string' ? body.sessionId : null,
114
+ result: typeof body.result === 'string' ? body.result : null,
115
+ error: typeof body.error === 'string' ? body.error : null,
116
+ outputFiles: Array.isArray(body.outputFiles)
117
+ ? body.outputFiles.filter((entry: unknown) => typeof entry === 'string').slice(0, 24)
118
+ : [],
119
+ artifacts: Array.isArray(body.artifacts)
120
+ ? body.artifacts
121
+ .filter((artifact: unknown) => artifact && typeof artifact === 'object')
122
+ .map((artifact: unknown) => {
123
+ const row = artifact as {
124
+ url?: unknown
125
+ type?: unknown
126
+ filename?: unknown
127
+ }
128
+ const normalizedType = String(row.type || '')
129
+ return {
130
+ url: String(row.url || ''),
131
+ type: ['image', 'video', 'pdf', 'file'].includes(normalizedType)
132
+ ? (normalizedType as 'image' | 'video' | 'pdf' | 'file')
133
+ : 'file',
134
+ filename: String(row.filename || ''),
135
+ }
136
+ })
137
+ .filter((artifact: { url: string; filename: string }) => artifact.url && artifact.filename)
138
+ .slice(0, 24)
139
+ : [],
140
+ archivedAt: null,
141
+ attempts: 0,
142
+ maxAttempts,
143
+ retryBackoffSec,
144
+ retryScheduledAt: null,
145
+ deadLetteredAt: null,
146
+ checkpoint: null,
147
+ blockedBy: Array.isArray(body.blockedBy) ? body.blockedBy.filter((s: unknown) => typeof s === 'string') : [],
148
+ blocks: Array.isArray(body.blocks) ? body.blocks.filter((s: unknown) => typeof s === 'string') : [],
149
+ tags: Array.isArray(body.tags) ? body.tags.filter((s: unknown) => typeof s === 'string') : [],
150
+ dueAt: typeof body.dueAt === 'number' ? body.dueAt : null,
151
+ customFields: body.customFields && typeof body.customFields === 'object' ? body.customFields : undefined,
152
+ priority: ['low', 'medium', 'high', 'critical'].includes(body.priority) ? body.priority : undefined,
153
+ },
154
+ })
155
+ if (!prepared.ok) {
156
+ return NextResponse.json({ error: prepared.error }, { status: 400 })
156
157
  }
157
158
 
158
- // Dedup: if a non-terminal task with same fingerprint exists, return it
159
- const dupe = findDuplicateTask(tasks, { fingerprint: tasks[id].fingerprint! })
160
- if (dupe && dupe.id !== id) {
161
- return NextResponse.json({ ...dupe, deduplicated: true })
159
+ if (prepared.duplicate) {
160
+ return NextResponse.json({ ...prepared.duplicate, deduplicated: true })
162
161
  }
163
162
 
164
- if (tasks[id].status === 'completed') {
165
- const report = ensureTaskCompletionReport(tasks[id])
166
- if (report?.relativePath) tasks[id].completionReportPath = report.relativePath
167
- const validation = validateTaskCompletion(tasks[id], { report, settings })
168
- tasks[id].validation = validation
169
- if (validation.ok) {
170
- tasks[id].completedAt = Date.now()
171
- tasks[id].error = null
172
- const agentPlugins = resolvedAgentId ? (loadAgents()[resolvedAgentId]?.plugins || []) : []
173
- getPluginManager().runHook(
174
- 'onTaskComplete',
175
- { taskId: id, result: tasks[id].result },
176
- { enabledIds: agentPlugins },
177
- )
178
- } else {
179
- tasks[id].status = 'failed'
180
- tasks[id].completedAt = null
181
- tasks[id].error = formatValidationFailure(validation.reasons).slice(0, 500)
182
- }
163
+ const task = prepared.task
164
+ if (task.status === 'completed') {
165
+ const agentPlugins = resolvedAgentId ? (loadAgents()[resolvedAgentId]?.plugins || []) : []
166
+ getPluginManager().runHook(
167
+ 'onTaskComplete',
168
+ { taskId: id, result: task.result },
169
+ { enabledIds: agentPlugins },
170
+ )
183
171
  }
184
172
 
185
- upsertTask(id, tasks[id])
186
- logActivity({ entityType: 'task', entityId: id, action: 'created', actor: 'user', summary: `Task created: "${tasks[id].title}"` })
173
+ upsertTask(id, task)
174
+ logActivity({ entityType: 'task', entityId: id, action: 'created', actor: 'user', summary: `Task created: "${task.title}"` })
187
175
  pushMainLoopEventToMainSessions({
188
176
  type: 'task_created',
189
- text: `Task created: "${tasks[id].title}" (${id}) with status ${tasks[id].status}.`,
177
+ text: `Task created: "${task.title}" (${id}) with status ${task.status}.`,
190
178
  })
191
- if (tasks[id].status === 'queued') {
179
+ if (task.status === 'queued') {
192
180
  enqueueTask(id)
193
181
  }
194
182
  notify('tasks')
195
- return NextResponse.json(tasks[id])
183
+ return NextResponse.json(task)
196
184
  }
@@ -4,6 +4,7 @@ import { notify } from '@/lib/server/ws-hub'
4
4
  import type { AgentWallet, WalletTransaction } from '@/types'
5
5
  import { getWalletAtomicAmount } from '@/lib/wallet'
6
6
  import { sendWalletNativeAsset, validateWalletSendLimits } from '@/lib/server/wallet-service'
7
+ import { errorMessage } from '@/lib/shared-utils'
7
8
  export const dynamic = 'force-dynamic'
8
9
 
9
10
  export async function POST(req: Request, { params }: { params: Promise<{ id: string }> }) {
@@ -68,7 +69,7 @@ export async function POST(req: Request, { params }: { params: Promise<{ id: str
68
69
  upsertWalletTransaction(transactionId, tx)
69
70
  notify('wallets')
70
71
  return NextResponse.json({
71
- error: err instanceof Error ? err.message : String(err),
72
+ error: errorMessage(err),
72
73
  transactionId,
73
74
  status: 'failed',
74
75
  }, { status: 500 })
@@ -1,5 +1,5 @@
1
1
  import { NextResponse } from 'next/server'
2
- import { loadWallets, upsertWallet, deleteWallet as deleteWalletFromStore, loadAgents, saveAgents } from '@/lib/server/storage'
2
+ import { loadWallets, upsertWallet, deleteWallet as deleteWalletFromStore, loadAgent, loadAgents, upsertAgent } from '@/lib/server/storage'
3
3
  import { notify } from '@/lib/server/ws-hub'
4
4
  import { getWalletLimitAtomic, normalizeAtomicString } from '@/lib/wallet'
5
5
  import type { AgentWallet, WalletAssetBalance, WalletPortfolioSummary } from '@/types'
@@ -89,8 +89,7 @@ export async function PATCH(req: Request, { params }: { params: Promise<{ id: st
89
89
 
90
90
  // Reassign wallet to a different agent
91
91
  if (typeof body.agentId === 'string' && body.agentId !== wallet.agentId) {
92
- const agents = loadAgents()
93
- const newAgent = agents[body.agentId]
92
+ const newAgent = loadAgent(body.agentId)
94
93
  if (!newAgent) return NextResponse.json({ error: 'Agent not found' }, { status: 404 })
95
94
 
96
95
  // Only one wallet per chain per agent.
@@ -98,28 +97,25 @@ export async function PATCH(req: Request, { params }: { params: Promise<{ id: st
98
97
  const conflict = Object.values(allWallets).find((w) => w.agentId === body.agentId && w.id !== id && w.chain === wallet.chain)
99
98
  if (conflict) return NextResponse.json({ error: `Target agent already has a ${wallet.chain} wallet` }, { status: 409 })
100
99
 
101
- const oldAgent = agents[wallet.agentId]
100
+ const oldAgent = loadAgent(wallet.agentId)
102
101
  if (oldAgent) {
103
- unlinkWalletFromAgent(oldAgent, id)
102
+ unlinkWalletFromAgent(oldAgent as any, id)
104
103
  oldAgent.updatedAt = Date.now()
105
- agents[wallet.agentId] = oldAgent
104
+ upsertAgent(wallet.agentId, oldAgent)
106
105
  }
107
106
 
108
- linkWalletToAgent(newAgent, id, shouldMakeActive || getAgentActiveWalletId(newAgent) == null)
107
+ linkWalletToAgent(newAgent as any, id, shouldMakeActive || getAgentActiveWalletId(newAgent as any) == null)
109
108
  newAgent.updatedAt = Date.now()
110
- agents[body.agentId] = newAgent
111
- saveAgents(agents)
109
+ upsertAgent(body.agentId, newAgent)
112
110
  notify('agents')
113
111
 
114
112
  wallet.agentId = body.agentId
115
113
  } else if (shouldMakeActive) {
116
- const agents = loadAgents()
117
- const agent = agents[wallet.agentId]
114
+ const agent = loadAgent(wallet.agentId)
118
115
  if (agent) {
119
- setAgentActiveWallet(agent, id)
116
+ setAgentActiveWallet(agent as any, id)
120
117
  agent.updatedAt = Date.now()
121
- agents[wallet.agentId] = agent
122
- saveAgents(agents)
118
+ upsertAgent(wallet.agentId, agent)
123
119
  notify('agents')
124
120
  }
125
121
  }
@@ -156,13 +152,11 @@ export async function DELETE(_req: Request, { params }: { params: Promise<{ id:
156
152
  } catch { /* ignore */ }
157
153
 
158
154
  // Unlink from agent
159
- const agents = loadAgents()
160
- const agent = agents[wallet.agentId]
155
+ const agent = loadAgent(wallet.agentId)
161
156
  if (agent) {
162
- unlinkWalletFromAgent(agent, id)
157
+ unlinkWalletFromAgent(agent as any, id)
163
158
  agent.updatedAt = Date.now()
164
- agents[wallet.agentId] = agent
165
- saveAgents(agents)
159
+ upsertAgent(wallet.agentId, agent)
166
160
  notify('agents')
167
161
  }
168
162
 
@@ -7,6 +7,7 @@ import {
7
7
  normalizeAtomicString,
8
8
  } from '@/lib/wallet'
9
9
  import { isValidWalletAddress, sendWalletNativeAsset, validateWalletSendLimits } from '@/lib/server/wallet-service'
10
+ import { errorMessage } from '@/lib/shared-utils'
10
11
  export const dynamic = 'force-dynamic'
11
12
 
12
13
  export async function POST(req: Request, { params }: { params: Promise<{ id: string }> }) {
@@ -96,7 +97,7 @@ export async function POST(req: Request, { params }: { params: Promise<{ id: str
96
97
  upsertWalletTransaction(txId, failedTx)
97
98
  notify('wallets')
98
99
  return NextResponse.json({
99
- error: err instanceof Error ? err.message : String(err),
100
+ error: errorMessage(err),
100
101
  transactionId: txId,
101
102
  status: 'failed',
102
103
  }, { status: 500 })
@@ -3,6 +3,7 @@ import { loadAgents, loadWallets } from '@/lib/server/storage'
3
3
  import { createAgentWallet, getAgentActiveWalletId, getWalletPortfolioSnapshot, stripWalletPrivateKey } from '@/lib/server/wallet-service'
4
4
  import { buildEmptyWalletPortfolio } from '@/lib/server/wallet-portfolio'
5
5
  import type { AgentWallet, WalletPortfolioSummary } from '@/types'
6
+ import { errorMessage } from '@/lib/shared-utils'
6
7
  export const dynamic = 'force-dynamic'
7
8
  const WALLET_LIST_PORTFOLIO_TIMEOUT_MS = 1500
8
9
 
@@ -73,7 +74,7 @@ export async function POST(req: Request) {
73
74
  })
74
75
  return NextResponse.json(stripWalletPrivateKey(wallet as unknown as Record<string, unknown>))
75
76
  } catch (err: unknown) {
76
- const message = err instanceof Error ? err.message : String(err)
77
+ const message = errorMessage(err)
77
78
  if (message === 'agentId is required') {
78
79
  return NextResponse.json({ error: message }, { status: 400 })
79
80
  }
@@ -9,6 +9,7 @@ import { requestHeartbeatNow } from '@/lib/server/heartbeat-wake'
9
9
  import { mutateItem, deleteItem, notFound, type CollectionOps } from '@/lib/server/collection-helpers'
10
10
  import type { WebhookRetryEntry } from '@/types'
11
11
  import { triggerWebhookWatchJobs } from '@/lib/server/watch-jobs'
12
+ import { errorMessage } from '@/lib/shared-utils'
12
13
 
13
14
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
14
15
  const ops: CollectionOps<any> = { load: loadWebhooks, save: saveWebhooks }
@@ -250,7 +251,7 @@ export async function handleWebhookPost(
250
251
  runId: run.runId,
251
252
  })
252
253
  } catch (err: unknown) {
253
- const errorMsg = err instanceof Error ? err.message : String(err)
254
+ const errorMsg = errorMessage(err)
254
255
 
255
256
  // Enqueue for retry with exponential backoff
256
257
  const retryId = genId()
@@ -88,12 +88,13 @@ test('handleWebhookPost creates a session, records success history, and triggers
88
88
  webhookId,
89
89
  {
90
90
  enqueueRun(input) {
91
- calls.runs.push(input as Record<string, unknown>)
91
+ calls.runs.push(input as any)
92
92
  return {
93
93
  runId: 'run-success-smoke',
94
94
  position: 0,
95
95
  promise: Promise.resolve({} as never),
96
96
  abort: () => {},
97
+ unsubscribe: () => {},
97
98
  }
98
99
  },
99
100
  enqueueEvent(sessionId, text) {
@@ -166,6 +167,7 @@ test('handleWebhookPost ignores filtered events without dispatching or logging d
166
167
  position: 0,
167
168
  promise: Promise.resolve({} as never),
168
169
  abort: () => {},
170
+ unsubscribe: () => {},
169
171
  }
170
172
  },
171
173
  enqueueEvent() {},