@renxqoo/renx-code 0.0.2 → 0.0.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 (298) hide show
  1. package/README.md +59 -223
  2. package/bin/renx.cjs +34 -0
  3. package/package.json +27 -83
  4. package/src/App.tsx +297 -0
  5. package/src/agent/runtime/event-format.ts +258 -0
  6. package/src/agent/runtime/model-types.ts +13 -0
  7. package/src/agent/runtime/runtime.context-usage.test.ts +193 -0
  8. package/src/agent/runtime/runtime.error-handling.test.ts +236 -0
  9. package/src/agent/runtime/runtime.simple.test.ts +16 -0
  10. package/src/agent/runtime/runtime.test.ts +293 -0
  11. package/src/agent/runtime/runtime.ts +881 -0
  12. package/src/agent/runtime/runtime.usage-forwarding.test.ts +229 -0
  13. package/src/agent/runtime/source-modules.test.ts +57 -0
  14. package/src/agent/runtime/source-modules.ts +353 -0
  15. package/src/agent/runtime/tool-call-buffer.test.ts +65 -0
  16. package/src/agent/runtime/tool-call-buffer.ts +60 -0
  17. package/src/agent/runtime/tool-confirmation.test.ts +56 -0
  18. package/src/agent/runtime/tool-confirmation.ts +15 -0
  19. package/src/agent/runtime/types.ts +99 -0
  20. package/src/commands/slash-commands.test.ts +216 -0
  21. package/src/commands/slash-commands.ts +64 -0
  22. package/src/components/chat/assistant-reply.test.tsx +47 -0
  23. package/src/components/chat/assistant-reply.tsx +136 -0
  24. package/src/components/chat/assistant-segment.test.ts +99 -0
  25. package/src/components/chat/assistant-segment.tsx +125 -0
  26. package/src/components/chat/assistant-tool-group.tsx +900 -0
  27. package/src/components/chat/code-block.test.tsx +206 -0
  28. package/src/components/chat/code-block.tsx +313 -0
  29. package/src/components/chat/prompt-card.tsx +81 -0
  30. package/src/components/chat/segment-groups.test.ts +52 -0
  31. package/src/components/chat/segment-groups.ts +106 -0
  32. package/src/components/chat/turn-item.tsx +39 -0
  33. package/src/components/conversation-panel.tsx +43 -0
  34. package/src/components/file-mention-menu.tsx +77 -0
  35. package/src/components/file-picker-dialog.tsx +206 -0
  36. package/src/components/footer-hints.tsx +75 -0
  37. package/src/components/model-picker-dialog.tsx +248 -0
  38. package/src/components/prompt.tsx +233 -0
  39. package/src/components/slash-command-menu.tsx +65 -0
  40. package/src/components/tool-confirm-dialog-content.test.ts +103 -0
  41. package/src/components/tool-confirm-dialog-content.ts +186 -0
  42. package/src/components/tool-confirm-dialog.tsx +187 -0
  43. package/src/components/tool-display-config.ts +119 -0
  44. package/src/context-usage-regressions.test.ts +26 -0
  45. package/src/files/attachment-capabilities.test.ts +30 -0
  46. package/src/files/attachment-capabilities.ts +50 -0
  47. package/src/files/attachment-content.ts +153 -0
  48. package/src/files/file-mention-query.test.ts +34 -0
  49. package/src/files/file-mention-query.ts +32 -0
  50. package/src/files/prompt-display.ts +13 -0
  51. package/src/files/types.ts +5 -0
  52. package/src/files/workspace-files.ts +63 -0
  53. package/src/hooks/agent-event-handlers.test.ts +207 -0
  54. package/src/hooks/agent-event-handlers.ts +196 -0
  55. package/src/hooks/chat-local-replies.fixed.test.ts +119 -0
  56. package/src/hooks/chat-local-replies.test.ts +153 -0
  57. package/src/hooks/chat-local-replies.ts +63 -0
  58. package/src/hooks/turn-updater.test.ts +70 -0
  59. package/src/hooks/turn-updater.ts +166 -0
  60. package/src/hooks/use-agent-chat.context.test.ts +10 -0
  61. package/src/hooks/use-agent-chat.status.test.ts +14 -0
  62. package/src/hooks/use-agent-chat.test.ts +80 -0
  63. package/src/hooks/use-agent-chat.ts +621 -0
  64. package/src/hooks/use-file-mention-menu.ts +196 -0
  65. package/src/hooks/use-file-picker.ts +185 -0
  66. package/src/hooks/use-model-picker.ts +196 -0
  67. package/src/hooks/use-slash-command-menu.ts +154 -0
  68. package/src/index.tsx +55 -0
  69. package/src/runtime/clipboard.test.ts +43 -0
  70. package/src/runtime/clipboard.ts +89 -0
  71. package/src/runtime/exit.test.ts +177 -0
  72. package/src/runtime/exit.ts +98 -0
  73. package/src/runtime/runtime-support.test.ts +31 -0
  74. package/src/runtime/terminal-theme.test.ts +55 -0
  75. package/src/runtime/terminal-theme.ts +196 -0
  76. package/src/types/chat.ts +32 -0
  77. package/src/types/message-content.ts +48 -0
  78. package/src/ui/open-code-theme.ts +176 -0
  79. package/src/ui/opencode-markdown.ts +211 -0
  80. package/src/ui/theme.simple.test.ts +52 -0
  81. package/src/ui/theme.test.ts +151 -0
  82. package/src/ui/theme.ts +152 -0
  83. package/src/utils/time.test.ts +144 -0
  84. package/src/utils/time.ts +7 -0
  85. package/tsconfig.json +30 -0
  86. package/LICENSE +0 -21
  87. package/dist/App.d.ts +0 -2
  88. package/dist/App.d.ts.map +0 -1
  89. package/dist/App.js +0 -170
  90. package/dist/App.js.map +0 -1
  91. package/dist/agent/prompts/system.d.ts +0 -24
  92. package/dist/agent/prompts/system.d.ts.map +0 -1
  93. package/dist/agent/prompts/system.js +0 -222
  94. package/dist/agent/prompts/system.js.map +0 -1
  95. package/dist/agent/runtime/event-format.d.ts +0 -17
  96. package/dist/agent/runtime/event-format.d.ts.map +0 -1
  97. package/dist/agent/runtime/event-format.js +0 -194
  98. package/dist/agent/runtime/event-format.js.map +0 -1
  99. package/dist/agent/runtime/model-types.d.ts +0 -13
  100. package/dist/agent/runtime/model-types.d.ts.map +0 -1
  101. package/dist/agent/runtime/model-types.js +0 -1
  102. package/dist/agent/runtime/model-types.js.map +0 -1
  103. package/dist/agent/runtime/runtime.d.ts +0 -16
  104. package/dist/agent/runtime/runtime.d.ts.map +0 -1
  105. package/dist/agent/runtime/runtime.js +0 -691
  106. package/dist/agent/runtime/runtime.js.map +0 -1
  107. package/dist/agent/runtime/source-modules.d.ts +0 -176
  108. package/dist/agent/runtime/source-modules.d.ts.map +0 -1
  109. package/dist/agent/runtime/source-modules.js +0 -110
  110. package/dist/agent/runtime/source-modules.js.map +0 -1
  111. package/dist/agent/runtime/tool-call-buffer.d.ts +0 -12
  112. package/dist/agent/runtime/tool-call-buffer.d.ts.map +0 -1
  113. package/dist/agent/runtime/tool-call-buffer.js +0 -48
  114. package/dist/agent/runtime/tool-call-buffer.js.map +0 -1
  115. package/dist/agent/runtime/tool-confirmation.d.ts +0 -3
  116. package/dist/agent/runtime/tool-confirmation.d.ts.map +0 -1
  117. package/dist/agent/runtime/tool-confirmation.js +0 -9
  118. package/dist/agent/runtime/tool-confirmation.js.map +0 -1
  119. package/dist/agent/runtime/types.d.ts +0 -86
  120. package/dist/agent/runtime/types.d.ts.map +0 -1
  121. package/dist/agent/runtime/types.js +0 -1
  122. package/dist/agent/runtime/types.js.map +0 -1
  123. package/dist/cli.d.ts +0 -3
  124. package/dist/cli.d.ts.map +0 -1
  125. package/dist/cli.js +0 -43
  126. package/dist/cli.js.map +0 -1
  127. package/dist/commands/slash-commands.d.ts +0 -11
  128. package/dist/commands/slash-commands.d.ts.map +0 -1
  129. package/dist/commands/slash-commands.js +0 -48
  130. package/dist/commands/slash-commands.js.map +0 -1
  131. package/dist/components/chat/assistant-reply.d.ts +0 -13
  132. package/dist/components/chat/assistant-reply.d.ts.map +0 -1
  133. package/dist/components/chat/assistant-reply.js +0 -78
  134. package/dist/components/chat/assistant-reply.js.map +0 -1
  135. package/dist/components/chat/assistant-segment.d.ts +0 -8
  136. package/dist/components/chat/assistant-segment.d.ts.map +0 -1
  137. package/dist/components/chat/assistant-segment.js +0 -54
  138. package/dist/components/chat/assistant-segment.js.map +0 -1
  139. package/dist/components/chat/assistant-tool-group.d.ts +0 -7
  140. package/dist/components/chat/assistant-tool-group.d.ts.map +0 -1
  141. package/dist/components/chat/assistant-tool-group.js +0 -695
  142. package/dist/components/chat/assistant-tool-group.js.map +0 -1
  143. package/dist/components/chat/code-block.d.ts +0 -16
  144. package/dist/components/chat/code-block.d.ts.map +0 -1
  145. package/dist/components/chat/code-block.js +0 -194
  146. package/dist/components/chat/code-block.js.map +0 -1
  147. package/dist/components/chat/prompt-card.d.ts +0 -9
  148. package/dist/components/chat/prompt-card.d.ts.map +0 -1
  149. package/dist/components/chat/prompt-card.js +0 -18
  150. package/dist/components/chat/prompt-card.js.map +0 -1
  151. package/dist/components/chat/segment-groups.d.ts +0 -24
  152. package/dist/components/chat/segment-groups.d.ts.map +0 -1
  153. package/dist/components/chat/segment-groups.js +0 -69
  154. package/dist/components/chat/segment-groups.js.map +0 -1
  155. package/dist/components/chat/turn-item.d.ts +0 -9
  156. package/dist/components/chat/turn-item.d.ts.map +0 -1
  157. package/dist/components/chat/turn-item.js +0 -11
  158. package/dist/components/chat/turn-item.js.map +0 -1
  159. package/dist/components/conversation-panel.d.ts +0 -8
  160. package/dist/components/conversation-panel.d.ts.map +0 -1
  161. package/dist/components/conversation-panel.js +0 -8
  162. package/dist/components/conversation-panel.js.map +0 -1
  163. package/dist/components/file-mention-menu.d.ts +0 -11
  164. package/dist/components/file-mention-menu.d.ts.map +0 -1
  165. package/dist/components/file-mention-menu.js +0 -15
  166. package/dist/components/file-mention-menu.js.map +0 -1
  167. package/dist/components/file-picker-dialog.d.ts +0 -21
  168. package/dist/components/file-picker-dialog.d.ts.map +0 -1
  169. package/dist/components/file-picker-dialog.js +0 -48
  170. package/dist/components/file-picker-dialog.js.map +0 -1
  171. package/dist/components/footer-hints.d.ts +0 -7
  172. package/dist/components/footer-hints.d.ts.map +0 -1
  173. package/dist/components/footer-hints.js +0 -29
  174. package/dist/components/footer-hints.js.map +0 -1
  175. package/dist/components/model-picker-dialog.d.ts +0 -20
  176. package/dist/components/model-picker-dialog.d.ts.map +0 -1
  177. package/dist/components/model-picker-dialog.js +0 -72
  178. package/dist/components/model-picker-dialog.js.map +0 -1
  179. package/dist/components/prompt.d.ts +0 -18
  180. package/dist/components/prompt.d.ts.map +0 -1
  181. package/dist/components/prompt.js +0 -96
  182. package/dist/components/prompt.js.map +0 -1
  183. package/dist/components/slash-command-menu.d.ts +0 -9
  184. package/dist/components/slash-command-menu.d.ts.map +0 -1
  185. package/dist/components/slash-command-menu.js +0 -20
  186. package/dist/components/slash-command-menu.js.map +0 -1
  187. package/dist/components/tool-confirm-dialog-content.d.ts +0 -15
  188. package/dist/components/tool-confirm-dialog-content.d.ts.map +0 -1
  189. package/dist/components/tool-confirm-dialog-content.js +0 -143
  190. package/dist/components/tool-confirm-dialog-content.js.map +0 -1
  191. package/dist/components/tool-confirm-dialog.d.ts +0 -12
  192. package/dist/components/tool-confirm-dialog.d.ts.map +0 -1
  193. package/dist/components/tool-confirm-dialog.js +0 -21
  194. package/dist/components/tool-confirm-dialog.js.map +0 -1
  195. package/dist/components/tool-display-config.d.ts +0 -11
  196. package/dist/components/tool-display-config.d.ts.map +0 -1
  197. package/dist/components/tool-display-config.js +0 -94
  198. package/dist/components/tool-display-config.js.map +0 -1
  199. package/dist/config/paths.d.ts +0 -7
  200. package/dist/config/paths.d.ts.map +0 -1
  201. package/dist/config/paths.js +0 -24
  202. package/dist/config/paths.js.map +0 -1
  203. package/dist/files/attachment-capabilities.d.ts +0 -19
  204. package/dist/files/attachment-capabilities.d.ts.map +0 -1
  205. package/dist/files/attachment-capabilities.js +0 -26
  206. package/dist/files/attachment-capabilities.js.map +0 -1
  207. package/dist/files/attachment-content.d.ts +0 -5
  208. package/dist/files/attachment-content.d.ts.map +0 -1
  209. package/dist/files/attachment-content.js +0 -117
  210. package/dist/files/attachment-content.js.map +0 -1
  211. package/dist/files/file-mention-query.d.ts +0 -9
  212. package/dist/files/file-mention-query.d.ts.map +0 -1
  213. package/dist/files/file-mention-query.js +0 -23
  214. package/dist/files/file-mention-query.js.map +0 -1
  215. package/dist/files/prompt-display.d.ts +0 -3
  216. package/dist/files/prompt-display.d.ts.map +0 -1
  217. package/dist/files/prompt-display.js +0 -11
  218. package/dist/files/prompt-display.js.map +0 -1
  219. package/dist/files/types.d.ts +0 -6
  220. package/dist/files/types.d.ts.map +0 -1
  221. package/dist/files/types.js +0 -1
  222. package/dist/files/types.js.map +0 -1
  223. package/dist/files/workspace-files.d.ts +0 -3
  224. package/dist/files/workspace-files.d.ts.map +0 -1
  225. package/dist/files/workspace-files.js +0 -50
  226. package/dist/files/workspace-files.js.map +0 -1
  227. package/dist/hooks/agent-event-handlers.d.ts +0 -11
  228. package/dist/hooks/agent-event-handlers.d.ts.map +0 -1
  229. package/dist/hooks/agent-event-handlers.js +0 -137
  230. package/dist/hooks/agent-event-handlers.js.map +0 -1
  231. package/dist/hooks/chat-local-replies.d.ts +0 -9
  232. package/dist/hooks/chat-local-replies.d.ts.map +0 -1
  233. package/dist/hooks/chat-local-replies.js +0 -54
  234. package/dist/hooks/chat-local-replies.js.map +0 -1
  235. package/dist/hooks/turn-updater.d.ts +0 -9
  236. package/dist/hooks/turn-updater.d.ts.map +0 -1
  237. package/dist/hooks/turn-updater.js +0 -103
  238. package/dist/hooks/turn-updater.js.map +0 -1
  239. package/dist/hooks/use-agent-chat.d.ts +0 -29
  240. package/dist/hooks/use-agent-chat.d.ts.map +0 -1
  241. package/dist/hooks/use-agent-chat.js +0 -455
  242. package/dist/hooks/use-agent-chat.js.map +0 -1
  243. package/dist/hooks/use-file-mention-menu.d.ts +0 -22
  244. package/dist/hooks/use-file-mention-menu.d.ts.map +0 -1
  245. package/dist/hooks/use-file-mention-menu.js +0 -137
  246. package/dist/hooks/use-file-mention-menu.js.map +0 -1
  247. package/dist/hooks/use-file-picker.d.ts +0 -21
  248. package/dist/hooks/use-file-picker.d.ts.map +0 -1
  249. package/dist/hooks/use-file-picker.js +0 -145
  250. package/dist/hooks/use-file-picker.js.map +0 -1
  251. package/dist/hooks/use-model-picker.d.ts +0 -23
  252. package/dist/hooks/use-model-picker.d.ts.map +0 -1
  253. package/dist/hooks/use-model-picker.js +0 -151
  254. package/dist/hooks/use-model-picker.js.map +0 -1
  255. package/dist/hooks/use-slash-command-menu.d.ts +0 -19
  256. package/dist/hooks/use-slash-command-menu.d.ts.map +0 -1
  257. package/dist/hooks/use-slash-command-menu.js +0 -101
  258. package/dist/hooks/use-slash-command-menu.js.map +0 -1
  259. package/dist/index.d.ts +0 -2
  260. package/dist/index.d.ts.map +0 -1
  261. package/dist/index.js +0 -39
  262. package/dist/index.js.map +0 -1
  263. package/dist/runtime/clipboard.d.ts +0 -10
  264. package/dist/runtime/clipboard.d.ts.map +0 -1
  265. package/dist/runtime/clipboard.js +0 -64
  266. package/dist/runtime/clipboard.js.map +0 -1
  267. package/dist/runtime/exit.d.ts +0 -7
  268. package/dist/runtime/exit.d.ts.map +0 -1
  269. package/dist/runtime/exit.js +0 -85
  270. package/dist/runtime/exit.js.map +0 -1
  271. package/dist/runtime/terminal-theme.d.ts +0 -25
  272. package/dist/runtime/terminal-theme.d.ts.map +0 -1
  273. package/dist/runtime/terminal-theme.js +0 -148
  274. package/dist/runtime/terminal-theme.js.map +0 -1
  275. package/dist/types/chat.d.ts +0 -29
  276. package/dist/types/chat.d.ts.map +0 -1
  277. package/dist/types/chat.js +0 -1
  278. package/dist/types/chat.js.map +0 -1
  279. package/dist/types/message-content.d.ts +0 -38
  280. package/dist/types/message-content.d.ts.map +0 -1
  281. package/dist/types/message-content.js +0 -1
  282. package/dist/types/message-content.js.map +0 -1
  283. package/dist/ui/open-code-theme.d.ts +0 -58
  284. package/dist/ui/open-code-theme.d.ts.map +0 -1
  285. package/dist/ui/open-code-theme.js +0 -113
  286. package/dist/ui/open-code-theme.js.map +0 -1
  287. package/dist/ui/opencode-markdown.d.ts +0 -7
  288. package/dist/ui/opencode-markdown.d.ts.map +0 -1
  289. package/dist/ui/opencode-markdown.js +0 -169
  290. package/dist/ui/opencode-markdown.js.map +0 -1
  291. package/dist/ui/theme.d.ts +0 -68
  292. package/dist/ui/theme.d.ts.map +0 -1
  293. package/dist/ui/theme.js +0 -80
  294. package/dist/ui/theme.js.map +0 -1
  295. package/dist/utils/time.d.ts +0 -2
  296. package/dist/utils/time.d.ts.map +0 -1
  297. package/dist/utils/time.js +0 -7
  298. package/dist/utils/time.js.map +0 -1
