@swarmclawai/swarmclaw 0.7.7 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (281) hide show
  1. package/README.md +12 -14
  2. package/next.config.ts +13 -2
  3. package/package.json +4 -2
  4. package/src/app/api/agents/[id]/thread/route.ts +9 -0
  5. package/src/app/api/agents/route.ts +4 -0
  6. package/src/app/api/agents/thread-route.test.ts +133 -0
  7. package/src/app/api/approvals/route.test.ts +148 -0
  8. package/src/app/api/canvas/[sessionId]/route.ts +3 -1
  9. package/src/app/api/chatrooms/[id]/chat/route.ts +4 -2
  10. package/src/app/api/chats/[id]/devserver/route.ts +48 -7
  11. package/src/app/api/chats/[id]/messages/route.ts +42 -18
  12. package/src/app/api/chats/[id]/route.ts +1 -1
  13. package/src/app/api/chats/[id]/stop/route.ts +5 -4
  14. package/src/app/api/chats/route.ts +23 -2
  15. package/src/app/api/clawhub/install/route.ts +28 -8
  16. package/src/app/api/connectors/[id]/route.ts +46 -3
  17. package/src/app/api/connectors/route.ts +12 -8
  18. package/src/app/api/external-agents/route.test.ts +165 -0
  19. package/src/app/api/gateways/[id]/health/route.ts +27 -12
  20. package/src/app/api/gateways/[id]/route.ts +2 -0
  21. package/src/app/api/gateways/health-route.test.ts +135 -0
  22. package/src/app/api/gateways/route.ts +2 -0
  23. package/src/app/api/mcp-servers/route.test.ts +130 -0
  24. package/src/app/api/openclaw/deploy/route.ts +38 -5
  25. package/src/app/api/plugins/install/route.ts +46 -6
  26. package/src/app/api/plugins/marketplace/route.ts +48 -15
  27. package/src/app/api/preview-server/route.ts +26 -11
  28. package/src/app/api/projects/[id]/route.ts +6 -2
  29. package/src/app/api/projects/route.ts +4 -3
  30. package/src/app/api/schedules/[id]/run/route.ts +4 -0
  31. package/src/app/api/schedules/route.test.ts +86 -0
  32. package/src/app/api/schedules/route.ts +6 -1
  33. package/src/app/api/secrets/[id]/route.ts +1 -0
  34. package/src/app/api/secrets/route.ts +2 -1
  35. package/src/app/api/settings/route.ts +2 -0
  36. package/src/app/api/setup/check-provider/route.test.ts +19 -0
  37. package/src/app/api/setup/check-provider/route.ts +40 -10
  38. package/src/app/api/skills/[id]/route.ts +12 -0
  39. package/src/app/api/skills/import/route.ts +14 -12
  40. package/src/app/api/skills/route.ts +13 -1
  41. package/src/app/api/tasks/[id]/route.ts +10 -1
  42. package/src/app/api/tasks/import/github/route.test.ts +65 -0
  43. package/src/app/api/tasks/import/github/route.ts +337 -0
  44. package/src/app/api/wallets/[id]/approve/route.ts +17 -3
  45. package/src/app/api/wallets/[id]/route.ts +79 -33
  46. package/src/app/api/wallets/[id]/send/route.ts +19 -33
  47. package/src/app/api/wallets/route.ts +78 -61
  48. package/src/app/api/webhooks/[id]/route.ts +33 -6
  49. package/src/app/api/webhooks/route.test.ts +272 -0
  50. package/src/cli/index.js +1 -0
  51. package/src/cli/spec.js +1 -0
  52. package/src/components/agents/agent-card.tsx +9 -2
  53. package/src/components/agents/agent-chat-list.tsx +18 -2
  54. package/src/components/agents/agent-list.tsx +1 -0
  55. package/src/components/agents/agent-sheet.tsx +257 -38
  56. package/src/components/agents/inspector-panel.tsx +41 -0
  57. package/src/components/canvas/canvas-panel.tsx +236 -65
  58. package/src/components/chat/chat-area.tsx +36 -19
  59. package/src/components/chat/chat-card.tsx +36 -13
  60. package/src/components/chat/chat-header.tsx +48 -16
  61. package/src/components/chat/chat-list.tsx +28 -4
  62. package/src/components/chat/checkpoint-timeline.tsx +50 -34
  63. package/src/components/chat/delegation-banner.test.ts +14 -1
  64. package/src/components/chat/delegation-banner.tsx +1 -1
  65. package/src/components/chat/message-bubble.tsx +208 -145
  66. package/src/components/chat/message-list.tsx +48 -19
  67. package/src/components/chatrooms/chatroom-message.tsx +2 -2
  68. package/src/components/chatrooms/chatroom-sheet.tsx +16 -2
  69. package/src/components/connectors/connector-health.tsx +1 -1
  70. package/src/components/connectors/connector-list.tsx +7 -2
  71. package/src/components/connectors/connector-sheet.tsx +337 -148
  72. package/src/components/gateways/gateway-sheet.tsx +2 -2
  73. package/src/components/layout/app-layout.tsx +40 -23
  74. package/src/components/mcp-servers/mcp-server-list.tsx +26 -5
  75. package/src/components/mcp-servers/mcp-server-sheet.tsx +19 -2
  76. package/src/components/openclaw/openclaw-deploy-panel.tsx +269 -21
  77. package/src/components/plugins/plugin-list.tsx +45 -9
  78. package/src/components/plugins/plugin-sheet.tsx +55 -7
  79. package/src/components/projects/project-detail.tsx +217 -0
  80. package/src/components/projects/project-sheet.tsx +176 -4
  81. package/src/components/providers/provider-list.tsx +2 -1
  82. package/src/components/providers/provider-sheet.tsx +21 -2
  83. package/src/components/schedules/schedule-card.tsx +25 -1
  84. package/src/components/schedules/schedule-sheet.tsx +44 -2
  85. package/src/components/secrets/secret-sheet.tsx +21 -2
  86. package/src/components/shared/agent-switch-dialog.tsx +12 -1
  87. package/src/components/shared/bottom-sheet.tsx +13 -3
  88. package/src/components/shared/command-palette.tsx +8 -1
  89. package/src/components/shared/confirm-dialog.tsx +19 -4
  90. package/src/components/shared/connector-platform-icon.test.ts +28 -0
  91. package/src/components/shared/connector-platform-icon.tsx +39 -6
  92. package/src/components/shared/settings/plugin-manager.tsx +29 -6
  93. package/src/components/shared/settings/section-capability-policy.tsx +45 -3
  94. package/src/components/shared/settings/section-voice.tsx +11 -3
  95. package/src/components/skills/skill-list.tsx +25 -0
  96. package/src/components/skills/skill-sheet.tsx +84 -12
  97. package/src/components/tasks/approvals-panel.tsx +289 -34
  98. package/src/components/tasks/task-board.tsx +410 -25
  99. package/src/components/tasks/task-card.tsx +66 -8
  100. package/src/components/tasks/task-sheet.tsx +16 -4
  101. package/src/components/ui/dialog.tsx +2 -2
  102. package/src/components/wallets/wallet-approval-dialog.tsx +4 -2
  103. package/src/components/wallets/wallet-panel.tsx +435 -90
  104. package/src/components/wallets/wallet-section.tsx +198 -48
  105. package/src/components/webhooks/webhook-sheet.tsx +22 -2
  106. package/src/lib/approval-display.ts +20 -0
  107. package/src/lib/canvas-content.ts +198 -0
  108. package/src/lib/chat-artifact-summary.ts +165 -0
  109. package/src/lib/chat-display.test.ts +91 -0
  110. package/src/lib/chat-display.ts +58 -0
  111. package/src/lib/chat-streaming-state.test.ts +47 -1
  112. package/src/lib/chat-streaming-state.ts +42 -0
  113. package/src/lib/ollama-model.ts +10 -0
  114. package/src/lib/openclaw-endpoint.test.ts +8 -0
  115. package/src/lib/openclaw-endpoint.ts +6 -1
  116. package/src/lib/plugin-install-cors.ts +46 -0
  117. package/src/lib/plugin-sources.test.ts +43 -0
  118. package/src/lib/plugin-sources.ts +77 -0
  119. package/src/lib/providers/ollama.ts +16 -6
  120. package/src/lib/providers/openclaw.test.ts +54 -0
  121. package/src/lib/providers/openclaw.ts +127 -11
  122. package/src/lib/schedule-dedupe-advanced.test.ts +1335 -0
  123. package/src/lib/schedule-dedupe.test.ts +66 -1
  124. package/src/lib/schedule-dedupe.ts +169 -12
  125. package/src/lib/schedule-origin.test.ts +20 -0
  126. package/src/lib/schedule-origin.ts +15 -0
  127. package/src/lib/server/__fixtures__/fake-mcp-stdio-server.mjs +27 -0
  128. package/src/lib/server/agent-availability.ts +16 -0
  129. package/src/lib/server/agent-runtime-config.ts +12 -4
  130. package/src/lib/server/agent-thread-session.test.ts +51 -0
  131. package/src/lib/server/agent-thread-session.ts +7 -0
  132. package/src/lib/server/approval-match.ts +205 -0
  133. package/src/lib/server/approvals-auto-approve.test.ts +538 -1
  134. package/src/lib/server/approvals.ts +214 -1
  135. package/src/lib/server/assistant-control.test.ts +29 -0
  136. package/src/lib/server/assistant-control.ts +23 -0
  137. package/src/lib/server/build-llm.test.ts +79 -0
  138. package/src/lib/server/build-llm.ts +14 -4
  139. package/src/lib/server/canvas-content.test.ts +32 -0
  140. package/src/lib/server/canvas-content.ts +6 -0
  141. package/src/lib/server/capability-router.test.ts +33 -0
  142. package/src/lib/server/capability-router.ts +80 -19
  143. package/src/lib/server/chat-execution-advanced.test.ts +651 -0
  144. package/src/lib/server/chat-execution-disabled.test.ts +94 -0
  145. package/src/lib/server/chat-execution-tool-events.test.ts +157 -0
  146. package/src/lib/server/chat-execution.ts +378 -73
  147. package/src/lib/server/clawhub-client.test.ts +14 -8
  148. package/src/lib/server/connectors/manager-reconnect.test.ts +47 -0
  149. package/src/lib/server/connectors/manager.test.ts +1147 -0
  150. package/src/lib/server/connectors/manager.ts +461 -137
  151. package/src/lib/server/connectors/pairing.ts +26 -5
  152. package/src/lib/server/connectors/types.ts +2 -0
  153. package/src/lib/server/connectors/whatsapp.test.ts +134 -0
  154. package/src/lib/server/connectors/whatsapp.ts +271 -47
  155. package/src/lib/server/context-manager.ts +6 -1
  156. package/src/lib/server/daemon-state.ts +84 -47
  157. package/src/lib/server/data-dir.test.ts +37 -0
  158. package/src/lib/server/data-dir.ts +20 -1
  159. package/src/lib/server/delegation-jobs-advanced.test.ts +513 -0
  160. package/src/lib/server/devserver-launch.test.ts +60 -0
  161. package/src/lib/server/devserver-launch.ts +85 -0
  162. package/src/lib/server/elevenlabs.test.ts +247 -1
  163. package/src/lib/server/elevenlabs.ts +147 -43
  164. package/src/lib/server/ethereum.ts +590 -0
  165. package/src/lib/server/eval/agent-regression-advanced.test.ts +302 -0
  166. package/src/lib/server/eval/agent-regression.test.ts +18 -1
  167. package/src/lib/server/eval/agent-regression.ts +383 -11
  168. package/src/lib/server/evm-swap.ts +475 -0
  169. package/src/lib/server/execution-log.ts +1 -0
  170. package/src/lib/server/heartbeat-service-timer.test.ts +173 -0
  171. package/src/lib/server/heartbeat-service.ts +20 -11
  172. package/src/lib/server/heartbeat-wake.test.ts +112 -0
  173. package/src/lib/server/heartbeat-wake.ts +338 -57
  174. package/src/lib/server/main-agent-loop-advanced.test.ts +538 -0
  175. package/src/lib/server/main-agent-loop.test.ts +260 -0
  176. package/src/lib/server/main-agent-loop.ts +559 -14
  177. package/src/lib/server/mcp-client.test.ts +16 -0
  178. package/src/lib/server/mcp-client.ts +25 -0
  179. package/src/lib/server/memory-integration.test.ts +719 -0
  180. package/src/lib/server/memory-policy.test.ts +43 -0
  181. package/src/lib/server/memory-policy.ts +132 -0
  182. package/src/lib/server/memory-tiers.test.ts +60 -0
  183. package/src/lib/server/memory-tiers.ts +16 -0
  184. package/src/lib/server/ollama-runtime.ts +58 -0
  185. package/src/lib/server/openclaw-deploy.test.ts +109 -1
  186. package/src/lib/server/openclaw-deploy.ts +557 -81
  187. package/src/lib/server/openclaw-gateway.test.ts +131 -0
  188. package/src/lib/server/openclaw-gateway.ts +10 -4
  189. package/src/lib/server/openclaw-health.test.ts +35 -0
  190. package/src/lib/server/openclaw-health.ts +215 -47
  191. package/src/lib/server/orchestrator-lg.ts +3 -2
  192. package/src/lib/server/orchestrator.ts +2 -0
  193. package/src/lib/server/plugins-advanced.test.ts +351 -0
  194. package/src/lib/server/plugins.ts +211 -6
  195. package/src/lib/server/project-context.ts +162 -0
  196. package/src/lib/server/project-utils.ts +150 -0
  197. package/src/lib/server/queue-advanced.test.ts +528 -0
  198. package/src/lib/server/queue-followups.test.ts +409 -2
  199. package/src/lib/server/queue-reconcile.test.ts +128 -0
  200. package/src/lib/server/queue.ts +527 -68
  201. package/src/lib/server/scheduler.ts +29 -1
  202. package/src/lib/server/session-note.test.ts +36 -0
  203. package/src/lib/server/session-note.ts +42 -0
  204. package/src/lib/server/session-run-manager.ts +83 -4
  205. package/src/lib/server/session-tools/canvas.ts +14 -12
  206. package/src/lib/server/session-tools/connector-inputs.test.ts +37 -0
  207. package/src/lib/server/session-tools/connector.test.ts +138 -0
  208. package/src/lib/server/session-tools/connector.ts +366 -54
  209. package/src/lib/server/session-tools/context.ts +17 -3
  210. package/src/lib/server/session-tools/crud.ts +484 -84
  211. package/src/lib/server/session-tools/delegate-fallback.test.ts +103 -0
  212. package/src/lib/server/session-tools/delegate-resume.test.ts +50 -0
  213. package/src/lib/server/session-tools/delegate.ts +102 -10
  214. package/src/lib/server/session-tools/discovery-approvals.test.ts +142 -0
  215. package/src/lib/server/session-tools/discovery.ts +80 -12
  216. package/src/lib/server/session-tools/file-normalize.test.ts +36 -0
  217. package/src/lib/server/session-tools/file.ts +43 -4
  218. package/src/lib/server/session-tools/human-loop.ts +35 -5
  219. package/src/lib/server/session-tools/index.ts +44 -9
  220. package/src/lib/server/session-tools/manage-connectors.test.ts +139 -0
  221. package/src/lib/server/session-tools/manage-schedules-advanced.test.ts +564 -0
  222. package/src/lib/server/session-tools/manage-schedules.test.ts +283 -0
  223. package/src/lib/server/session-tools/manage-tasks-advanced.test.ts +852 -0
  224. package/src/lib/server/session-tools/manage-tasks.test.ts +114 -0
  225. package/src/lib/server/session-tools/memory.test.ts +93 -0
  226. package/src/lib/server/session-tools/memory.ts +554 -75
  227. package/src/lib/server/session-tools/normalize-tool-args.ts +1 -1
  228. package/src/lib/server/session-tools/platform-access.test.ts +58 -0
  229. package/src/lib/server/session-tools/platform.ts +60 -19
  230. package/src/lib/server/session-tools/plugin-creator.ts +57 -1
  231. package/src/lib/server/session-tools/primitive-tools.test.ts +6 -0
  232. package/src/lib/server/session-tools/schedule.ts +6 -1
  233. package/src/lib/server/session-tools/shell-normalize.test.ts +25 -1
  234. package/src/lib/server/session-tools/shell.ts +22 -3
  235. package/src/lib/server/session-tools/wallet-tool.test.ts +254 -0
  236. package/src/lib/server/session-tools/wallet.ts +1374 -139
  237. package/src/lib/server/session-tools/web-inputs.test.ts +178 -0
  238. package/src/lib/server/session-tools/web.ts +621 -70
  239. package/src/lib/server/skill-discovery.ts +128 -0
  240. package/src/lib/server/skill-eligibility.test.ts +84 -0
  241. package/src/lib/server/skill-eligibility.ts +95 -0
  242. package/src/lib/server/skill-prompt-budget.test.ts +102 -0
  243. package/src/lib/server/skill-prompt-budget.ts +125 -0
  244. package/src/lib/server/skills-normalize.test.ts +54 -0
  245. package/src/lib/server/skills-normalize.ts +372 -26
  246. package/src/lib/server/solana.ts +214 -29
  247. package/src/lib/server/storage.ts +65 -36
  248. package/src/lib/server/stream-agent-chat.test.ts +437 -2
  249. package/src/lib/server/stream-agent-chat.ts +957 -79
  250. package/src/lib/server/system-events.ts +1 -1
  251. package/src/lib/server/tool-aliases.ts +2 -0
  252. package/src/lib/server/tool-capability-policy-advanced.test.ts +502 -0
  253. package/src/lib/server/tool-capability-policy.test.ts +24 -0
  254. package/src/lib/server/tool-capability-policy.ts +29 -1
  255. package/src/lib/server/tool-loop-detection.test.ts +105 -0
  256. package/src/lib/server/tool-loop-detection.ts +260 -0
  257. package/src/lib/server/tool-planning.test.ts +44 -0
  258. package/src/lib/server/tool-planning.ts +271 -0
  259. package/src/lib/server/wallet-execution.test.ts +198 -0
  260. package/src/lib/server/wallet-portfolio.test.ts +98 -0
  261. package/src/lib/server/wallet-portfolio.ts +724 -0
  262. package/src/lib/server/wallet-service.test.ts +57 -0
  263. package/src/lib/server/wallet-service.ts +213 -0
  264. package/src/lib/server/watch-jobs-advanced.test.ts +594 -0
  265. package/src/lib/server/watch-jobs.ts +17 -2
  266. package/src/lib/server/workspace-context.ts +111 -0
  267. package/src/lib/skill-save-payload.test.ts +39 -0
  268. package/src/lib/skill-save-payload.ts +37 -0
  269. package/src/lib/tasks.ts +28 -0
  270. package/src/lib/tool-definitions.ts +2 -1
  271. package/src/lib/tool-event-summary.test.ts +30 -0
  272. package/src/lib/tool-event-summary.ts +37 -0
  273. package/src/lib/validation/schemas.ts +1 -0
  274. package/src/lib/wallet-transactions.test.ts +75 -0
  275. package/src/lib/wallet-transactions.ts +43 -0
  276. package/src/lib/wallet.test.ts +17 -0
  277. package/src/lib/wallet.ts +183 -0
  278. package/src/proxy.test.ts +31 -0
  279. package/src/proxy.ts +34 -2
  280. package/src/stores/use-chat-store.ts +15 -1
  281. package/src/types/index.ts +249 -14
