@swarmclawai/swarmclaw 0.7.2 → 0.7.4

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 (274) hide show
  1. package/README.md +116 -50
  2. package/bin/package-manager.js +157 -0
  3. package/bin/package-manager.test.js +90 -0
  4. package/bin/server-cmd.js +38 -7
  5. package/bin/swarmclaw.js +54 -4
  6. package/bin/update-cmd.js +48 -10
  7. package/bin/update-cmd.test.js +55 -0
  8. package/package.json +8 -3
  9. package/scripts/postinstall.mjs +26 -0
  10. package/src/app/api/agents/[id]/route.ts +43 -0
  11. package/src/app/api/agents/[id]/thread/route.ts +39 -8
  12. package/src/app/api/agents/route.ts +35 -2
  13. package/src/app/api/auth/route.ts +77 -8
  14. package/src/app/api/chatrooms/[id]/chat/route.ts +22 -6
  15. package/src/app/api/chatrooms/[id]/pins/route.ts +2 -1
  16. package/src/app/api/chatrooms/[id]/reactions/route.ts +2 -1
  17. package/src/app/api/chatrooms/[id]/route.ts +6 -0
  18. package/src/app/api/chats/[id]/browser/route.ts +5 -1
  19. package/src/app/api/chats/[id]/chat/route.ts +7 -3
  20. package/src/app/api/chats/[id]/messages/route.ts +19 -13
  21. package/src/app/api/chats/[id]/route.ts +30 -0
  22. package/src/app/api/chats/[id]/stop/route.ts +6 -1
  23. package/src/app/api/chats/heartbeat/route.ts +2 -1
  24. package/src/app/api/chats/route.ts +23 -1
  25. package/src/app/api/connectors/[id]/doctor/route.ts +26 -0
  26. package/src/app/api/connectors/doctor/route.ts +13 -0
  27. package/src/app/api/external-agents/[id]/heartbeat/route.ts +33 -0
  28. package/src/app/api/external-agents/[id]/route.ts +31 -0
  29. package/src/app/api/external-agents/register/route.ts +3 -0
  30. package/src/app/api/external-agents/route.ts +66 -0
  31. package/src/app/api/files/open/route.ts +16 -14
  32. package/src/app/api/gateways/[id]/health/route.ts +28 -0
  33. package/src/app/api/gateways/[id]/route.ts +79 -0
  34. package/src/app/api/gateways/route.ts +57 -0
  35. package/src/app/api/memory/maintenance/route.ts +11 -1
  36. package/src/app/api/openclaw/agent-files/route.ts +27 -4
  37. package/src/app/api/openclaw/gateway/route.ts +10 -7
  38. package/src/app/api/openclaw/skills/route.ts +12 -4
  39. package/src/app/api/plugins/dependencies/route.ts +24 -0
  40. package/src/app/api/plugins/install/route.ts +15 -92
  41. package/src/app/api/plugins/route.ts +3 -26
  42. package/src/app/api/plugins/settings/route.ts +17 -12
  43. package/src/app/api/plugins/ui/route.ts +1 -0
  44. package/src/app/api/providers/[id]/discover-models/route.ts +27 -0
  45. package/src/app/api/schedules/[id]/route.ts +38 -9
  46. package/src/app/api/schedules/route.ts +51 -28
  47. package/src/app/api/settings/route.ts +55 -17
  48. package/src/app/api/setup/doctor/route.ts +6 -4
  49. package/src/app/api/tasks/[id]/route.ts +16 -6
  50. package/src/app/api/tasks/bulk/route.ts +3 -3
  51. package/src/app/api/tasks/route.ts +9 -4
  52. package/src/app/api/webhooks/[id]/route.ts +8 -1
  53. package/src/app/page.tsx +135 -17
  54. package/src/cli/binary.test.js +142 -0
  55. package/src/cli/index.js +38 -11
  56. package/src/cli/index.test.js +195 -0
  57. package/src/cli/index.ts +21 -12
  58. package/src/cli/server-cmd.test.js +59 -0
  59. package/src/cli/spec.js +20 -2
  60. package/src/components/agents/agent-card.tsx +15 -12
  61. package/src/components/agents/agent-chat-list.tsx +101 -1
  62. package/src/components/agents/agent-list.tsx +46 -9
  63. package/src/components/agents/agent-sheet.tsx +456 -23
  64. package/src/components/agents/inspector-panel.tsx +110 -49
  65. package/src/components/agents/sandbox-env-panel.tsx +4 -1
  66. package/src/components/auth/access-key-gate.tsx +36 -97
  67. package/src/components/auth/setup-wizard.tsx +970 -275
  68. package/src/components/chat/chat-area.tsx +70 -27
  69. package/src/components/chat/chat-card.tsx +6 -21
  70. package/src/components/chat/chat-header.tsx +263 -366
  71. package/src/components/chat/chat-list.tsx +62 -26
  72. package/src/components/chat/checkpoint-timeline.tsx +1 -1
  73. package/src/components/chat/message-list.tsx +145 -19
  74. package/src/components/chatrooms/chatroom-input.tsx +96 -33
  75. package/src/components/chatrooms/chatroom-list.tsx +141 -72
  76. package/src/components/chatrooms/chatroom-message.tsx +7 -6
  77. package/src/components/chatrooms/chatroom-sheet.tsx +13 -1
  78. package/src/components/chatrooms/chatroom-tool-request-banner.tsx +5 -2
  79. package/src/components/chatrooms/chatroom-view.tsx +422 -209
  80. package/src/components/chatrooms/reaction-picker.tsx +38 -33
  81. package/src/components/connectors/connector-list.tsx +265 -127
  82. package/src/components/connectors/connector-sheet.tsx +217 -0
  83. package/src/components/gateways/gateway-sheet.tsx +567 -0
  84. package/src/components/home/home-view.tsx +128 -4
  85. package/src/components/input/chat-input.tsx +135 -86
  86. package/src/components/layout/app-layout.tsx +385 -194
  87. package/src/components/layout/mobile-header.tsx +26 -8
  88. package/src/components/memory/memory-browser.tsx +71 -6
  89. package/src/components/memory/memory-card.tsx +18 -0
  90. package/src/components/memory/memory-detail.tsx +58 -31
  91. package/src/components/memory/memory-sheet.tsx +32 -4
  92. package/src/components/plugins/plugin-list.tsx +15 -3
  93. package/src/components/plugins/plugin-sheet.tsx +118 -9
  94. package/src/components/projects/project-detail.tsx +189 -1
  95. package/src/components/providers/provider-list.tsx +158 -2
  96. package/src/components/providers/provider-sheet.tsx +81 -70
  97. package/src/components/shared/agent-picker-list.tsx +2 -2
  98. package/src/components/shared/bottom-sheet.tsx +31 -15
  99. package/src/components/shared/command-palette.tsx +111 -24
  100. package/src/components/shared/confirm-dialog.tsx +45 -30
  101. package/src/components/shared/model-combobox.tsx +90 -8
  102. package/src/components/shared/settings/plugin-manager.tsx +20 -4
  103. package/src/components/shared/settings/section-capability-policy.tsx +105 -0
  104. package/src/components/shared/settings/section-heartbeat.tsx +88 -6
  105. package/src/components/shared/settings/section-orchestrator.tsx +6 -3
  106. package/src/components/shared/settings/section-runtime-loop.tsx +5 -5
  107. package/src/components/shared/settings/section-secrets.tsx +6 -6
  108. package/src/components/shared/settings/section-user-preferences.tsx +1 -1
  109. package/src/components/shared/settings/section-voice.tsx +5 -1
  110. package/src/components/shared/settings/section-web-search.tsx +10 -2
  111. package/src/components/shared/settings/settings-page.tsx +248 -47
  112. package/src/components/tasks/approvals-panel.tsx +211 -18
  113. package/src/components/tasks/task-board.tsx +242 -46
  114. package/src/components/ui/dialog.tsx +2 -2
  115. package/src/components/usage/metrics-dashboard.tsx +74 -1
  116. package/src/components/wallets/wallet-approval-dialog.tsx +59 -54
  117. package/src/components/wallets/wallet-panel.tsx +17 -5
  118. package/src/components/webhooks/webhook-sheet.tsx +7 -7
  119. package/src/lib/auth.ts +17 -0
  120. package/src/lib/chat-streaming-state.test.ts +108 -0
  121. package/src/lib/chat-streaming-state.ts +108 -0
  122. package/src/lib/heartbeat-defaults.ts +48 -0
  123. package/src/lib/memory-presentation.ts +59 -0
  124. package/src/lib/openclaw-agent-id.test.ts +14 -0
  125. package/src/lib/openclaw-agent-id.ts +31 -0
  126. package/src/lib/provider-model-discovery-client.ts +29 -0
  127. package/src/lib/providers/index.ts +12 -5
  128. package/src/lib/runtime-loop.ts +105 -3
  129. package/src/lib/safe-storage.ts +6 -1
  130. package/src/lib/server/agent-assignment.test.ts +112 -0
  131. package/src/lib/server/agent-assignment.ts +169 -0
  132. package/src/lib/server/agent-runtime-config.test.ts +141 -0
  133. package/src/lib/server/agent-runtime-config.ts +277 -0
  134. package/src/lib/server/approval-connector-notify.test.ts +253 -0
  135. package/src/lib/server/approvals-auto-approve.test.ts +264 -0
  136. package/src/lib/server/approvals.ts +483 -75
  137. package/src/lib/server/autonomy-runtime.test.ts +341 -0
  138. package/src/lib/server/browser-state.test.ts +118 -0
  139. package/src/lib/server/browser-state.ts +123 -0
  140. package/src/lib/server/build-llm.test.ts +44 -0
  141. package/src/lib/server/build-llm.ts +11 -4
  142. package/src/lib/server/builtin-plugins.ts +34 -0
  143. package/src/lib/server/chat-execution-heartbeat.test.ts +40 -0
  144. package/src/lib/server/chat-execution-tool-events.test.ts +219 -0
  145. package/src/lib/server/chat-execution.ts +402 -125
  146. package/src/lib/server/chatroom-health.test.ts +26 -0
  147. package/src/lib/server/chatroom-health.ts +2 -3
  148. package/src/lib/server/chatroom-helpers.test.ts +74 -2
  149. package/src/lib/server/chatroom-helpers.ts +144 -11
  150. package/src/lib/server/chatroom-session-persistence.test.ts +87 -0
  151. package/src/lib/server/connectors/discord.ts +175 -11
  152. package/src/lib/server/connectors/doctor.test.ts +80 -0
  153. package/src/lib/server/connectors/doctor.ts +116 -0
  154. package/src/lib/server/connectors/manager.ts +994 -130
  155. package/src/lib/server/connectors/policy.test.ts +222 -0
  156. package/src/lib/server/connectors/policy.ts +452 -0
  157. package/src/lib/server/connectors/slack.ts +189 -10
  158. package/src/lib/server/connectors/telegram.ts +65 -15
  159. package/src/lib/server/connectors/thread-context.test.ts +44 -0
  160. package/src/lib/server/connectors/thread-context.ts +72 -0
  161. package/src/lib/server/connectors/types.ts +41 -11
  162. package/src/lib/server/daemon-state.ts +62 -3
  163. package/src/lib/server/data-dir.ts +13 -0
  164. package/src/lib/server/delegation-jobs.test.ts +140 -0
  165. package/src/lib/server/delegation-jobs.ts +248 -0
  166. package/src/lib/server/document-utils.test.ts +47 -0
  167. package/src/lib/server/document-utils.ts +397 -0
  168. package/src/lib/server/eval/agent-regression.test.ts +47 -0
  169. package/src/lib/server/eval/agent-regression.ts +1742 -0
  170. package/src/lib/server/eval/runner.ts +11 -1
  171. package/src/lib/server/eval/store.ts +2 -1
  172. package/src/lib/server/heartbeat-service.ts +23 -43
  173. package/src/lib/server/heartbeat-source.test.ts +22 -0
  174. package/src/lib/server/heartbeat-source.ts +7 -0
  175. package/src/lib/server/identity-continuity.test.ts +77 -0
  176. package/src/lib/server/identity-continuity.ts +127 -0
  177. package/src/lib/server/mailbox-utils.ts +347 -0
  178. package/src/lib/server/main-agent-loop.ts +31 -964
  179. package/src/lib/server/memory-db.ts +4 -6
  180. package/src/lib/server/memory-tiers.ts +40 -0
  181. package/src/lib/server/openclaw-agent-resolver.test.ts +70 -0
  182. package/src/lib/server/openclaw-agent-resolver.ts +128 -0
  183. package/src/lib/server/openclaw-exec-config.ts +6 -5
  184. package/src/lib/server/openclaw-gateway.ts +123 -36
  185. package/src/lib/server/openclaw-skills-normalize.test.ts +56 -0
  186. package/src/lib/server/openclaw-skills-normalize.ts +136 -0
  187. package/src/lib/server/openclaw-sync.ts +3 -2
  188. package/src/lib/server/orchestrator-lg.ts +18 -8
  189. package/src/lib/server/orchestrator.ts +5 -4
  190. package/src/lib/server/playwright-proxy.mjs +27 -3
  191. package/src/lib/server/plugins.test.ts +215 -0
  192. package/src/lib/server/plugins.ts +832 -69
  193. package/src/lib/server/provider-health.ts +33 -3
  194. package/src/lib/server/provider-model-discovery.ts +481 -0
  195. package/src/lib/server/queue.ts +4 -21
  196. package/src/lib/server/runtime-settings.test.ts +119 -0
  197. package/src/lib/server/runtime-settings.ts +12 -92
  198. package/src/lib/server/schedule-normalization.ts +187 -0
  199. package/src/lib/server/scheduler.ts +2 -0
  200. package/src/lib/server/session-archive-memory.test.ts +85 -0
  201. package/src/lib/server/session-archive-memory.ts +230 -0
  202. package/src/lib/server/session-mailbox.ts +8 -18
  203. package/src/lib/server/session-reset-policy.test.ts +99 -0
  204. package/src/lib/server/session-reset-policy.ts +311 -0
  205. package/src/lib/server/session-run-manager.ts +33 -80
  206. package/src/lib/server/session-tools/autonomy-tools.test.ts +128 -0
  207. package/src/lib/server/session-tools/calendar.ts +2 -12
  208. package/src/lib/server/session-tools/connector.ts +109 -8
  209. package/src/lib/server/session-tools/context.ts +14 -2
  210. package/src/lib/server/session-tools/crawl.ts +447 -0
  211. package/src/lib/server/session-tools/crud.ts +96 -34
  212. package/src/lib/server/session-tools/delegate-fallback.test.ts +219 -0
  213. package/src/lib/server/session-tools/delegate.ts +406 -20
  214. package/src/lib/server/session-tools/discovery-approvals.test.ts +170 -0
  215. package/src/lib/server/session-tools/discovery.ts +40 -12
  216. package/src/lib/server/session-tools/document.ts +283 -0
  217. package/src/lib/server/session-tools/email.ts +1 -3
  218. package/src/lib/server/session-tools/extract.ts +137 -0
  219. package/src/lib/server/session-tools/file-normalize.test.ts +98 -0
  220. package/src/lib/server/session-tools/file-send.test.ts +84 -1
  221. package/src/lib/server/session-tools/file.ts +243 -24
  222. package/src/lib/server/session-tools/http.ts +9 -3
  223. package/src/lib/server/session-tools/human-loop.ts +227 -0
  224. package/src/lib/server/session-tools/image-gen.ts +1 -3
  225. package/src/lib/server/session-tools/index.ts +87 -2
  226. package/src/lib/server/session-tools/mailbox.ts +276 -0
  227. package/src/lib/server/session-tools/manage-schedules.test.ts +137 -0
  228. package/src/lib/server/session-tools/memory.ts +35 -3
  229. package/src/lib/server/session-tools/monitor.ts +162 -12
  230. package/src/lib/server/session-tools/normalize-tool-args.ts +17 -14
  231. package/src/lib/server/session-tools/openclaw-nodes.test.ts +111 -0
  232. package/src/lib/server/session-tools/openclaw-nodes.ts +86 -20
  233. package/src/lib/server/session-tools/platform-normalize.test.ts +142 -0
  234. package/src/lib/server/session-tools/platform.ts +142 -4
  235. package/src/lib/server/session-tools/plugin-creator.ts +95 -25
  236. package/src/lib/server/session-tools/primitive-tools.test.ts +257 -0
  237. package/src/lib/server/session-tools/replicate.ts +1 -3
  238. package/src/lib/server/session-tools/sandbox.ts +51 -92
  239. package/src/lib/server/session-tools/schedule.ts +20 -10
  240. package/src/lib/server/session-tools/session-info.ts +58 -4
  241. package/src/lib/server/session-tools/session-tools-wiring.test.ts +54 -17
  242. package/src/lib/server/session-tools/shell.ts +2 -2
  243. package/src/lib/server/session-tools/subagent.ts +195 -27
  244. package/src/lib/server/session-tools/table.ts +587 -0
  245. package/src/lib/server/session-tools/wallet.ts +13 -10
  246. package/src/lib/server/session-tools/web-browser-config.test.ts +39 -0
  247. package/src/lib/server/session-tools/web.ts +947 -108
  248. package/src/lib/server/storage.ts +255 -10
  249. package/src/lib/server/stream-agent-chat.test.ts +61 -0
  250. package/src/lib/server/stream-agent-chat.ts +185 -25
  251. package/src/lib/server/structured-extract.test.ts +72 -0
  252. package/src/lib/server/structured-extract.ts +373 -0
  253. package/src/lib/server/task-mention.test.ts +16 -2
  254. package/src/lib/server/task-mention.ts +61 -11
  255. package/src/lib/server/tool-aliases.ts +80 -12
  256. package/src/lib/server/tool-capability-policy.ts +7 -1
  257. package/src/lib/server/tool-retry.ts +2 -0
  258. package/src/lib/server/watch-jobs.test.ts +173 -0
  259. package/src/lib/server/watch-jobs.ts +532 -0
  260. package/src/lib/server/ws-hub.ts +5 -3
  261. package/src/lib/setup-defaults.ts +352 -11
  262. package/src/lib/tool-definitions.ts +3 -4
  263. package/src/lib/validation/schemas.test.ts +26 -0
  264. package/src/lib/validation/schemas.ts +62 -1
  265. package/src/lib/ws-client.ts +14 -12
  266. package/src/proxy.ts +5 -5
  267. package/src/stores/use-app-store.ts +43 -7
  268. package/src/stores/use-chat-store.ts +31 -2
  269. package/src/stores/use-chatroom-store.ts +153 -26
  270. package/src/types/index.ts +470 -44
  271. package/src/app/api/chats/[id]/main-loop/route.ts +0 -94
  272. package/src/components/chat/new-chat-sheet.tsx +0 -253
  273. package/src/lib/server/main-session.ts +0 -17
  274. package/src/lib/server/session-run-manager.test.ts +0 -26
