@swarmclawai/swarmclaw 0.7.1 → 0.7.3

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 (237) hide show
  1. package/README.md +155 -150
  2. package/package.json +1 -1
  3. package/src/app/api/agents/[id]/route.ts +26 -0
  4. package/src/app/api/agents/[id]/thread/route.ts +37 -9
  5. package/src/app/api/agents/route.ts +13 -2
  6. package/src/app/api/auth/route.ts +76 -7
  7. package/src/app/api/chatrooms/[id]/chat/route.ts +7 -2
  8. package/src/app/api/{sessions → chats}/[id]/browser/route.ts +5 -1
  9. package/src/app/api/{sessions → chats}/[id]/chat/route.ts +7 -3
  10. package/src/app/api/{sessions → chats}/[id]/checkpoints/route.ts +1 -1
  11. package/src/app/api/chats/[id]/main-loop/route.ts +13 -0
  12. package/src/app/api/{sessions → chats}/[id]/messages/route.ts +19 -13
  13. package/src/app/api/{sessions → chats}/[id]/restore/route.ts +1 -1
  14. package/src/app/api/{sessions → chats}/[id]/route.ts +22 -52
  15. package/src/app/api/{sessions → chats}/[id]/stop/route.ts +6 -1
  16. package/src/app/api/{sessions → chats}/route.ts +21 -7
  17. package/src/app/api/connectors/[id]/doctor/route.ts +26 -0
  18. package/src/app/api/connectors/doctor/route.ts +13 -0
  19. package/src/app/api/files/open/route.ts +16 -14
  20. package/src/app/api/memory/maintenance/route.ts +11 -1
  21. package/src/app/api/openclaw/agent-files/route.ts +27 -4
  22. package/src/app/api/openclaw/skills/route.ts +11 -3
  23. package/src/app/api/plugins/dependencies/route.ts +24 -0
  24. package/src/app/api/plugins/install/route.ts +15 -92
  25. package/src/app/api/plugins/route.ts +6 -26
  26. package/src/app/api/plugins/settings/route.ts +40 -0
  27. package/src/app/api/plugins/ui/route.ts +1 -0
  28. package/src/app/api/settings/route.ts +49 -7
  29. package/src/app/api/tasks/[id]/route.ts +15 -6
  30. package/src/app/api/tasks/bulk/route.ts +2 -2
  31. package/src/app/api/tasks/route.ts +9 -4
  32. package/src/app/api/usage/route.ts +30 -0
  33. package/src/app/api/webhooks/[id]/route.ts +8 -1
  34. package/src/app/page.tsx +9 -2
  35. package/src/cli/index.js +39 -33
  36. package/src/cli/index.ts +43 -49
  37. package/src/cli/spec.js +29 -27
  38. package/src/components/agents/agent-card.tsx +16 -13
  39. package/src/components/agents/agent-chat-list.tsx +104 -4
  40. package/src/components/agents/agent-list.tsx +54 -22
  41. package/src/components/agents/agent-sheet.tsx +209 -18
  42. package/src/components/agents/cron-job-form.tsx +3 -3
  43. package/src/components/agents/inspector-panel.tsx +110 -50
  44. package/src/components/auth/access-key-gate.tsx +36 -97
  45. package/src/components/auth/setup-wizard.tsx +5 -38
  46. package/src/components/chat/chat-area.tsx +39 -27
  47. package/src/components/{sessions/session-card.tsx → chat/chat-card.tsx} +7 -23
  48. package/src/components/chat/chat-header.tsx +299 -314
  49. package/src/components/{sessions/session-list.tsx → chat/chat-list.tsx} +11 -14
  50. package/src/components/chat/chat-tool-toggles.tsx +26 -17
  51. package/src/components/chat/checkpoint-timeline.tsx +4 -4
  52. package/src/components/chat/message-bubble.tsx +4 -1
  53. package/src/components/chat/message-list.tsx +5 -3
  54. package/src/components/chat/session-debug-panel.tsx +1 -1
  55. package/src/components/chat/tool-request-banner.tsx +3 -3
  56. package/src/components/chatrooms/agent-hover-card.tsx +3 -3
  57. package/src/components/chatrooms/chatroom-tool-request-banner.tsx +2 -2
  58. package/src/components/chatrooms/chatroom-view.tsx +347 -205
  59. package/src/components/connectors/connector-list.tsx +265 -127
  60. package/src/components/connectors/connector-sheet.tsx +218 -1
  61. package/src/components/home/home-view.tsx +129 -5
  62. package/src/components/layout/app-layout.tsx +392 -182
  63. package/src/components/layout/mobile-header.tsx +26 -8
  64. package/src/components/plugins/plugin-list.tsx +487 -254
  65. package/src/components/plugins/plugin-sheet.tsx +236 -13
  66. package/src/components/projects/project-detail.tsx +183 -0
  67. package/src/components/settings/gateway-connection-panel.tsx +1 -1
  68. package/src/components/shared/agent-picker-list.tsx +2 -2
  69. package/src/components/shared/command-palette.tsx +111 -25
  70. package/src/components/shared/settings/plugin-manager.tsx +20 -4
  71. package/src/components/shared/settings/section-capability-policy.tsx +105 -0
  72. package/src/components/shared/settings/section-heartbeat.tsx +78 -1
  73. package/src/components/shared/settings/section-orchestrator.tsx +3 -3
  74. package/src/components/shared/settings/section-providers.tsx +1 -1
  75. package/src/components/shared/settings/section-runtime-loop.tsx +5 -5
  76. package/src/components/shared/settings/section-secrets.tsx +6 -6
  77. package/src/components/shared/settings/section-user-preferences.tsx +1 -1
  78. package/src/components/shared/settings/section-voice.tsx +5 -1
  79. package/src/components/shared/settings/section-web-search.tsx +10 -2
  80. package/src/components/shared/settings/settings-page.tsx +244 -56
  81. package/src/components/tasks/approvals-panel.tsx +205 -18
  82. package/src/components/tasks/task-board.tsx +242 -46
  83. package/src/components/usage/metrics-dashboard.tsx +147 -1
  84. package/src/components/wallets/wallet-panel.tsx +17 -5
  85. package/src/components/webhooks/webhook-sheet.tsx +8 -8
  86. package/src/lib/auth.ts +17 -0
  87. package/src/lib/chat-streaming-state.test.ts +108 -0
  88. package/src/lib/chat-streaming-state.ts +108 -0
  89. package/src/lib/chat.ts +1 -1
  90. package/src/lib/{sessions.ts → chats.ts} +28 -18
  91. package/src/lib/openclaw-agent-id.test.ts +14 -0
  92. package/src/lib/openclaw-agent-id.ts +31 -0
  93. package/src/lib/providers/claude-cli.ts +1 -1
  94. package/src/lib/server/agent-assignment.test.ts +112 -0
  95. package/src/lib/server/agent-assignment.ts +169 -0
  96. package/src/lib/server/approval-connector-notify.test.ts +253 -0
  97. package/src/lib/server/approvals-auto-approve.test.ts +205 -0
  98. package/src/lib/server/approvals.ts +483 -75
  99. package/src/lib/server/autonomy-runtime.test.ts +341 -0
  100. package/src/lib/server/browser-state.test.ts +118 -0
  101. package/src/lib/server/browser-state.ts +123 -0
  102. package/src/lib/server/build-llm.test.ts +36 -0
  103. package/src/lib/server/build-llm.ts +11 -4
  104. package/src/lib/server/builtin-plugins.ts +34 -0
  105. package/src/lib/server/capability-router.ts +10 -8
  106. package/src/lib/server/chat-execution-heartbeat.test.ts +40 -0
  107. package/src/lib/server/chat-execution-tool-events.test.ts +134 -0
  108. package/src/lib/server/chat-execution.ts +285 -165
  109. package/src/lib/server/chatroom-health.test.ts +26 -0
  110. package/src/lib/server/chatroom-health.ts +2 -3
  111. package/src/lib/server/chatroom-helpers.test.ts +67 -2
  112. package/src/lib/server/chatroom-helpers.ts +48 -8
  113. package/src/lib/server/connectors/discord.ts +175 -11
  114. package/src/lib/server/connectors/doctor.test.ts +80 -0
  115. package/src/lib/server/connectors/doctor.ts +116 -0
  116. package/src/lib/server/connectors/manager.ts +948 -112
  117. package/src/lib/server/connectors/policy.test.ts +222 -0
  118. package/src/lib/server/connectors/policy.ts +452 -0
  119. package/src/lib/server/connectors/slack.ts +188 -9
  120. package/src/lib/server/connectors/telegram.ts +65 -15
  121. package/src/lib/server/connectors/thread-context.test.ts +44 -0
  122. package/src/lib/server/connectors/thread-context.ts +72 -0
  123. package/src/lib/server/connectors/types.ts +41 -11
  124. package/src/lib/server/cost.ts +34 -1
  125. package/src/lib/server/daemon-state.ts +61 -3
  126. package/src/lib/server/data-dir.ts +13 -0
  127. package/src/lib/server/delegation-jobs.test.ts +140 -0
  128. package/src/lib/server/delegation-jobs.ts +248 -0
  129. package/src/lib/server/document-utils.test.ts +47 -0
  130. package/src/lib/server/document-utils.ts +397 -0
  131. package/src/lib/server/heartbeat-service.ts +14 -40
  132. package/src/lib/server/heartbeat-source.test.ts +22 -0
  133. package/src/lib/server/heartbeat-source.ts +7 -0
  134. package/src/lib/server/identity-continuity.test.ts +77 -0
  135. package/src/lib/server/identity-continuity.ts +127 -0
  136. package/src/lib/server/mailbox-utils.ts +347 -0
  137. package/src/lib/server/main-agent-loop.ts +28 -1103
  138. package/src/lib/server/memory-db.ts +4 -6
  139. package/src/lib/server/memory-tiers.ts +40 -0
  140. package/src/lib/server/openclaw-agent-resolver.test.ts +70 -0
  141. package/src/lib/server/openclaw-agent-resolver.ts +128 -0
  142. package/src/lib/server/openclaw-exec-config.ts +5 -6
  143. package/src/lib/server/openclaw-skills-normalize.test.ts +56 -0
  144. package/src/lib/server/openclaw-skills-normalize.ts +136 -0
  145. package/src/lib/server/openclaw-sync.ts +3 -2
  146. package/src/lib/server/orchestrator-lg.ts +20 -9
  147. package/src/lib/server/orchestrator.ts +7 -7
  148. package/src/lib/server/playwright-proxy.mjs +27 -3
  149. package/src/lib/server/plugins.test.ts +207 -0
  150. package/src/lib/server/plugins.ts +927 -66
  151. package/src/lib/server/provider-health.ts +38 -6
  152. package/src/lib/server/queue.ts +13 -28
  153. package/src/lib/server/scheduler.ts +2 -0
  154. package/src/lib/server/session-archive-memory.test.ts +85 -0
  155. package/src/lib/server/session-archive-memory.ts +230 -0
  156. package/src/lib/server/session-mailbox.ts +8 -18
  157. package/src/lib/server/session-reset-policy.test.ts +99 -0
  158. package/src/lib/server/session-reset-policy.ts +311 -0
  159. package/src/lib/server/session-run-manager.ts +33 -82
  160. package/src/lib/server/session-tools/autonomy-tools.test.ts +105 -0
  161. package/src/lib/server/session-tools/calendar.ts +366 -0
  162. package/src/lib/server/session-tools/canvas.ts +1 -1
  163. package/src/lib/server/session-tools/chatroom.ts +4 -2
  164. package/src/lib/server/session-tools/connector.ts +114 -10
  165. package/src/lib/server/session-tools/context.ts +21 -5
  166. package/src/lib/server/session-tools/crawl.ts +447 -0
  167. package/src/lib/server/session-tools/crud.ts +74 -28
  168. package/src/lib/server/session-tools/delegate-fallback.test.ts +219 -0
  169. package/src/lib/server/session-tools/delegate.ts +497 -24
  170. package/src/lib/server/session-tools/discovery.ts +24 -6
  171. package/src/lib/server/session-tools/document.ts +283 -0
  172. package/src/lib/server/session-tools/edit_file.ts +4 -2
  173. package/src/lib/server/session-tools/email.ts +320 -0
  174. package/src/lib/server/session-tools/extract.ts +137 -0
  175. package/src/lib/server/session-tools/file-normalize.test.ts +93 -0
  176. package/src/lib/server/session-tools/file-send.test.ts +84 -1
  177. package/src/lib/server/session-tools/file.ts +241 -25
  178. package/src/lib/server/session-tools/git.ts +1 -1
  179. package/src/lib/server/session-tools/http.ts +1 -1
  180. package/src/lib/server/session-tools/human-loop.ts +227 -0
  181. package/src/lib/server/session-tools/image-gen.ts +380 -0
  182. package/src/lib/server/session-tools/index.ts +130 -50
  183. package/src/lib/server/session-tools/mailbox.ts +276 -0
  184. package/src/lib/server/session-tools/memory.ts +172 -3
  185. package/src/lib/server/session-tools/monitor.ts +151 -8
  186. package/src/lib/server/session-tools/normalize-tool-args.ts +17 -14
  187. package/src/lib/server/session-tools/openclaw-nodes.ts +1 -1
  188. package/src/lib/server/session-tools/openclaw-workspace.ts +1 -1
  189. package/src/lib/server/session-tools/platform-normalize.test.ts +142 -0
  190. package/src/lib/server/session-tools/platform.ts +148 -7
  191. package/src/lib/server/session-tools/plugin-creator.ts +89 -26
  192. package/src/lib/server/session-tools/primitive-tools.test.ts +257 -0
  193. package/src/lib/server/session-tools/replicate.ts +301 -0
  194. package/src/lib/server/session-tools/sample-ui.ts +1 -1
  195. package/src/lib/server/session-tools/sandbox.ts +4 -2
  196. package/src/lib/server/session-tools/schedule.ts +24 -12
  197. package/src/lib/server/session-tools/session-info.ts +43 -7
  198. package/src/lib/server/session-tools/session-tools-wiring.test.ts +31 -17
  199. package/src/lib/server/session-tools/shell.ts +5 -2
  200. package/src/lib/server/session-tools/subagent.ts +194 -28
  201. package/src/lib/server/session-tools/table.ts +587 -0
  202. package/src/lib/server/session-tools/wallet.ts +42 -12
  203. package/src/lib/server/session-tools/web-browser-config.test.ts +39 -0
  204. package/src/lib/server/session-tools/web.ts +926 -91
  205. package/src/lib/server/storage.ts +255 -16
  206. package/src/lib/server/stream-agent-chat.ts +116 -268
  207. package/src/lib/server/structured-extract.test.ts +72 -0
  208. package/src/lib/server/structured-extract.ts +373 -0
  209. package/src/lib/server/task-mention.test.ts +16 -2
  210. package/src/lib/server/task-mention.ts +61 -10
  211. package/src/lib/server/tool-aliases.ts +66 -18
  212. package/src/lib/server/tool-capability-policy.test.ts +9 -9
  213. package/src/lib/server/tool-capability-policy.ts +38 -27
  214. package/src/lib/server/tool-retry.ts +2 -0
  215. package/src/lib/server/watch-jobs.test.ts +173 -0
  216. package/src/lib/server/watch-jobs.ts +532 -0
  217. package/src/lib/server/ws-hub.ts +5 -3
  218. package/src/lib/tool-definitions.ts +4 -0
  219. package/src/lib/validation/schemas.test.ts +26 -0
  220. package/src/lib/validation/schemas.ts +10 -1
  221. package/src/lib/ws-client.ts +14 -12
  222. package/src/proxy.ts +5 -5
  223. package/src/stores/use-app-store.ts +5 -11
  224. package/src/stores/use-chat-store.ts +38 -9
  225. package/src/types/index.ts +352 -47
  226. package/src/app/api/sessions/[id]/main-loop/route.ts +0 -94
  227. package/src/components/sessions/new-session-sheet.tsx +0 -253
  228. package/src/lib/server/main-session.ts +0 -24
  229. package/src/lib/server/session-run-manager.test.ts +0 -23
  230. /package/src/app/api/{sessions → chats}/[id]/clear/route.ts +0 -0
  231. /package/src/app/api/{sessions → chats}/[id]/deploy/route.ts +0 -0
  232. /package/src/app/api/{sessions → chats}/[id]/devserver/route.ts +0 -0
  233. /package/src/app/api/{sessions → chats}/[id]/edit-resend/route.ts +0 -0
  234. /package/src/app/api/{sessions → chats}/[id]/fork/route.ts +0 -0
  235. /package/src/app/api/{sessions → chats}/[id]/mailbox/route.ts +0 -0
  236. /package/src/app/api/{sessions → chats}/[id]/retry/route.ts +0 -0
  237. /package/src/app/api/{sessions → chats}/heartbeat/route.ts +0 -0