@@ -1,7 +1,14 @@
1
1
  import { describe, it } from 'node:test'
2
2
  import assert from 'node:assert/strict'
3
- import type { BoardTask } from '@/types'
4
- import { resolveTaskOriginConnectorFollowupTarget } from './queue'
3
+ import type { BoardTask, Session } from '@/types'
4
+ import {
5
+ applyTaskResumeStateToSession,
6
+ collectTaskConnectorFollowupTargets,
7
+ dequeueNextRunnableTask,
8
+ resolveTaskOriginConnectorFollowupTarget,
9
+ resolveTaskResumeContext,
10
+ resolveReusableTaskSessionId,
11
+ } from './queue'
5
12
 
6
13
  function makeTask(partial?: Partial<BoardTask> & { createdInSessionId?: string | null }): BoardTask {
7
14
  const now = Date.now()
@@ -18,12 +25,19 @@ function makeTask(partial?: Partial<BoardTask> & { createdInSessionId?: string |
18
25
  }
19
26
 
20
27
  type SessionFixtureMap = Record<string, {
28
+ connectorContext?: {
29
+ connectorId?: string
30
+ channelId?: string
31
+ threadId?: string
32
+ }
21
33
  messages: Array<{
22
34
  role: string
23
35
  text?: string
36
+ historyExcluded?: boolean
24
37
  source?: {
25
38
  connectorId?: string
26
39
  channelId?: string
40
+ threadId?: string
27
41
  }
28
42
  }>
29
43
  }>
@@ -221,4 +235,397 @@ describe('resolveTaskOriginConnectorFollowupTarget', () => {
221
235
  channelId: '447700900123@s.whatsapp.net',
222
236
  })
223
237
  })