@@ -0,0 +1,34 @@
1
+ import '@/lib/server/session-tools/shell'
2
+ import '@/lib/server/session-tools/file'
3
+ import '@/lib/server/session-tools/edit_file'
4
+ import '@/lib/server/session-tools/web'
5
+ import '@/lib/server/session-tools/memory'
6
+ import '@/lib/server/session-tools/platform'
7
+ import '@/lib/server/session-tools/monitor'
8
+ import '@/lib/server/session-tools/discovery'
9
+ import '@/lib/server/session-tools/sample-ui'
10
+ import '@/lib/server/session-tools/git'
11
+ import '@/lib/server/session-tools/wallet'
12
+ import '@/lib/server/session-tools/connector'
13
+ import '@/lib/server/session-tools/http'
14
+ import '@/lib/server/session-tools/sandbox'
15
+ import '@/lib/server/session-tools/canvas'
16
+ import '@/lib/server/session-tools/chatroom'
17
+ import '@/lib/server/session-tools/delegate'
18
+ import '@/lib/server/session-tools/schedule'
19
+ import '@/lib/server/session-tools/session-info'
20
+ import '@/lib/server/session-tools/openclaw-nodes'
21
+ import '@/lib/server/session-tools/openclaw-workspace'
22
+ import '@/lib/server/session-tools/context-mgmt'
23
+ import '@/lib/server/session-tools/subagent'
24
+ import '@/lib/server/session-tools/plugin-creator'
25
+ import '@/lib/server/session-tools/image-gen'
26
+ import '@/lib/server/session-tools/email'
27
+ import '@/lib/server/session-tools/calendar'
28
+ import '@/lib/server/session-tools/replicate'
29
+ import '@/lib/server/session-tools/mailbox'
30
+ import '@/lib/server/session-tools/human-loop'
31
+ import '@/lib/server/session-tools/document'
32
+ import '@/lib/server/session-tools/extract'
33
+ import '@/lib/server/session-tools/table'
34
+ import '@/lib/server/session-tools/crawl'
@@ -0,0 +1,40 @@
1
+ import assert from 'node:assert/strict'
2
+ import { describe, it } from 'node:test'
3
+ import type { Message } from '@/types'
4
+ import { pruneSuppressedHeartbeatStreamMessage, shouldApplySessionFreshnessReset } from './chat-execution'
5
+
6
+ describe('pruneSuppressedHeartbeatStreamMessage', () => {
7
+ it('removes a trailing streaming assistant heartbeat artifact', () => {
8
+ const messages: Message[] = [
9
+ { role: 'assistant', text: 'real reply', time: 1, kind: 'chat' },
10
+ { role: 'assistant', text: 'HEARTBEAT_OK', time: 2, streaming: true },
11
+ ]
12
+
13
+ const changed = pruneSuppressedHeartbeatStreamMessage(messages)
14
+
15
+ assert.equal(changed, true)
16
+ assert.deepEqual(messages, [
17
+ { role: 'assistant', text: 'real reply', time: 1, kind: 'chat' },
18
+ ])
19
+ })
20
+
21
+ it('keeps non-streaming or user messages intact', () => {
22
+ const nonStreaming: Message[] = [
23
+ { role: 'assistant', text: 'HEARTBEAT_OK', time: 2, kind: 'heartbeat' },
24
+ ]
25
+ const userTail: Message[] = [
26
+ { role: 'user', text: 'ping', time: 3, streaming: true },
27
+ ]
28
+
29
+ assert.equal(pruneSuppressedHeartbeatStreamMessage(nonStreaming), false)
30
+ assert.equal(pruneSuppressedHeartbeatStreamMessage(userTail), false)
31
+ assert.equal(nonStreaming.length, 1)
32
+ assert.equal(userTail.length, 1)
33
+ })
34
+
35
+ it('applies freshness resets beyond heartbeat but skips eval runs', () => {
36
+ assert.equal(shouldApplySessionFreshnessReset('chat'), true)
37
+ assert.equal(shouldApplySessionFreshnessReset('heartbeat'), true)
38
+ assert.equal(shouldApplySessionFreshnessReset('eval'), false)
39
+ })
40
+ })
@@ -0,0 +1,219 @@
1
+ import { describe, it } from 'node:test'
2
+ import assert from 'node:assert/strict'
3
+ import fs from 'fs'
4
+ import os from 'os'
5
+ import path from 'path'
6
+ import type { MessageToolEvent, SSEEvent } from '@/types'
7
+ import {
8
+ collectToolEvent,
9
+ dedupeConsecutiveToolEvents,
10
+ isLikelyToolErrorOutput,
11
+ normalizeAssistantArtifactLinks,
12
+ requestedToolNamesFromMessage,
13
+ translateRequestedToolInvocation,
14
+ } from './chat-execution'
15
+
16
+ describe('collectToolEvent', () => {
17
+ it('dedupes consecutive identical pending tool_call events', () => {
18
+ const bag: MessageToolEvent[] = []
19
+ const toolCall: SSEEvent = {
20
+ t: 'tool_call',
21
+ toolName: 'files',
22
+ toolInput: '{"path":"spec.md"}',
23
+ }
24
+
25
+ collectToolEvent(toolCall, bag)
26
+ collectToolEvent(toolCall, bag)
27
+
28
+ assert.deepEqual(bag, [
29
+ {
30
+ name: 'files',
31
+ input: '{"path":"spec.md"}',
32
+ },
33
+ ])
34
+ })
35
+
36
+ it('still records a second tool call after the first one completed', () => {
37
+ const bag: MessageToolEvent[] = []
38
+ const toolCall: SSEEvent = {
39
+ t: 'tool_call',
40
+ toolName: 'files',
41
+ toolInput: '{"path":"spec.md"}',
42
+ }
43
+ const toolResult: SSEEvent = {
44
+ t: 'tool_result',
45
+ toolName: 'files',
46
+ toolOutput: 'Written spec.md (12 bytes)',
47
+ }
48
+
49
+ collectToolEvent(toolCall, bag)
50
+ collectToolEvent(toolResult, bag)
51
+ collectToolEvent(toolCall, bag)
52
+
53
+ assert.deepEqual(bag, [
54
+ {
55
+ name: 'files',
56
+ input: '{"path":"spec.md"}',
57
+ output: 'Written spec.md (12 bytes)',
58
+ error: undefined,
59
+ },
60
+ {
61
+ name: 'files',
62
+ input: '{"path":"spec.md"}',
63
+ },
64
+ ])
65
+ })
66
+
67
+ it('marks command-exit and MCP validation failures as tool errors', () => {
68
+ const bag: MessageToolEvent[] = []
69
+ collectToolEvent({
70
+ t: 'tool_call',
71
+ toolName: 'shell',
72
+ toolInput: '{"command":"python broken.py"}',
73
+ }, bag)
74
+ collectToolEvent({
75
+ t: 'tool_result',
76
+ toolName: 'shell',
77
+ toolOutput: 'Error (exit 1): Traceback...',
78
+ }, bag)
79
+ collectToolEvent({
80
+ t: 'tool_call',
81
+ toolName: 'browser',
82
+ toolInput: '{"command":"click"}',
83
+ }, bag)
84
+ collectToolEvent({
85
+ t: 'tool_result',
86
+ toolName: 'browser',
87
+ toolOutput: '{"error":"MCP error -32000: invalid_type","issues":[{"code":"invalid_type"}]}',
88
+ }, bag)
89
+
90
+ assert.equal(bag[0].error, true)
91
+ assert.equal(bag[1].error, true)
92
+ })
93
+ })
94
+
95
+ describe('dedupeConsecutiveToolEvents', () => {
96
+ it('removes consecutive duplicate persisted events', () => {
97
+ const events: MessageToolEvent[] = [
98
+ { name: 'web', input: '{"action":"search"}', output: 'ok' },
99
+ { name: 'web', input: '{"action":"search"}', output: 'ok' },
100
+ { name: 'files', input: '{"action":"list"}', output: 'spec.md' },
101
+ { name: 'files', input: '{"action":"list"}', output: 'spec.md' },
102
+ { name: 'web', input: '{"action":"search"}', output: 'ok' },
103
+ ]
104
+
105
+ assert.deepEqual(dedupeConsecutiveToolEvents(events), [
106
+ { name: 'web', input: '{"action":"search"}', output: 'ok' },
107
+ { name: 'files', input: '{"action":"list"}', output: 'spec.md' },
108
+ { name: 'web', input: '{"action":"search"}', output: 'ok' },
109
+ ])
110
+ })
111
+
112
+ it('collapses adjacent repeated event blocks', () => {
113
+ const events: MessageToolEvent[] = [
114
+ { name: 'delegate', input: '{"task":"draft"}', output: '{"status":"completed"}' },
115
+ { name: 'files', input: '{"action":"write"}', output: 'Written spec.md' },
116
+ { name: 'delegate', input: '{"task":"draft"}', output: '{"status":"completed"}' },
117
+ { name: 'files', input: '{"action":"write"}', output: 'Written spec.md' },
118
+ { name: 'delegate_to_codex_cli', input: '{"task":"full"}', output: '{"status":"completed"}' },
119
+ ]
120
+
121
+ assert.deepEqual(dedupeConsecutiveToolEvents(events), [
122
+ { name: 'delegate', input: '{"task":"draft"}', output: '{"status":"completed"}' },
123
+ { name: 'files', input: '{"action":"write"}', output: 'Written spec.md' },
124
+ { name: 'delegate_to_codex_cli', input: '{"task":"full"}', output: '{"status":"completed"}' },
125
+ ])
126
+ })
127
+ })
128
+
129
+ describe('requestedToolNamesFromMessage', () => {
130
+ it('does not infer delegate from ordinary delegation prose', () => {
131
+ assert.deepEqual(
132
+ requestedToolNamesFromMessage('If Molly delegates later, that is fine, but do not mention tools.'),
133
+ [],
134
+ )
135
+ })
136
+
137
+ it('still detects explicit delegate tool requests', () => {
138
+ assert.deepEqual(
139
+ requestedToolNamesFromMessage('Use the delegate tool if Codex is better suited.'),
140
+ ['delegate'],
141
+ )
142
+ })
143
+ })
144
+
145
+ describe('normalizeAssistantArtifactLinks', () => {
146
+ it('rewrites existing workspace sandbox links to served file URLs', () => {
147
+ const cwd = fs.mkdtempSync(path.join(os.tmpdir(), 'chat-execution-links-'))
148
+ const nestedDir = path.join(cwd, 'notes')
149
+ fs.mkdirSync(nestedDir, { recursive: true })
150
+ const filePath = path.join(nestedDir, 'spec.md')
151
+ fs.writeFileSync(filePath, '# spec\n')
152
+
153
+ const normalized = normalizeAssistantArtifactLinks(
154
+ `Saved \`sandbox:/workspace/notes/spec.md\`, [spec](${filePath}), [workspace spec](sandbox:/workspace/notes/spec.md), and sandbox:/api/uploads/demo.pdf`,
155
+ cwd,
156
+ )
157
+
158
+ assert.equal(
159
+ normalized,
160
+ `Saved \`sandbox:/workspace/notes/spec.md\`, [spec](/api/files/serve?path=${encodeURIComponent(filePath)}), [workspace spec](/api/files/serve?path=${encodeURIComponent(filePath)}), and /api/uploads/demo.pdf`,
161
+ )
162
+ })
163
+ })
164
+
165
+ describe('isLikelyToolErrorOutput', () => {
166
+ it('recognizes broader structured tool failures without flagging normal output', () => {
167
+ assert.equal(isLikelyToolErrorOutput('Error (exit 1): build failed'), true)
168
+ assert.equal(isLikelyToolErrorOutput('{"status":"failed","error":"timeout"}'), true)
169
+ assert.equal(isLikelyToolErrorOutput('{"status":"completed"}'), false)
170
+ assert.equal(isLikelyToolErrorOutput('Written spec.md (12 bytes)'), false)
171
+ })
172
+ })
173
+
174
+ describe('translateRequestedToolInvocation', () => {
175
+ it('keeps a specific manage tool when that tool is directly available', () => {
176
+ assert.deepEqual(
177
+ translateRequestedToolInvocation(
178
+ 'manage_schedules',
179
+ { action: 'create', scheduleType: 'interval' },
180
+ 'schedule a job',
181
+ ['manage_schedules'],
182
+ ),
183
+ {
184
+ toolName: 'manage_schedules',
185
+ args: { action: 'create', scheduleType: 'interval' },
186
+ },
187
+ )
188
+ })
189
+
190
+ it('maps manage_platform back to the specific manage tool when only the specific tool exists', () => {
191
+ assert.deepEqual(
192
+ translateRequestedToolInvocation(
193
+ 'manage_platform',
194
+ { resource: 'schedules', action: 'create', scheduleType: 'interval' },
195
+ 'schedule a job',
196
+ ['manage_schedules'],
197
+ ),
198
+ {
199
+ toolName: 'manage_schedules',
200
+ args: { resource: 'schedules', action: 'create', scheduleType: 'interval' },
201
+ },
202
+ )
203
+ })
204
+
205
+ it('still falls back to manage_platform when only the unified tool exists', () => {
206
+ assert.deepEqual(
207
+ translateRequestedToolInvocation(
208
+ 'manage_schedules',
209
+ { action: 'list' },
210
+ 'list schedules',
211
+ ['manage_platform'],
212
+ ),
213
+ {
214
+ toolName: 'manage_platform',
215
+ args: { resource: 'schedules', action: 'list' },
216
+ },
217
+ )
218
+ })
219
+ })