@@ -0,0 +1,219 @@
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 writeExecutable(dir: string, name: string, source: string) {
11
+ const filePath = path.join(dir, name)
12
+ fs.writeFileSync(filePath, source, { mode: 0o755 })
13
+ return filePath
14
+ }
15
+
16
+ function runWithFakeDelegates(script: string) {
17
+ const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'swarmclaw-delegate-fallback-'))
18
+ try {
19
+ writeExecutable(tempDir, 'claude', `#!/bin/sh
20
+ if [ "$1" = "auth" ] && [ "$2" = "status" ]; then
21
+ echo '{"loggedIn":false}'
22
+ exit 1
23
+ fi
24
+ echo "unexpected claude invocation" >&2
25
+ exit 2
26
+ `)
27
+
28
+ writeExecutable(tempDir, 'codex', `#!/bin/sh
29
+ if [ "$1" = "login" ] && [ "$2" = "status" ]; then
30
+ echo 'logged in'
31
+ exit 0
32
+ fi
33
+ if [ "$1" = "exec" ]; then
34
+ cat >/dev/null
35
+ printf '%s\\n' '{"type":"item.completed","item":{"type":"agent_message","text":"codex fallback ok"}}'
36
+ exit 0
37
+ fi
38
+ echo "unexpected codex invocation" >&2
39
+ exit 2
40
+ `)
41
+
42
+ const result = spawnSync(process.execPath, ['--import', 'tsx', '--input-type=module', '--eval', script], {
43
+ cwd: repoRoot,
44
+ env: {
45
+ ...process.env,
46
+ PATH: `${tempDir}:${process.env.PATH || ''}`,
47
+ },
48
+ encoding: 'utf-8',
49
+ })
50
+ assert.equal(result.status, 0, result.stderr || result.stdout || 'subprocess failed')
51
+ const lines = (result.stdout || '')
52
+ .trim()
53
+ .split('\n')
54
+ .map((line) => line.trim())
55
+ .filter(Boolean)
56
+ const jsonLine = [...lines].reverse().find((line) => line.startsWith('{') || line.startsWith('['))
57
+ return JSON.parse(jsonLine || '{}')
58
+ } finally {
59
+ fs.rmSync(tempDir, { recursive: true, force: true })
60
+ }
61
+ }
62
+
63
+ describe('delegate fallback', () => {
64
+ it('falls back to another backend when Claude Code is unavailable', () => {
65
+ const output = runWithFakeDelegates(`
66
+ const mod = await import('./src/lib/server/session-tools/delegate.ts')
67
+ const { buildDelegateTools } = mod.default || mod['module.exports'] || mod
68
+
69
+ const tools = buildDelegateTools({
70
+ cwd: process.cwd(),
71
+ ctx: { sessionId: 'session-test', agentId: 'agent-test', platformAssignScope: 'self' },
72
+ hasPlugin: (name) => name === 'delegate',
73
+ hasTool: (name) => name === 'delegate',
74
+ cleanupFns: [],
75
+ commandTimeoutMs: 5000,
76
+ claudeTimeoutMs: 5000,
77
+ cliProcessTimeoutMs: 5000,
78
+ persistDelegateResumeId: () => {},
79
+ readStoredDelegateResumeId: () => null,
80
+ resolveCurrentSession: () => null,
81
+ activePlugins: ['delegate'],
82
+ })
83
+
84
+ const delegateTool = tools.find((tool) => tool.name === 'delegate')
85
+ const raw = await delegateTool.invoke({ task: 'write a helper', backend: 'claude' })
86
+ console.log(raw)
87
+ `)
88
+
89
+ assert.equal(output.backend, 'codex')
90
+ assert.equal(output.status, 'completed')
91
+ assert.match(String(output.response || ''), /codex fallback ok/i)
92
+ })
93
+
94
+ it('accepts wrapped function-call payloads with tool_name aliases', () => {
95
+ const output = runWithFakeDelegates(`
96
+ const mod = await import('./src/lib/server/session-tools/delegate.ts')
97
+ const { buildDelegateTools } = mod.default || mod['module.exports'] || mod
98
+
99
+ const tools = buildDelegateTools({
100
+ cwd: process.cwd(),
101
+ ctx: { sessionId: 'session-test', agentId: 'agent-test', platformAssignScope: 'self' },
102
+ hasPlugin: (name) => name === 'delegate',
103
+ hasTool: (name) => name === 'delegate',
104
+ cleanupFns: [],
105
+ commandTimeoutMs: 5000,
106
+ claudeTimeoutMs: 5000,
107
+ cliProcessTimeoutMs: 5000,
108
+ persistDelegateResumeId: () => {},
109
+ readStoredDelegateResumeId: () => null,
110
+ resolveCurrentSession: () => null,
111
+ activePlugins: ['delegate'],
112
+ })
113
+
114
+ const delegateTool = tools.find((tool) => tool.name === 'delegate')
115
+ const raw = await delegateTool.invoke({
116
+ input: JSON.stringify({
117
+ function: 'delegate',
118
+ parameters: {
119
+ tool_name: 'Claude Code',
120
+ parameters: {
121
+ task: 'Create a proof file',
122
+ },
123
+ },
124
+ }),
125
+ })
126
+ console.log(raw)
127
+ `)
128
+
129
+ assert.equal(output.backend, 'codex')
130
+ assert.equal(output.status, 'completed')
131
+ assert.match(String(output.response || ''), /codex fallback ok/i)
132
+ })
133
+
134
+ it('synthesizes a delegated task from write-style payloads', () => {
135
+ const output = runWithFakeDelegates(`
136
+ const mod = await import('./src/lib/server/session-tools/delegate.ts')
137
+ const { buildDelegateTools } = mod.default || mod['module.exports'] || mod
138
+
139
+ const tools = buildDelegateTools({
140
+ cwd: process.cwd(),
141
+ ctx: { sessionId: 'session-test', agentId: 'agent-test', platformAssignScope: 'self' },
142
+ hasPlugin: (name) => name === 'delegate',
143
+ hasTool: (name) => name === 'delegate',
144
+ cleanupFns: [],
145
+ commandTimeoutMs: 5000,
146
+ claudeTimeoutMs: 5000,
147
+ cliProcessTimeoutMs: 5000,
148
+ persistDelegateResumeId: () => {},
149
+ readStoredDelegateResumeId: () => null,
150
+ resolveCurrentSession: () => null,
151
+ activePlugins: ['delegate'],
152
+ })
153
+
154
+ const delegateTool = tools.find((tool) => tool.name === 'delegate')
155
+ const raw = await delegateTool.invoke({
156
+ input: JSON.stringify({
157
+ action: 'write',
158
+ target: 'delegate-proof.md',
159
+ content: 'Proof content',
160
+ }),
161
+ })
162
+ console.log(raw)
163
+ `)
164
+
165
+ assert.equal(output.backend, 'codex')
166
+ assert.equal(output.status, 'completed')
167
+ assert.match(String(output.response || ''), /codex fallback ok/i)
168
+ })
169
+
170
+ it('synthesizes a delegated task for action=start payloads that only provide files', () => {
171
+ const output = runWithFakeDelegates(`
172
+ const mod = await import('./src/lib/server/session-tools/delegate.ts')
173
+ const { buildDelegateTools } = mod.default || mod['module.exports'] || mod
174
+
175
+ const tools = buildDelegateTools({
176
+ cwd: process.cwd(),
177
+ ctx: { sessionId: 'session-test', agentId: 'agent-test', platformAssignScope: 'self' },
178
+ hasPlugin: (name) => name === 'delegate',
179
+ hasTool: (name) => name === 'delegate',
180
+ cleanupFns: [],
181
+ commandTimeoutMs: 5000,
182
+ claudeTimeoutMs: 5000,
183
+ cliProcessTimeoutMs: 5000,
184
+ persistDelegateResumeId: () => {},
185
+ readStoredDelegateResumeId: () => null,
186
+ resolveCurrentSession: () => null,
187
+ activePlugins: ['delegate'],
188
+ })
189
+
190
+ const delegateTool = tools.find((tool) => tool.name === 'delegate')
191
+ const raw = await delegateTool.invoke({
192
+ input: JSON.stringify({
193
+ action: 'start',
194
+ name: 'Create Weather Script',
195
+ files: [{
196
+ path: 'weather_update/weather_fetcher.py',
197
+ content: 'print("weather")',
198
+ }],
199
+ }),
200
+ })
201
+ console.log(raw)
202
+ `)
203
+
204
+ assert.equal(output.backend, 'codex')
205
+ assert.equal(output.status, 'completed')
206
+ assert.match(String(output.response || ''), /codex fallback ok/i)
207
+ })
208
+
209
+ it('ranks authenticated delegate backends ahead of unauthenticated ones', () => {
210
+ const output = runWithFakeDelegates(`
211
+ const mod = await import('./src/lib/server/provider-health.ts')
212
+ const { rankDelegatesByHealth } = mod.default || mod['module.exports'] || mod
213
+ const ranked = rankDelegatesByHealth(['delegate_to_claude_code', 'delegate_to_codex_cli'])
214
+ console.log(JSON.stringify(ranked))
215
+ `)
216
+
217
+ assert.deepEqual(output, ['delegate_to_codex_cli', 'delegate_to_claude_code'])
218
+ })
219
+ })