238
+
239
+ it('prefers explicit task followup metadata over later thread traffic', () => {
240
+ const task = makeTask({
241
+ createdInSessionId: 'session-1',
242
+ followupConnectorId: 'conn-wa',
243
+ followupChannelId: '447700900111@s.whatsapp.net',
244
+ followupThreadId: 'thread-me',
245
+ })
246
+ const sessions = {
247
+ 'session-1': {
248
+ messages: [
249
+ {
250
+ role: 'user',
251
+ text: 'wife said hello',
252
+ source: {
253
+ connectorId: 'conn-wa',
254
+ channelId: '447700900222@s.whatsapp.net',
255
+ threadId: 'thread-wife',
256
+ },
257
+ },
258
+ ],
259
+ },
260
+ }
261
+ const connectors = {
262
+ 'conn-wa': {
263
+ id: 'conn-wa',
264
+ platform: 'whatsapp',
265
+ agentId: 'agent-a',
266
+ config: {},
267
+ },
268
+ }
269
+ const running = [
270
+ {
271
+ id: 'conn-wa',
272
+ platform: 'whatsapp',
273
+ agentId: 'agent-a',
274
+ supportsSend: true,
275
+ configuredTargets: [],
276
+ recentChannelId: '447700900222@s.whatsapp.net',
277
+ },
278
+ ]
279
+
280
+ const target = resolveTaskOriginConnectorFollowupTarget({
281
+ task,
282
+ sessions: sessions as SessionFixtureMap,
283
+ connectors,
284
+ running,
285
+ })
286
+
287
+ assert.deepEqual(target, {
288
+ connectorId: 'conn-wa',
289
+ channelId: '447700900111@s.whatsapp.net',
290
+ threadId: 'thread-me',
291
+ })
292
+ })
293
+
294
+ it('ignores mirrored connector transcript copies when resolving delayed followups', () => {
295
+ const task = makeTask({ createdInSessionId: 'session-main' })
296
+ const sessions = {
297
+ 'session-main': {
298
+ messages: [
299
+ {
300
+ role: 'user',
301
+ text: 'from me over whatsapp',
302
+ historyExcluded: true,
303
+ source: {
304
+ connectorId: 'conn-wa',
305
+ channelId: '447700900111@s.whatsapp.net',
306
+ },
307
+ },
308
+ {
309
+ role: 'user',
310
+ text: 'from wife over whatsapp later',
311
+ historyExcluded: true,
312
+ source: {
313
+ connectorId: 'conn-wa',
314
+ channelId: '447700900222@s.whatsapp.net',
315
+ },
316
+ },
317
+ ],
318
+ },
319
+ }
320
+ const connectors = {
321
+ 'conn-wa': {
322
+ id: 'conn-wa',
323
+ platform: 'whatsapp',
324
+ agentId: 'agent-a',
325
+ config: {
326
+ taskFollowups: 'true',
327
+ },
328
+ },
329
+ }
330
+ const running = [
331
+ {
332
+ id: 'conn-wa',
333
+ platform: 'whatsapp',
334
+ agentId: 'agent-a',
335
+ supportsSend: true,
336
+ configuredTargets: [],
337
+ recentChannelId: '447700900222@s.whatsapp.net',
338
+ },
339
+ ]
340
+
341
+ const target = resolveTaskOriginConnectorFollowupTarget({
342
+ task,
343
+ sessions: sessions as SessionFixtureMap,
344
+ connectors,
345
+ running,
346
+ })
347
+
348
+ assert.equal(target, null)
349
+ })
350
+ })
351
+
352
+ describe('collectTaskConnectorFollowupTargets', () => {
353
+ it('does not fall back to a connector recent channel when there is no explicit origin target', () => {
354
+ const task = makeTask({ createdInSessionId: 'session-main' })
355
+ const sessions = {
356
+ 'session-main': {
357
+ messages: [
358
+ {
359
+ role: 'user',
360
+ text: 'mirrored from me',
361
+ historyExcluded: true,
362
+ source: {
363
+ connectorId: 'conn-wa',
364
+ channelId: '447700900111@s.whatsapp.net',
365
+ },
366
+ },
367
+ ],
368
+ },
369
+ }
370
+ const connectors = {
371
+ 'conn-wa': {
372
+ id: 'conn-wa',
373
+ platform: 'whatsapp',
374
+ agentId: 'agent-a',
375
+ config: {
376
+ taskFollowups: 'true',
377
+ },
378
+ },
379
+ }
380
+ const running = [
381
+ {
382
+ id: 'conn-wa',
383
+ platform: 'whatsapp',
384
+ agentId: 'agent-a',
385
+ supportsSend: true,
386
+ configuredTargets: [],
387
+ recentChannelId: '447700900222@s.whatsapp.net',
388
+ },
389
+ ]
390
+
391
+ const targets = collectTaskConnectorFollowupTargets({
392
+ task,
393
+ sessions: sessions as SessionFixtureMap,
394
+ connectors,
395
+ running,
396
+ })
397
+
398
+ assert.deepEqual(targets, [])
399
+ })
400
+
401
+ it('uses only the origin target when both origin and a different recent channel exist', () => {
402
+ const task = makeTask({
403
+ createdInSessionId: 'session-origin',
404
+ followupConnectorId: 'conn-wa',
405
+ followupChannelId: '447700900111@s.whatsapp.net',
406
+ })
407
+ const sessions = {
408
+ 'session-origin': {
409
+ messages: [],
410
+ },
411
+ }
412
+ const connectors = {
413
+ 'conn-wa': {
414
+ id: 'conn-wa',
415
+ platform: 'whatsapp',
416
+ agentId: 'agent-a',
417
+ config: {
418
+ taskFollowups: 'true',
419
+ outboundJid: '447700900333@s.whatsapp.net',
420
+ },
421
+ },
422
+ }
423
+ const running = [
424
+ {
425
+ id: 'conn-wa',
426
+ platform: 'whatsapp',
427
+ agentId: 'agent-a',
428
+ supportsSend: true,
429
+ configuredTargets: [],
430
+ recentChannelId: '447700900222@s.whatsapp.net',
431
+ },
432
+ ]
433
+
434
+ const targets = collectTaskConnectorFollowupTargets({
435
+ task,
436
+ sessions: sessions as SessionFixtureMap,
437
+ connectors,
438
+ running,
439
+ })
440
+
441
+ assert.deepEqual(targets, [
442
+ {
443
+ connectorId: 'conn-wa',
444
+ channelId: '447700900111@s.whatsapp.net',
445
+ },
446
+ ])
447
+ })
448
+
449
+ it('uses configured outbound targets for generic task followups', () => {
450
+ const task = makeTask({ createdInSessionId: 'session-main' })
451
+ const sessions = {
452
+ 'session-main': {
453
+ messages: [],
454
+ },
455
+ }
456
+ const connectors = {
457
+ 'conn-wa': {
458
+ id: 'conn-wa',
459
+ platform: 'whatsapp',
460
+ agentId: 'agent-a',
461
+ config: {
462
+ taskFollowups: 'true',
463
+ outboundJid: '+44 7700 900333',
464
+ },
465
+ },
466
+ }
467
+ const running = [
468
+ {
469
+ id: 'conn-wa',
470
+ platform: 'whatsapp',
471
+ agentId: 'agent-a',
472
+ supportsSend: true,
473
+ configuredTargets: [],
474
+ recentChannelId: '447700900222@s.whatsapp.net',
475
+ },
476
+ ]
477
+
478
+ const targets = collectTaskConnectorFollowupTargets({
479
+ task,
480
+ sessions: sessions as SessionFixtureMap,
481
+ connectors,
482
+ running,
483
+ })
484
+
485
+ assert.deepEqual(targets, [
486
+ {
487
+ connectorId: 'conn-wa',
488
+ channelId: '447700900333@s.whatsapp.net',
489
+ },
490
+ ])
491
+ })
492
+ })
493
+
494
+ describe('task resume context', () => {
495
+ it('falls back to delegated parent task resume handles for follow-up work', () => {
496
+ const parent = makeTask({
497
+ id: 'task-parent',
498
+ title: 'Parent task',
499
+ codexResumeId: 'codex-thread-123',
500
+ geminiResumeId: 'gemini-session-123',
501
+ sessionId: 'session-parent',
502
+ })
503
+ const child = makeTask({
504
+ id: 'task-child',
505
+ title: 'Child task',
506
+ delegatedFromTaskId: 'task-parent',
507
+ })
508
+
509
+ const context = resolveTaskResumeContext(child, {
510
+ [parent.id]: parent,
511
+ [child.id]: child,
512
+ })
513
+
514
+ assert.ok(context)
515
+ assert.equal(context?.source, 'delegated_from_task')
516
+ assert.equal(context?.sourceTaskId, 'task-parent')
517
+ assert.equal(context?.sourceSessionId, 'session-parent')
518
+ assert.equal(context?.resume.codexThreadId, 'codex-thread-123')
519
+ assert.equal(context?.resume.delegateResumeIds.gemini, 'gemini-session-123')
520
+ })
521
+
522
+ it('hydrates task execution sessions with stored resume state', () => {
523
+ const session = {
524
+ id: 'session-task',
525
+ name: 'Task session',
526
+ cwd: process.cwd(),
527
+ user: 'system',
528
+ provider: 'codex-cli',
529
+ model: 'gpt-5-codex',
530
+ claudeSessionId: null,
531
+ codexThreadId: null,
532
+ opencodeSessionId: null,
533
+ delegateResumeIds: { claudeCode: null, codex: null, opencode: null, gemini: null },
534
+ messages: [],
535
+ createdAt: Date.now(),
536
+ lastActiveAt: Date.now(),
537
+ sessionType: 'human',
538
+ agentId: 'agent-a',
539
+ parentSessionId: null,
540
+ plugins: ['delegate'],
541
+ } satisfies Session
542
+
543
+ const changed = applyTaskResumeStateToSession(session, {
544
+ claudeSessionId: 'claude-resume-1',
545
+ codexThreadId: 'codex-resume-1',
546
+ opencodeSessionId: 'opencode-resume-1',
547
+ delegateResumeIds: {
548
+ claudeCode: 'claude-resume-1',
549
+ codex: 'codex-resume-1',
550
+ opencode: 'opencode-resume-1',
551
+ gemini: 'gemini-resume-1',
552
+ },
553
+ })
554
+
555
+ assert.equal(changed, true)
556
+ assert.equal(session.claudeSessionId, 'claude-resume-1')
557
+ assert.equal(session.codexThreadId, 'codex-resume-1')
558
+ assert.equal(session.opencodeSessionId, 'opencode-resume-1')
559
+ assert.equal(session.delegateResumeIds?.gemini, 'gemini-resume-1')
560
+ })
561
+ })
562
+
563
+ describe('dequeueNextRunnableTask', () => {
564
+ it('leaves blocked queued tasks in place until their dependencies are completed', () => {
565
+ const source = makeTask({
566
+ id: 'task-source',
567
+ title: 'Source task',
568
+ status: 'running',
569
+ })
570
+ const followup = makeTask({
571
+ id: 'task-followup',
572
+ title: 'Follow-up task',
573
+ status: 'queued',
574
+ blockedBy: ['task-source'],
575
+ })
576
+ const queue = ['task-followup']
577
+
578
+ const selectedWhileBlocked = dequeueNextRunnableTask(queue, {
579
+ [source.id]: source,
580
+ [followup.id]: followup,
581
+ })
582
+
583
+ assert.equal(selectedWhileBlocked, null)
584
+ assert.deepEqual(queue, ['task-followup'])
585
+
586
+ source.status = 'completed'
587
+ const selectedAfterUnblock = dequeueNextRunnableTask(queue, {
588
+ [source.id]: source,
589
+ [followup.id]: followup,
590
+ })
591
+
592
+ assert.equal(selectedAfterUnblock, 'task-followup')
593
+ assert.deepEqual(queue, [])
594
+ })
595
+ })
596
+
597
+ describe('resolveReusableTaskSessionId', () => {
598
+ it('reuses the completed dependency session for continuation tasks once it exists', () => {
599
+ const source = makeTask({
600
+ id: 'task-source',
601
+ title: 'Source task',
602
+ status: 'completed',
603
+ sessionId: 'session-source',
604
+ checkpoint: {
605
+ lastSessionId: 'session-source',
606
+ updatedAt: Date.now(),
607
+ },
608
+ })
609
+ const followup = makeTask({
610
+ id: 'task-followup',
611
+ title: 'Follow-up task',
612
+ status: 'queued',
613
+ blockedBy: ['task-source'],
614
+ })
615
+
616
+ const sessionId = resolveReusableTaskSessionId(
617
+ followup,
618
+ {
619
+ [source.id]: source,
620
+ [followup.id]: followup,
621
+ },
622
+ {
623
+ 'session-source': {
624
+ messages: [],
625
+ },
626
+ } as SessionFixtureMap,
627
+ )
628
+
629
+ assert.equal(sessionId, 'session-source')
630
+ })
224
631
  })