@@ -0,0 +1,229 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
2
+
3
+ vi.mock('./tool-confirmation', () => ({
4
+ resolveToolConfirmDecision: vi.fn(),
5
+ }));
6
+
7
+ vi.mock('./source-modules', () => ({
8
+ getSourceModules: vi.fn(),
9
+ resolveWorkspaceRoot: vi.fn(),
10
+ }));
11
+
12
+ vi.mock('../../../../src/agent/prompts/system', () => ({
13
+ buildSystemPrompt: vi.fn(() => 'Test system prompt'),
14
+ }));
15
+
16
+ import { disposeAgentRuntime, runAgentPrompt } from './runtime';
17
+ import * as sourceModules from './source-modules';
18
+ import type { AgentEventHandlers } from './types';
19
+
20
+ describe('runAgentPrompt usage forwarding', () => {
21
+ const mockGetSourceModules = sourceModules.getSourceModules as unknown as ReturnType<
22
+ typeof vi.fn
23
+ >;
24
+ const mockResolveWorkspaceRoot = sourceModules.resolveWorkspaceRoot as unknown as ReturnType<
25
+ typeof vi.fn
26
+ >;
27
+ const originalApiKey = process.env.TEST_API_KEY;
28
+ const loggerFromEnv = {
29
+ calls: [] as string[],
30
+ info(message: string) {
31
+ this.calls.push(`info:${message}`);
32
+ },
33
+ warn(message: string) {
34
+ this.calls.push(`warn:${message}`);
35
+ },
36
+ error(message: string) {
37
+ this.calls.push(`error:${message}`);
38
+ },
39
+ close: vi.fn(),
40
+ };
41
+ let receivedAgentConfig: Record<string, unknown> | undefined;
42
+
43
+ beforeEach(() => {
44
+ vi.clearAllMocks();
45
+ process.env.TEST_API_KEY = 'test-key';
46
+ mockResolveWorkspaceRoot.mockReturnValue('/test/workspace');
47
+ receivedAgentConfig = undefined;
48
+ loggerFromEnv.calls.length = 0;
49
+
50
+ class FakeToolManager {
51
+ registerTool = vi.fn();
52
+ getTools = vi.fn(() => []);
53
+ }
54
+
55
+ class FakeTool {
56
+ toToolSchema() {
57
+ return {
58
+ type: 'function',
59
+ function: {
60
+ name: 'fake_tool',
61
+ },
62
+ };
63
+ }
64
+ }
65
+
66
+ class FakeAgent {
67
+ constructor(_provider: unknown, _toolManager: unknown, config: Record<string, unknown>) {
68
+ receivedAgentConfig = config;
69
+ }
70
+
71
+ on = vi.fn();
72
+ off = vi.fn();
73
+ }
74
+
75
+ class FakeAppService {
76
+ async listContextMessages() {
77
+ return [];
78
+ }
79
+
80
+ async runForeground(
81
+ _request: unknown,
82
+ callbacks?: {
83
+ onUsage?: (usage: {
84
+ sequence: number;
85
+ stepIndex: number;
86
+ messageId: string;
87
+ usage: {
88
+ prompt_tokens: number;
89
+ completion_tokens: number;
90
+ total_tokens: number;
91
+ };
92
+ cumulativeUsage: {
93
+ prompt_tokens: number;
94
+ completion_tokens: number;
95
+ total_tokens: number;
96
+ };
97
+ contextTokens?: number;
98
+ contextLimitTokens?: number;
99
+ contextUsagePercent?: number;
100
+ }) => void | Promise<void>;
101
+ }
102
+ ) {
103
+ await callbacks?.onUsage?.({
104
+ sequence: 1,
105
+ stepIndex: 1,
106
+ messageId: 'msg_usage',
107
+ usage: {
108
+ prompt_tokens: 10,
109
+ completion_tokens: 5,
110
+ total_tokens: 15,
111
+ },
112
+ cumulativeUsage: {
113
+ prompt_tokens: 10,
114
+ completion_tokens: 5,
115
+ total_tokens: 15,
116
+ },
117
+ contextTokens: 123,
118
+ contextLimitTokens: 1000,
119
+ contextUsagePercent: 12.3,
120
+ });
121
+
122
+ return {
123
+ executionId: 'exec_usage',
124
+ conversationId: 'conv_usage',
125
+ messages: [
126
+ {
127
+ messageId: 'msg_assistant',
128
+ role: 'assistant',
129
+ type: 'assistant-text',
130
+ content: 'done',
131
+ },
132
+ ],
133
+ finishReason: 'stop' as const,
134
+ steps: 1,
135
+ run: {},
136
+ };
137
+ }
138
+ }
139
+
140
+ mockGetSourceModules.mockResolvedValue({
141
+ ProviderRegistry: {
142
+ getModelIds: () => ['test-model'],
143
+ getModelConfig: () => ({
144
+ name: 'Test Model',
145
+ envApiKey: 'TEST_API_KEY',
146
+ model: 'test-model',
147
+ }),
148
+ createFromEnv: () => ({}),
149
+ },
150
+ loadEnvFiles: vi.fn().mockResolvedValue([]),
151
+ loadConfigToEnv: vi.fn().mockReturnValue([]),
152
+ createLoggerFromEnv: vi.fn(() => loggerFromEnv),
153
+ createAgentLoggerAdapter: vi.fn((logger: typeof loggerFromEnv) => ({
154
+ info: (message: string, _context?: Record<string, unknown>, _data?: unknown) =>
155
+ logger.info(message),
156
+ warn: (message: string, _context?: Record<string, unknown>, _data?: unknown) =>
157
+ logger.warn(message),
158
+ error: (message: string, _error?: unknown, _context?: Record<string, unknown>) =>
159
+ logger.error(message),
160
+ })),
161
+ StatelessAgent: FakeAgent,
162
+ AgentAppService: FakeAppService,
163
+ createSqliteAgentAppStore: () => ({
164
+ prepare: vi.fn().mockResolvedValue(undefined),
165
+ close: vi.fn().mockResolvedValue(undefined),
166
+ }),
167
+ DefaultToolManager: FakeToolManager,
168
+ BashTool: FakeTool,
169
+ WriteFileTool: FakeTool,
170
+ FileReadTool: FakeTool,
171
+ FileEditTool: FakeTool,
172
+ FileHistoryListTool: FakeTool,
173
+ FileHistoryRestoreTool: FakeTool,
174
+ GlobTool: FakeTool,
175
+ GrepTool: FakeTool,
176
+ SkillTool: FakeTool,
177
+ TaskTool: FakeTool,
178
+ TaskCreateTool: FakeTool,
179
+ TaskGetTool: FakeTool,
180
+ TaskListTool: FakeTool,
181
+ TaskUpdateTool: FakeTool,
182
+ TaskStopTool: FakeTool,
183
+ TaskOutputTool: FakeTool,
184
+ TaskStore: class {},
185
+ RealSubagentRunnerAdapter: class {},
186
+ } as unknown as Awaited<ReturnType<typeof sourceModules.getSourceModules>>);
187
+ });
188
+
189
+ afterEach(async () => {
190
+ await disposeAgentRuntime();
191
+ if (originalApiKey === undefined) {
192
+ delete process.env.TEST_API_KEY;
193
+ } else {
194
+ process.env.TEST_API_KEY = originalApiKey;
195
+ }
196
+ });
197
+
198
+ it('forwards usage events to TUI handlers and returns final usage', async () => {
199
+ const onUsage = vi.fn();
200
+ const handlers = {
201
+ onUsage,
202
+ } as AgentEventHandlers;
203
+
204
+ const result = await runAgentPrompt('show usage', handlers);
205
+
206
+ expect(onUsage).toHaveBeenCalledTimes(1);
207
+ expect(onUsage).toHaveBeenCalledWith(
208
+ expect.objectContaining({
209
+ promptTokens: 10,
210
+ completionTokens: 5,
211
+ totalTokens: 15,
212
+ cumulativePromptTokens: 10,
213
+ cumulativeCompletionTokens: 5,
214
+ cumulativeTotalTokens: 15,
215
+ })
216
+ );
217
+ expect(result.usage).toEqual(
218
+ expect.objectContaining({
219
+ promptTokens: 10,
220
+ completionTokens: 5,
221
+ totalTokens: 15,
222
+ })
223
+ );
224
+ expect(receivedAgentConfig?.logger).not.toBe(loggerFromEnv);
225
+ expect(typeof (receivedAgentConfig?.logger as { info?: unknown }).info).toBe('function');
226
+ (receivedAgentConfig?.logger as { info?: (message: string) => void }).info?.('bound');
227
+ expect(loggerFromEnv.calls).toContain('info:bound');
228
+ });
229
+ });
@@ -0,0 +1,57 @@
1
+ import { afterEach, describe, expect, it, vi } from 'vitest';
2
+ import { resolve } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+
5
+ import { resolveRepoRoot, resolveWorkspaceRoot } from './source-modules';
6
+
7
+ const runtimeDir = fileURLToPath(new URL('.', import.meta.url));
8
+ const packageRoot = resolve(runtimeDir, '../../..');
9
+ const workspaceRoot = resolve(runtimeDir, '../../../..');
10
+ const originalAgentRepoRoot = process.env.AGENT_REPO_ROOT;
11
+ const originalAgentWorkdir = process.env.AGENT_WORKDIR;
12
+
13
+ afterEach(() => {
14
+ vi.restoreAllMocks();
15
+
16
+ if (originalAgentRepoRoot === undefined) {
17
+ delete process.env.AGENT_REPO_ROOT;
18
+ } else {
19
+ process.env.AGENT_REPO_ROOT = originalAgentRepoRoot;
20
+ }
21
+
22
+ if (originalAgentWorkdir === undefined) {
23
+ delete process.env.AGENT_WORKDIR;
24
+ } else {
25
+ process.env.AGENT_WORKDIR = originalAgentWorkdir;
26
+ }
27
+ });
28
+
29
+ describe('resolveRepoRoot', () => {
30
+ it('finds the workspace root when launched from opentui-agent-cli', () => {
31
+ vi.spyOn(process, 'cwd').mockReturnValue(packageRoot);
32
+
33
+ expect(resolveRepoRoot()).toBe(workspaceRoot);
34
+ });
35
+
36
+ it('respects AGENT_REPO_ROOT when provided', () => {
37
+ process.env.AGENT_REPO_ROOT = workspaceRoot;
38
+ vi.spyOn(process, 'cwd').mockReturnValue('D:\\temp\\example-workspace');
39
+
40
+ expect(resolveRepoRoot()).toBe(workspaceRoot);
41
+ });
42
+ });
43
+
44
+ describe('resolveWorkspaceRoot', () => {
45
+ it('defaults to the current working directory', () => {
46
+ vi.spyOn(process, 'cwd').mockReturnValue(packageRoot);
47
+
48
+ expect(resolveWorkspaceRoot()).toBe(packageRoot);
49
+ });
50
+
51
+ it('respects AGENT_WORKDIR when provided', () => {
52
+ process.env.AGENT_WORKDIR = workspaceRoot;
53
+ vi.spyOn(process, 'cwd').mockReturnValue(packageRoot);
54
+
55
+ expect(resolveWorkspaceRoot()).toBe(workspaceRoot);
56
+ });
57
+ });
@@ -0,0 +1,353 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { dirname, resolve } from 'node:path';
3
+ import { fileURLToPath, pathToFileURL } from 'node:url';
4
+
5
+ import type { MessageContent } from '../../types/message-content';
6
+
7
+ type ProviderModelConfig = {
8
+ name: string;
9
+ envApiKey: string;
10
+ provider?: string;
11
+ model?: string;
12
+ LLMMAX_TOKENS?: number;
13
+ modalities?: {
14
+ image?: boolean;
15
+ audio?: boolean;
16
+ video?: boolean;
17
+ };
18
+ };
19
+
20
+ export type ProviderRegistryLike = {
21
+ getModelIds: () => string[];
22
+ getModelConfig: (modelId: string) => ProviderModelConfig;
23
+ createFromEnv: (modelId: string, options?: Record<string, unknown>) => unknown;
24
+ };
25
+
26
+ export type ToolDecisionLike = {
27
+ approved: boolean;
28
+ message?: string;
29
+ };
30
+
31
+ export type ToolConfirmEventLike = {
32
+ toolCallId: string;
33
+ toolName: string;
34
+ arguments: string;
35
+ reason?: string;
36
+ metadata?: Record<string, unknown>;
37
+ resolve: (decision: ToolDecisionLike) => void;
38
+ };
39
+
40
+ export type AgentV4MessageLike = {
41
+ messageId: string;
42
+ role: 'system' | 'user' | 'assistant' | 'tool';
43
+ type: string;
44
+ content: unknown;
45
+ tool_call_id?: string;
46
+ usage?: {
47
+ prompt_tokens?: number;
48
+ completion_tokens?: number;
49
+ total_tokens?: number;
50
+ };
51
+ };
52
+
53
+ export type CliEventEnvelopeLike = {
54
+ conversationId: string;
55
+ executionId: string;
56
+ seq: number;
57
+ eventType: string;
58
+ data: unknown;
59
+ createdAt: number;
60
+ };
61
+
62
+ export type AgentAppRunResultLike = {
63
+ executionId: string;
64
+ conversationId: string;
65
+ messages: AgentV4MessageLike[];
66
+ finishReason: 'stop' | 'max_steps' | 'error';
67
+ steps: number;
68
+ run: {
69
+ errorMessage?: string;
70
+ };
71
+ };
72
+
73
+ type AgentAppRunRequestLike = {
74
+ conversationId: string;
75
+ userInput: MessageContent;
76
+ historyMessages?: AgentV4MessageLike[];
77
+ systemPrompt?: string;
78
+ tools?: Array<{ type: string; function: Record<string, unknown> }>;
79
+ config?: Record<string, unknown>;
80
+ maxSteps?: number;
81
+ contextLimitTokens?: number;
82
+ abortSignal?: AbortSignal;
83
+ modelLabel?: string;
84
+ };
85
+
86
+ export type AgentAppUsageLike = {
87
+ sequence: number;
88
+ stepIndex: number;
89
+ messageId: string;
90
+ usage: {
91
+ prompt_tokens: number;
92
+ completion_tokens: number;
93
+ total_tokens: number;
94
+ };
95
+ cumulativeUsage: {
96
+ prompt_tokens: number;
97
+ completion_tokens: number;
98
+ total_tokens: number;
99
+ };
100
+ contextTokens?: number;
101
+ contextLimitTokens?: number;
102
+ contextUsagePercent?: number;
103
+ };
104
+
105
+ export type AgentAppContextUsageLike = {
106
+ stepIndex: number;
107
+ messageCount: number;
108
+ contextTokens: number;
109
+ contextLimitTokens: number;
110
+ contextUsagePercent: number;
111
+ };
112
+
113
+ type AgentAppRunCallbacksLike = {
114
+ onEvent?: (event: CliEventEnvelopeLike) => void | Promise<void>;
115
+ onContextUsage?: (usage: AgentAppContextUsageLike) => void | Promise<void>;
116
+ onUsage?: (usage: AgentAppUsageLike) => void | Promise<void>;
117
+ onError?: (error: unknown) => void | Promise<void>;
118
+ };
119
+
120
+ export type AgentAppServiceLike = {
121
+ runForeground: (
122
+ request: AgentAppRunRequestLike,
123
+ callbacks?: AgentAppRunCallbacksLike
124
+ ) => Promise<AgentAppRunResultLike>;
125
+ listContextMessages: (conversationId: string) => Promise<AgentV4MessageLike[]>;
126
+ };
127
+
128
+ type AgentLoggerLike = {
129
+ debug?: (message: string, context?: Record<string, unknown>, data?: unknown) => void;
130
+ info?: (message: string, context?: Record<string, unknown>, data?: unknown) => void;
131
+ warn?: (message: string, context?: Record<string, unknown>, data?: unknown) => void;
132
+ error?: (message: string, error?: unknown, context?: Record<string, unknown>) => void;
133
+ };
134
+
135
+ export type StatelessAgentLike = {
136
+ on: (eventName: 'tool_confirm', listener: (event: ToolConfirmEventLike) => void) => void;
137
+ off: (eventName: 'tool_confirm', listener: (event: ToolConfirmEventLike) => void) => void;
138
+ };
139
+
140
+ export type ToolManagerLike = {
141
+ registerTool: (tool: unknown) => void;
142
+ getTools: () => Array<{ name?: string; toToolSchema?: () => unknown }>;
143
+ };
144
+
145
+ export type AgentAppStoreLike = {
146
+ close: () => Promise<void>;
147
+ };
148
+
149
+ type StatelessAgentCtor = new (
150
+ provider: unknown,
151
+ toolExecutor: ToolManagerLike,
152
+ config: Record<string, unknown>
153
+ ) => StatelessAgentLike;
154
+ type AgentAppServiceCtor = new (deps: {
155
+ agent: StatelessAgentLike;
156
+ executionStore: AgentAppStoreLike;
157
+ eventStore: AgentAppStoreLike;
158
+ messageStore: AgentAppStoreLike;
159
+ }) => AgentAppServiceLike;
160
+ type ToolManagerCtor = new (config?: Record<string, unknown>) => ToolManagerLike;
161
+ type ToolCtor = new (options?: Record<string, unknown>) => unknown;
162
+ type TaskStoreCtor = new (options?: Record<string, unknown>) => unknown;
163
+ type TaskRunnerCtor = new (options: Record<string, unknown>) => unknown;
164
+
165
+ export type SourceModules = {
166
+ repoRoot: string;
167
+ ProviderRegistry: ProviderRegistryLike;
168
+ loadEnvFiles: (cwd?: string) => Promise<string[]>;
169
+ loadConfigToEnv: (options?: Record<string, unknown>) => string[];
170
+ createLoggerFromEnv: (env?: NodeJS.ProcessEnv, cwd?: string) => unknown;
171
+ createAgentLoggerAdapter: (
172
+ logger: Record<string, unknown>,
173
+ baseContext?: Record<string, unknown>
174
+ ) => AgentLoggerLike;
175
+ StatelessAgent: StatelessAgentCtor;
176
+ AgentAppService: AgentAppServiceCtor;
177
+ createSqliteAgentAppStore: (dbPath: string) => AgentAppStoreLike;
178
+ DefaultToolManager: ToolManagerCtor;
179
+ BashTool: ToolCtor;
180
+ WriteFileTool: ToolCtor;
181
+ FileReadTool: ToolCtor;
182
+ FileEditTool: ToolCtor;
183
+ FileHistoryListTool: ToolCtor;
184
+ FileHistoryRestoreTool: ToolCtor;
185
+ GlobTool: ToolCtor;
186
+ GrepTool: ToolCtor;
187
+ SkillTool: ToolCtor;
188
+ TaskTool: ToolCtor;
189
+ TaskCreateTool: ToolCtor;
190
+ TaskGetTool: ToolCtor;
191
+ TaskListTool: ToolCtor;
192
+ TaskUpdateTool: ToolCtor;
193
+ TaskStopTool: ToolCtor;
194
+ TaskOutputTool: ToolCtor;
195
+ TaskStore: TaskStoreCtor;
196
+ RealSubagentRunnerAdapter: TaskRunnerCtor;
197
+ };
198
+
199
+ let modulesPromise: Promise<SourceModules> | null = null;
200
+
201
+ const moduleDir = dirname(fileURLToPath(import.meta.url));
202
+ const requiredSourceModulePaths = [
203
+ 'src/providers/index.ts',
204
+ 'src/config/index.ts',
205
+ 'src/agent/app/index.ts',
206
+ ] as const;
207
+
208
+ const hasRequiredSourceModules = (root: string) =>
209
+ requiredSourceModulePaths.every(relativePath => existsSync(resolve(root, relativePath)));
210
+
211
+ export const resolveRepoRoot = () => {
212
+ const cwd = process.cwd();
213
+ const explicit = process.env.AGENT_REPO_ROOT?.trim();
214
+ const candidates = [
215
+ explicit ? resolve(explicit) : null,
216
+ resolve(moduleDir, '../../../../'),
217
+ resolve(moduleDir, '../../..'),
218
+ resolve(cwd),
219
+ resolve(cwd, '..'),
220
+ ].filter((candidate): candidate is string => Boolean(candidate));
221
+
222
+ for (const candidate of new Set(candidates)) {
223
+ if (hasRequiredSourceModules(candidate)) {
224
+ return candidate;
225
+ }
226
+ }
227
+
228
+ throw new Error(
229
+ `Unable to resolve agent source repo root from ${cwd}. Set AGENT_REPO_ROOT to override.`
230
+ );
231
+ };
232
+
233
+ export const resolveWorkspaceRoot = () => {
234
+ const explicit = process.env.AGENT_WORKDIR?.trim();
235
+ if (explicit) {
236
+ return resolve(explicit);
237
+ }
238
+ return resolve(process.cwd());
239
+ };
240
+
241
+ const toModuleUrl = (path: string) => pathToFileURL(path).href;
242
+
243
+ const getRequiredExport = <T>(moduleObj: Record<string, unknown>, name: string): T => {
244
+ const value = moduleObj[name];
245
+ if (!value) {
246
+ throw new Error(`Missing export ${name}.`);
247
+ }
248
+ return value as T;
249
+ };
250
+
251
+ const loadSourceModules = async (): Promise<SourceModules> => {
252
+ const repoRoot = resolveRepoRoot();
253
+
254
+ const [
255
+ providerMod,
256
+ configMod,
257
+ appMod,
258
+ agentV4Mod,
259
+ agentLoggerMod,
260
+ toolManagerMod,
261
+ bashToolMod,
262
+ writeToolMod,
263
+ fileReadToolMod,
264
+ fileEditToolMod,
265
+ fileHistoryListToolMod,
266
+ fileHistoryRestoreToolMod,
267
+ globToolMod,
268
+ grepToolMod,
269
+ skillToolMod,
270
+ taskToolMod,
271
+ taskCreateToolMod,
272
+ taskGetToolMod,
273
+ taskListToolMod,
274
+ taskUpdateToolMod,
275
+ taskStopToolMod,
276
+ taskOutputToolMod,
277
+ taskStoreMod,
278
+ taskRunnerAdapterMod,
279
+ ] = await Promise.all([
280
+ import(toModuleUrl(resolve(repoRoot, 'src/providers/index.ts'))),
281
+ import(toModuleUrl(resolve(repoRoot, 'src/config/index.ts'))),
282
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/app/index.ts'))),
283
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/agent/index.ts'))),
284
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/agent/logger.ts'))),
285
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/tool-manager.ts'))),
286
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/bash.ts'))),
287
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/write-file.ts'))),
288
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/file-read-tool.ts'))),
289
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/file-edit-tool.ts'))),
290
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/file-history-list.ts'))),
291
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/file-history-restore.ts'))),
292
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/glob.ts'))),
293
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/grep.ts'))),
294
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/skill-tool.ts'))),
295
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task.ts'))),
296
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-create.ts'))),
297
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-get.ts'))),
298
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-list.ts'))),
299
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-update.ts'))),
300
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-stop.ts'))),
301
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-output.ts'))),
302
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-store.ts'))),
303
+ import(toModuleUrl(resolve(repoRoot, 'src/agent/tool/task-runner-adapter.ts'))),
304
+ ]);
305
+
306
+ return {
307
+ repoRoot,
308
+ ProviderRegistry: getRequiredExport<ProviderRegistryLike>(providerMod, 'ProviderRegistry'),
309
+ loadEnvFiles: getRequiredExport(configMod, 'loadEnvFiles'),
310
+ loadConfigToEnv: getRequiredExport(configMod, 'loadConfigToEnv'),
311
+ createLoggerFromEnv: getRequiredExport(configMod, 'createLoggerFromEnv'),
312
+ createAgentLoggerAdapter: getRequiredExport(agentLoggerMod, 'createAgentLoggerAdapter'),
313
+ StatelessAgent: getRequiredExport<StatelessAgentCtor>(agentV4Mod, 'StatelessAgent'),
314
+ AgentAppService: getRequiredExport<AgentAppServiceCtor>(appMod, 'AgentAppService'),
315
+ createSqliteAgentAppStore: getRequiredExport<(dbPath: string) => AgentAppStoreLike>(
316
+ appMod,
317
+ 'createSqliteAgentAppStore'
318
+ ),
319
+ DefaultToolManager: getRequiredExport<ToolManagerCtor>(toolManagerMod, 'DefaultToolManager'),
320
+ BashTool: getRequiredExport<ToolCtor>(bashToolMod, 'BashTool'),
321
+ WriteFileTool: getRequiredExport<ToolCtor>(writeToolMod, 'WriteFileTool'),
322
+ FileReadTool: getRequiredExport<ToolCtor>(fileReadToolMod, 'FileReadTool'),
323
+ FileEditTool: getRequiredExport<ToolCtor>(fileEditToolMod, 'FileEditTool'),
324
+ FileHistoryListTool: getRequiredExport<ToolCtor>(fileHistoryListToolMod, 'FileHistoryListTool'),
325
+ FileHistoryRestoreTool: getRequiredExport<ToolCtor>(
326
+ fileHistoryRestoreToolMod,
327
+ 'FileHistoryRestoreTool'
328
+ ),
329
+ GlobTool: getRequiredExport<ToolCtor>(globToolMod, 'GlobTool'),
330
+ GrepTool: getRequiredExport<ToolCtor>(grepToolMod, 'GrepTool'),
331
+ SkillTool: getRequiredExport<ToolCtor>(skillToolMod, 'SkillTool'),
332
+ TaskTool: getRequiredExport<ToolCtor>(taskToolMod, 'TaskTool'),
333
+ TaskCreateTool: getRequiredExport<ToolCtor>(taskCreateToolMod, 'TaskCreateTool'),
334
+ TaskGetTool: getRequiredExport<ToolCtor>(taskGetToolMod, 'TaskGetTool'),
335
+ TaskListTool: getRequiredExport<ToolCtor>(taskListToolMod, 'TaskListTool'),
336
+ TaskUpdateTool: getRequiredExport<ToolCtor>(taskUpdateToolMod, 'TaskUpdateTool'),
337
+ TaskStopTool: getRequiredExport<ToolCtor>(taskStopToolMod, 'TaskStopTool'),
338
+ TaskOutputTool: getRequiredExport<ToolCtor>(taskOutputToolMod, 'TaskOutputTool'),
339
+ TaskStore: getRequiredExport<TaskStoreCtor>(taskStoreMod, 'TaskStore'),
340
+ RealSubagentRunnerAdapter: getRequiredExport<TaskRunnerCtor>(
341
+ taskRunnerAdapterMod,
342
+ 'RealSubagentRunnerAdapter'
343
+ ),
344
+ };
345
+ };
346
+
347
+ export const getSourceModules = async () => {
348
+ modulesPromise ??= loadSourceModules().catch(error => {
349
+ modulesPromise = null;
350
+ throw error;
351
+ });
352
+ return modulesPromise;
353
+ };
@@ -0,0 +1,65 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import type { AgentToolUseEvent } from './types';
4
+ import { ToolCallBuffer } from './tool-call-buffer';
5
+
6
+ const createFileReadToolUseEvent = (callId: string, path: string): AgentToolUseEvent => ({
7
+ id: callId,
8
+ function: {
9
+ name: 'file_read',
10
+ arguments: JSON.stringify({ path }),
11
+ },
12
+ });
13
+
14
+ describe('ToolCallBuffer', () => {
15
+ it('keeps llm-planned tool calls hidden until tool execution starts', () => {
16
+ const buffer = new ToolCallBuffer();
17
+ const emitted: AgentToolUseEvent[] = [];
18
+
19
+ buffer.register(createFileReadToolUseEvent('call_1', '/tmp/a.ts'), event =>
20
+ emitted.push(event)
21
+ );
22
+ buffer.register(createFileReadToolUseEvent('call_1', '/tmp/a.ts'), event =>
23
+ emitted.push(event)
24
+ );
25
+ buffer.register(createFileReadToolUseEvent('call_2', '/tmp/b.ts'), event =>
26
+ emitted.push(event)
27
+ );
28
+ buffer.register(createFileReadToolUseEvent('call_3', '/tmp/c.ts'), event =>
29
+ emitted.push(event)
30
+ );
31
+
32
+ expect(emitted).toEqual([]);
33
+
34
+ buffer.flush(event => emitted.push(event));
35
+
36
+ expect(emitted).toEqual([
37
+ createFileReadToolUseEvent('call_1', '/tmp/a.ts'),
38
+ createFileReadToolUseEvent('call_2', '/tmp/b.ts'),
39
+ createFileReadToolUseEvent('call_3', '/tmp/c.ts'),
40
+ ]);
41
+ });
42
+
43
+ it('emits a planned tool call as soon as its stream starts', () => {
44
+ const buffer = new ToolCallBuffer();
45
+ const emitted: AgentToolUseEvent[] = [];
46
+
47
+ buffer.register(createFileReadToolUseEvent('call_1', '/tmp/a.ts'), event =>
48
+ emitted.push(event)
49
+ );
50
+ buffer.register(createFileReadToolUseEvent('call_2', '/tmp/b.ts'), event =>
51
+ emitted.push(event)
52
+ );
53
+
54
+ buffer.ensureEmitted('call_2', event => emitted.push(event));
55
+
56
+ expect(emitted).toEqual([createFileReadToolUseEvent('call_2', '/tmp/b.ts')]);
57
+
58
+ buffer.flush(event => emitted.push(event));
59
+
60
+ expect(emitted).toEqual([
61
+ createFileReadToolUseEvent('call_2', '/tmp/b.ts'),
62
+ createFileReadToolUseEvent('call_1', '/tmp/a.ts'),
63
+ ]);
64
+ });
65
+ });