@@ -0,0 +1,128 @@
1
+ import assert from 'node:assert/strict'
2
+ import fs from 'node:fs'
3
+ import os from 'node:os'
4
+ import path from 'node:path'
5
+ import { spawnSync } from 'node:child_process'
6
+ import { describe, it } from 'node:test'
7
+
8
+ const repoRoot = path.resolve(path.dirname(new URL(import.meta.url).pathname), '../../..')
9
+
10
+ function runWithTempDataDir(script: string) {
11
+ const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'swarmclaw-queue-reconcile-'))
12
+ try {
13
+ const result = spawnSync(process.execPath, ['--import', 'tsx', '--input-type=module', '--eval', script], {
14
+ cwd: repoRoot,
15
+ env: {
16
+ ...process.env,
17
+ DATA_DIR: path.join(tempDir, 'data'),
18
+ WORKSPACE_DIR: path.join(tempDir, 'workspace'),
19
+ SWARMCLAW_BUILD_MODE: '1',
20
+ },
21
+ encoding: 'utf-8',
22
+ })
23
+ assert.equal(result.status, 0, result.stderr || result.stdout || 'subprocess failed')
24
+ const lines = (result.stdout || '')
25
+ .trim()
26
+ .split('\n')
27
+ .map((line) => line.trim())
28
+ .filter(Boolean)
29
+ const jsonLine = [...lines].reverse().find((line) => line.startsWith('{'))
30
+ return JSON.parse(jsonLine || '{}')
31
+ } finally {
32
+ fs.rmSync(tempDir, { recursive: true, force: true })
33
+ }
34
+ }
35
+
36
+ describe('reconcileFinishedRunningTasks', () => {
37
+ it('finalizes a completed one-off scheduled task from its finished session and deletes the schedule', () => {
38
+ const output = runWithTempDataDir(`
39
+ const storageMod = await import('./src/lib/server/storage.ts')
40
+ const queueMod = await import('./src/lib/server/queue.ts')
41
+ const storage = storageMod.default || storageMod
42
+ const queue = queueMod.default || queueMod
43
+
44
+ const now = Date.now()
45
+ const workspace = process.env.WORKSPACE_DIR
46
+ storage.saveAgents({
47
+ agent_birthday: {
48
+ id: 'agent_birthday',
49
+ name: 'Birthday Bot',
50
+ description: '',
51
+ systemPrompt: '',
52
+ provider: 'openai',
53
+ model: 'gpt-test',
54
+ threadSessionId: null,
55
+ createdAt: now,
56
+ updatedAt: now,
57
+ },
58
+ })
59
+ storage.saveSessions({
60
+ 'session-birthday': {
61
+ id: 'session-birthday',
62
+ name: 'Birthday Run',
63
+ cwd: workspace,
64
+ user: 'tester',
65
+ provider: 'openai',
66
+ model: 'gpt-test',
67
+ claudeSessionId: null,
68
+ messages: [
69
+ {
70
+ role: 'assistant',
71
+ text: 'Happy birthday. I sent a WhatsApp follow-up to the user directly and confirmed delivery with message id 3EB0B7262FF68B7BD261D4.',
72
+ time: now,
73
+ },
74
+ ],
75
+ createdAt: now - 10_000,
76
+ lastActiveAt: now,
77
+ active: false,
78
+ currentRunId: null,
79
+ heartbeatEnabled: true,
80
+ },
81
+ })
82
+ storage.saveSchedules({
83
+ 'schedule-birthday': {
84
+ id: 'schedule-birthday',
85
+ name: 'Birthday Reminder',
86
+ scheduleType: 'once',
87
+ status: 'completed',
88
+ agentId: 'agent_birthday',
89
+ createdByAgentId: 'agent_birthday',
90
+ createdAt: now - 20_000,
91
+ updatedAt: now - 5_000,
92
+ },
93
+ })
94
+ storage.saveTasks({
95
+ 'task-birthday': {
96
+ id: 'task-birthday',
97
+ title: 'Birthday follow-up',
98
+ description: 'Wish me happy birthday tomorrow over WhatsApp.',
99
+ status: 'running',
100
+ agentId: 'agent_birthday',
101
+ createdAt: now - 20_000,
102
+ updatedAt: now - 5_000,
103
+ startedAt: now - 15_000,
104
+ sessionId: 'session-birthday',
105
+ sourceType: 'schedule',
106
+ sourceScheduleId: 'schedule-birthday',
107
+ sourceScheduleName: 'Birthday Reminder',
108
+ maxAttempts: 3,
109
+ retryBackoffSec: 30,
110
+ },
111
+ })
112
+
113
+ const result = queue.reconcileFinishedRunningTasks()
114
+ console.log(JSON.stringify({
115
+ result,
116
+ task: storage.loadTasks()['task-birthday'],
117
+ schedule: storage.loadSchedules()['schedule-birthday'] || null,
118
+ session: storage.loadSessions()['session-birthday'],
119
+ }))
120
+ `)
121
+
122
+ assert.equal(output.result.reconciled, 1)
123
+ assert.equal(output.task.status, 'completed')
124
+ assert.equal(output.schedule, null)
125
+ assert.equal(output.session.heartbeatEnabled, false)
126
+ assert.match(output.task.result, /WhatsApp follow-up/i)
127
+ })
128
+ })