@google/gemini-cli-core 0.10.0-preview.3 → 0.11.0-nightly.20251020.a96f0659

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 (175) hide show
  1. package/README.md +9 -0
  2. package/dist/google-gemini-cli-core-0.11.0-nightly.20251015.203bad7c.tgz +0 -0
  3. package/dist/src/agents/codebase-investigator.js +2 -5
  4. package/dist/src/agents/codebase-investigator.js.map +1 -1
  5. package/dist/src/agents/executor.js +6 -13
  6. package/dist/src/agents/executor.js.map +1 -1
  7. package/dist/src/agents/executor.test.js +36 -31
  8. package/dist/src/agents/executor.test.js.map +1 -1
  9. package/dist/src/config/config.d.ts +1 -3
  10. package/dist/src/config/config.js +0 -5
  11. package/dist/src/config/config.js.map +1 -1
  12. package/dist/src/config/config.test.js +0 -4
  13. package/dist/src/config/config.test.js.map +1 -1
  14. package/dist/src/confirmation-bus/types.d.ts +5 -0
  15. package/dist/src/core/client.test.js +0 -1
  16. package/dist/src/core/client.test.js.map +1 -1
  17. package/dist/src/core/coreToolScheduler.d.ts +6 -0
  18. package/dist/src/core/coreToolScheduler.js +30 -6
  19. package/dist/src/core/coreToolScheduler.js.map +1 -1
  20. package/dist/src/core/coreToolScheduler.test.js +37 -3
  21. package/dist/src/core/coreToolScheduler.test.js.map +1 -1
  22. package/dist/src/core/geminiChat.js +1 -0
  23. package/dist/src/core/geminiChat.js.map +1 -1
  24. package/dist/src/core/nonInteractiveToolExecutor.test.js +3 -0
  25. package/dist/src/core/nonInteractiveToolExecutor.test.js.map +1 -1
  26. package/dist/src/core/prompts.js +13 -20
  27. package/dist/src/core/prompts.js.map +1 -1
  28. package/dist/src/generated/git-commit.d.ts +2 -2
  29. package/dist/src/generated/git-commit.js +2 -2
  30. package/dist/src/generated/git-commit.js.map +1 -1
  31. package/dist/src/index.d.ts +2 -0
  32. package/dist/src/index.js +2 -0
  33. package/dist/src/index.js.map +1 -1
  34. package/dist/src/output/stream-json-formatter.d.ts +32 -0
  35. package/dist/src/output/stream-json-formatter.js +52 -0
  36. package/dist/src/output/stream-json-formatter.js.map +1 -0
  37. package/dist/src/output/stream-json-formatter.test.js +479 -0
  38. package/dist/src/output/stream-json-formatter.test.js.map +1 -0
  39. package/dist/src/output/types.d.ts +63 -1
  40. package/dist/src/output/types.js +11 -0
  41. package/dist/src/output/types.js.map +1 -1
  42. package/dist/src/services/shellExecutionService.js +9 -9
  43. package/dist/src/services/shellExecutionService.js.map +1 -1
  44. package/dist/src/services/shellExecutionService.test.js +16 -11
  45. package/dist/src/services/shellExecutionService.test.js.map +1 -1
  46. package/dist/src/telemetry/activity-monitor.d.ts +116 -0
  47. package/dist/src/telemetry/activity-monitor.js +208 -0
  48. package/dist/src/telemetry/activity-monitor.js.map +1 -0
  49. package/dist/src/telemetry/activity-monitor.test.d.ts +6 -0
  50. package/dist/src/telemetry/activity-monitor.test.js +248 -0
  51. package/dist/src/telemetry/activity-monitor.test.js.map +1 -0
  52. package/dist/src/telemetry/clearcut-logger/clearcut-logger.d.ts +1 -0
  53. package/dist/src/telemetry/clearcut-logger/clearcut-logger.js +7 -7
  54. package/dist/src/telemetry/clearcut-logger/clearcut-logger.js.map +1 -1
  55. package/dist/src/telemetry/clearcut-logger/clearcut-logger.test.js +3 -10
  56. package/dist/src/telemetry/clearcut-logger/clearcut-logger.test.js.map +1 -1
  57. package/dist/src/telemetry/index.d.ts +2 -1
  58. package/dist/src/telemetry/index.js +2 -1
  59. package/dist/src/telemetry/index.js.map +1 -1
  60. package/dist/src/telemetry/loggers.test.js +0 -1
  61. package/dist/src/telemetry/loggers.test.js.map +1 -1
  62. package/dist/src/tools/edit.d.ts +0 -1
  63. package/dist/src/tools/edit.js +4 -5
  64. package/dist/src/tools/edit.js.map +1 -1
  65. package/dist/src/tools/edit.test.js +0 -1
  66. package/dist/src/tools/edit.test.js.map +1 -1
  67. package/dist/src/tools/glob.d.ts +0 -1
  68. package/dist/src/tools/glob.js +1 -2
  69. package/dist/src/tools/glob.js.map +1 -1
  70. package/dist/src/tools/grep.d.ts +0 -1
  71. package/dist/src/tools/grep.js +10 -5
  72. package/dist/src/tools/grep.js.map +1 -1
  73. package/dist/src/tools/ls.d.ts +0 -1
  74. package/dist/src/tools/ls.js +2 -2
  75. package/dist/src/tools/ls.js.map +1 -1
  76. package/dist/src/tools/memoryTool.d.ts +0 -1
  77. package/dist/src/tools/memoryTool.js +3 -3
  78. package/dist/src/tools/memoryTool.js.map +1 -1
  79. package/dist/src/tools/message-bus-integration.test.js +14 -1
  80. package/dist/src/tools/message-bus-integration.test.js.map +1 -1
  81. package/dist/src/tools/read-file.d.ts +0 -1
  82. package/dist/src/tools/read-file.js +3 -3
  83. package/dist/src/tools/read-file.js.map +1 -1
  84. package/dist/src/tools/read-many-files.d.ts +0 -1
  85. package/dist/src/tools/read-many-files.js +3 -3
  86. package/dist/src/tools/read-many-files.js.map +1 -1
  87. package/dist/src/tools/ripGrep.d.ts +0 -1
  88. package/dist/src/tools/ripGrep.js +2 -2
  89. package/dist/src/tools/ripGrep.js.map +1 -1
  90. package/dist/src/tools/shell.d.ts +0 -1
  91. package/dist/src/tools/shell.js +21 -58
  92. package/dist/src/tools/shell.js.map +1 -1
  93. package/dist/src/tools/shell.test.js +61 -8
  94. package/dist/src/tools/shell.test.js.map +1 -1
  95. package/dist/src/tools/smart-edit.d.ts +0 -1
  96. package/dist/src/tools/smart-edit.js +3 -4
  97. package/dist/src/tools/smart-edit.js.map +1 -1
  98. package/dist/src/tools/smart-edit.test.js +0 -1
  99. package/dist/src/tools/smart-edit.test.js.map +1 -1
  100. package/dist/src/tools/tool-names.d.ts +8 -0
  101. package/dist/src/tools/tool-names.js +8 -5
  102. package/dist/src/tools/tool-names.js.map +1 -1
  103. package/dist/src/tools/tools.d.ts +10 -6
  104. package/dist/src/tools/tools.js +19 -33
  105. package/dist/src/tools/tools.js.map +1 -1
  106. package/dist/src/tools/web-fetch.d.ts +3 -3
  107. package/dist/src/tools/web-fetch.js +43 -18
  108. package/dist/src/tools/web-fetch.js.map +1 -1
  109. package/dist/src/tools/web-fetch.test.js +261 -0
  110. package/dist/src/tools/web-fetch.test.js.map +1 -1
  111. package/dist/src/tools/web-search.d.ts +0 -1
  112. package/dist/src/tools/web-search.js +1 -2
  113. package/dist/src/tools/web-search.js.map +1 -1
  114. package/dist/src/tools/write-file.d.ts +0 -1
  115. package/dist/src/tools/write-file.js +2 -3
  116. package/dist/src/tools/write-file.js.map +1 -1
  117. package/dist/src/tools/write-file.test.js +0 -1
  118. package/dist/src/tools/write-file.test.js.map +1 -1
  119. package/dist/src/tools/write-todos.d.ts +1 -7
  120. package/dist/src/tools/write-todos.js +2 -3
  121. package/dist/src/tools/write-todos.js.map +1 -1
  122. package/dist/src/tools/write-todos.test.js +2 -2
  123. package/dist/src/tools/write-todos.test.js.map +1 -1
  124. package/dist/src/utils/debugLogger.d.ts +25 -0
  125. package/dist/src/utils/debugLogger.js +33 -0
  126. package/dist/src/utils/debugLogger.js.map +1 -0
  127. package/dist/src/utils/debugLogger.test.d.ts +6 -0
  128. package/dist/src/utils/debugLogger.test.js +67 -0
  129. package/dist/src/utils/debugLogger.test.js.map +1 -0
  130. package/dist/src/utils/delay.d.ts +16 -0
  131. package/dist/src/utils/delay.js +43 -0
  132. package/dist/src/utils/delay.js.map +1 -0
  133. package/dist/src/utils/delay.test.d.ts +6 -0
  134. package/dist/src/utils/delay.test.js +88 -0
  135. package/dist/src/utils/delay.test.js.map +1 -0
  136. package/dist/src/utils/editCorrector.js +5 -9
  137. package/dist/src/utils/editCorrector.js.map +1 -1
  138. package/dist/src/utils/editCorrector.test.js +3 -5
  139. package/dist/src/utils/editCorrector.test.js.map +1 -1
  140. package/dist/src/utils/environmentContext.js +0 -33
  141. package/dist/src/utils/environmentContext.js.map +1 -1
  142. package/dist/src/utils/environmentContext.test.js +0 -34
  143. package/dist/src/utils/environmentContext.test.js.map +1 -1
  144. package/dist/src/utils/fileUtils.d.ts +4 -0
  145. package/dist/src/utils/fileUtils.js +31 -0
  146. package/dist/src/utils/fileUtils.js.map +1 -1
  147. package/dist/src/utils/gitIgnoreParser.js +6 -5
  148. package/dist/src/utils/gitIgnoreParser.js.map +1 -1
  149. package/dist/src/utils/gitIgnoreParser.test.js +14 -0
  150. package/dist/src/utils/gitIgnoreParser.test.js.map +1 -1
  151. package/dist/src/utils/retry.d.ts +1 -0
  152. package/dist/src/utils/retry.js +14 -12
  153. package/dist/src/utils/retry.js.map +1 -1
  154. package/dist/src/utils/retry.test.js +17 -0
  155. package/dist/src/utils/retry.test.js.map +1 -1
  156. package/dist/src/utils/safeJsonStringify.d.ts +4 -4
  157. package/dist/src/utils/safeJsonStringify.js +31 -7
  158. package/dist/src/utils/safeJsonStringify.js.map +1 -1
  159. package/dist/src/utils/shell-utils.d.ts +2 -2
  160. package/dist/src/utils/shell-utils.js +307 -136
  161. package/dist/src/utils/shell-utils.js.map +1 -1
  162. package/dist/src/utils/shell-utils.test.js +126 -60
  163. package/dist/src/utils/shell-utils.test.js.map +1 -1
  164. package/dist/src/utils/tool-utils.d.ts +2 -2
  165. package/dist/src/utils/tool-utils.js +14 -5
  166. package/dist/src/utils/tool-utils.js.map +1 -1
  167. package/dist/tsconfig.tsbuildinfo +1 -1
  168. package/package.json +5 -3
  169. package/dist/google-gemini-cli-core-0.10.0-preview.2.tgz +0 -0
  170. package/dist/src/core/subagent.d.ts +0 -236
  171. package/dist/src/core/subagent.js +0 -482
  172. package/dist/src/core/subagent.js.map +0 -1
  173. package/dist/src/core/subagent.test.js +0 -556
  174. package/dist/src/core/subagent.test.js.map +0 -1
  175. /package/dist/src/{core/subagent.test.d.ts → output/stream-json-formatter.test.d.ts} +0 -0
@@ -0,0 +1,479 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2025 Google LLC
4
+ * SPDX-License-Identifier: Apache-2.0
5
+ */
6
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
7
+ import { StreamJsonFormatter } from './stream-json-formatter.js';
8
+ import { JsonStreamEventType } from './types.js';
9
+ import { ToolCallDecision } from '../telemetry/tool-call-decision.js';
10
+ describe('StreamJsonFormatter', () => {
11
+ let formatter;
12
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
13
+ let stdoutWriteSpy;
14
+ beforeEach(() => {
15
+ formatter = new StreamJsonFormatter();
16
+ stdoutWriteSpy = vi
17
+ .spyOn(process.stdout, 'write')
18
+ .mockImplementation(() => true);
19
+ });
20
+ afterEach(() => {
21
+ stdoutWriteSpy.mockRestore();
22
+ });
23
+ describe('formatEvent', () => {
24
+ it('should format init event as JSONL', () => {
25
+ const event = {
26
+ type: JsonStreamEventType.INIT,
27
+ timestamp: '2025-10-10T12:00:00.000Z',
28
+ session_id: 'test-session-123',
29
+ model: 'gemini-2.0-flash-exp',
30
+ };
31
+ const result = formatter.formatEvent(event);
32
+ expect(result).toBe(JSON.stringify(event) + '\n');
33
+ expect(JSON.parse(result.trim())).toEqual(event);
34
+ });
35
+ it('should format user message event', () => {
36
+ const event = {
37
+ type: JsonStreamEventType.MESSAGE,
38
+ timestamp: '2025-10-10T12:00:00.000Z',
39
+ role: 'user',
40
+ content: 'What is 2+2?',
41
+ };
42
+ const result = formatter.formatEvent(event);
43
+ expect(result).toBe(JSON.stringify(event) + '\n');
44
+ expect(JSON.parse(result.trim())).toEqual(event);
45
+ });
46
+ it('should format assistant message event with delta', () => {
47
+ const event = {
48
+ type: JsonStreamEventType.MESSAGE,
49
+ timestamp: '2025-10-10T12:00:00.000Z',
50
+ role: 'assistant',
51
+ content: '4',
52
+ delta: true,
53
+ };
54
+ const result = formatter.formatEvent(event);
55
+ expect(result).toBe(JSON.stringify(event) + '\n');
56
+ const parsed = JSON.parse(result.trim());
57
+ expect(parsed.delta).toBe(true);
58
+ });
59
+ it('should format tool_use event', () => {
60
+ const event = {
61
+ type: JsonStreamEventType.TOOL_USE,
62
+ timestamp: '2025-10-10T12:00:00.000Z',
63
+ tool_name: 'Read',
64
+ tool_id: 'read-123',
65
+ parameters: { file_path: '/path/to/file.txt' },
66
+ };
67
+ const result = formatter.formatEvent(event);
68
+ expect(result).toBe(JSON.stringify(event) + '\n');
69
+ expect(JSON.parse(result.trim())).toEqual(event);
70
+ });
71
+ it('should format tool_result event (success)', () => {
72
+ const event = {
73
+ type: JsonStreamEventType.TOOL_RESULT,
74
+ timestamp: '2025-10-10T12:00:00.000Z',
75
+ tool_id: 'read-123',
76
+ status: 'success',
77
+ output: 'File contents here',
78
+ };
79
+ const result = formatter.formatEvent(event);
80
+ expect(result).toBe(JSON.stringify(event) + '\n');
81
+ expect(JSON.parse(result.trim())).toEqual(event);
82
+ });
83
+ it('should format tool_result event (error)', () => {
84
+ const event = {
85
+ type: JsonStreamEventType.TOOL_RESULT,
86
+ timestamp: '2025-10-10T12:00:00.000Z',
87
+ tool_id: 'read-123',
88
+ status: 'error',
89
+ error: {
90
+ type: 'FILE_NOT_FOUND',
91
+ message: 'File not found',
92
+ },
93
+ };
94
+ const result = formatter.formatEvent(event);
95
+ expect(result).toBe(JSON.stringify(event) + '\n');
96
+ expect(JSON.parse(result.trim())).toEqual(event);
97
+ });
98
+ it('should format error event', () => {
99
+ const event = {
100
+ type: JsonStreamEventType.ERROR,
101
+ timestamp: '2025-10-10T12:00:00.000Z',
102
+ severity: 'warning',
103
+ message: 'Loop detected, stopping execution',
104
+ };
105
+ const result = formatter.formatEvent(event);
106
+ expect(result).toBe(JSON.stringify(event) + '\n');
107
+ expect(JSON.parse(result.trim())).toEqual(event);
108
+ });
109
+ it('should format result event with success status', () => {
110
+ const event = {
111
+ type: JsonStreamEventType.RESULT,
112
+ timestamp: '2025-10-10T12:00:00.000Z',
113
+ status: 'success',
114
+ stats: {
115
+ total_tokens: 100,
116
+ input_tokens: 50,
117
+ output_tokens: 50,
118
+ duration_ms: 1200,
119
+ tool_calls: 2,
120
+ },
121
+ };
122
+ const result = formatter.formatEvent(event);
123
+ expect(result).toBe(JSON.stringify(event) + '\n');
124
+ expect(JSON.parse(result.trim())).toEqual(event);
125
+ });
126
+ it('should format result event with error status', () => {
127
+ const event = {
128
+ type: JsonStreamEventType.RESULT,
129
+ timestamp: '2025-10-10T12:00:00.000Z',
130
+ status: 'error',
131
+ error: {
132
+ type: 'MaxSessionTurnsError',
133
+ message: 'Maximum session turns exceeded',
134
+ },
135
+ stats: {
136
+ total_tokens: 100,
137
+ input_tokens: 50,
138
+ output_tokens: 50,
139
+ duration_ms: 1200,
140
+ tool_calls: 0,
141
+ },
142
+ };
143
+ const result = formatter.formatEvent(event);
144
+ expect(result).toBe(JSON.stringify(event) + '\n');
145
+ expect(JSON.parse(result.trim())).toEqual(event);
146
+ });
147
+ it('should produce minified JSON without pretty-printing', () => {
148
+ const event = {
149
+ type: JsonStreamEventType.MESSAGE,
150
+ timestamp: '2025-10-10T12:00:00.000Z',
151
+ role: 'user',
152
+ content: 'Test',
153
+ };
154
+ const result = formatter.formatEvent(event);
155
+ // Should not contain multiple spaces or newlines (except trailing)
156
+ expect(result).not.toContain(' ');
157
+ expect(result.split('\n').length).toBe(2); // JSON + trailing newline
158
+ });
159
+ });
160
+ describe('emitEvent', () => {
161
+ it('should write formatted event to stdout', () => {
162
+ const event = {
163
+ type: JsonStreamEventType.INIT,
164
+ timestamp: '2025-10-10T12:00:00.000Z',
165
+ session_id: 'test-session',
166
+ model: 'gemini-2.0-flash-exp',
167
+ };
168
+ formatter.emitEvent(event);
169
+ expect(stdoutWriteSpy).toHaveBeenCalledTimes(1);
170
+ expect(stdoutWriteSpy).toHaveBeenCalledWith(JSON.stringify(event) + '\n');
171
+ });
172
+ it('should emit multiple events sequentially', () => {
173
+ const event1 = {
174
+ type: JsonStreamEventType.INIT,
175
+ timestamp: '2025-10-10T12:00:00.000Z',
176
+ session_id: 'test-session',
177
+ model: 'gemini-2.0-flash-exp',
178
+ };
179
+ const event2 = {
180
+ type: JsonStreamEventType.MESSAGE,
181
+ timestamp: '2025-10-10T12:00:01.000Z',
182
+ role: 'user',
183
+ content: 'Hello',
184
+ };
185
+ formatter.emitEvent(event1);
186
+ formatter.emitEvent(event2);
187
+ expect(stdoutWriteSpy).toHaveBeenCalledTimes(2);
188
+ expect(stdoutWriteSpy).toHaveBeenNthCalledWith(1, JSON.stringify(event1) + '\n');
189
+ expect(stdoutWriteSpy).toHaveBeenNthCalledWith(2, JSON.stringify(event2) + '\n');
190
+ });
191
+ });
192
+ describe('convertToStreamStats', () => {
193
+ it('should aggregate token counts from single model', () => {
194
+ const metrics = {
195
+ models: {
196
+ 'gemini-2.0-flash': {
197
+ api: {
198
+ totalRequests: 1,
199
+ totalErrors: 0,
200
+ totalLatencyMs: 1000,
201
+ },
202
+ tokens: {
203
+ prompt: 50,
204
+ candidates: 30,
205
+ total: 80,
206
+ cached: 0,
207
+ thoughts: 0,
208
+ tool: 0,
209
+ },
210
+ },
211
+ },
212
+ tools: {
213
+ totalCalls: 2,
214
+ totalSuccess: 2,
215
+ totalFail: 0,
216
+ totalDurationMs: 500,
217
+ totalDecisions: {
218
+ [ToolCallDecision.ACCEPT]: 0,
219
+ [ToolCallDecision.REJECT]: 0,
220
+ [ToolCallDecision.MODIFY]: 0,
221
+ [ToolCallDecision.AUTO_ACCEPT]: 2,
222
+ },
223
+ byName: {},
224
+ },
225
+ files: {
226
+ totalLinesAdded: 0,
227
+ totalLinesRemoved: 0,
228
+ },
229
+ };
230
+ const result = formatter.convertToStreamStats(metrics, 1200);
231
+ expect(result).toEqual({
232
+ total_tokens: 80,
233
+ input_tokens: 50,
234
+ output_tokens: 30,
235
+ duration_ms: 1200,
236
+ tool_calls: 2,
237
+ });
238
+ });
239
+ it('should aggregate token counts from multiple models', () => {
240
+ const metrics = {
241
+ models: {
242
+ 'gemini-2.0-flash': {
243
+ api: {
244
+ totalRequests: 1,
245
+ totalErrors: 0,
246
+ totalLatencyMs: 1000,
247
+ },
248
+ tokens: {
249
+ prompt: 50,
250
+ candidates: 30,
251
+ total: 80,
252
+ cached: 0,
253
+ thoughts: 0,
254
+ tool: 0,
255
+ },
256
+ },
257
+ 'gemini-1.5-pro': {
258
+ api: {
259
+ totalRequests: 1,
260
+ totalErrors: 0,
261
+ totalLatencyMs: 2000,
262
+ },
263
+ tokens: {
264
+ prompt: 100,
265
+ candidates: 70,
266
+ total: 170,
267
+ cached: 0,
268
+ thoughts: 0,
269
+ tool: 0,
270
+ },
271
+ },
272
+ },
273
+ tools: {
274
+ totalCalls: 5,
275
+ totalSuccess: 5,
276
+ totalFail: 0,
277
+ totalDurationMs: 1000,
278
+ totalDecisions: {
279
+ [ToolCallDecision.ACCEPT]: 0,
280
+ [ToolCallDecision.REJECT]: 0,
281
+ [ToolCallDecision.MODIFY]: 0,
282
+ [ToolCallDecision.AUTO_ACCEPT]: 5,
283
+ },
284
+ byName: {},
285
+ },
286
+ files: {
287
+ totalLinesAdded: 0,
288
+ totalLinesRemoved: 0,
289
+ },
290
+ };
291
+ const result = formatter.convertToStreamStats(metrics, 3000);
292
+ expect(result).toEqual({
293
+ total_tokens: 250, // 80 + 170
294
+ input_tokens: 150, // 50 + 100
295
+ output_tokens: 100, // 30 + 70
296
+ duration_ms: 3000,
297
+ tool_calls: 5,
298
+ });
299
+ });
300
+ it('should handle empty metrics', () => {
301
+ const metrics = {
302
+ models: {},
303
+ tools: {
304
+ totalCalls: 0,
305
+ totalSuccess: 0,
306
+ totalFail: 0,
307
+ totalDurationMs: 0,
308
+ totalDecisions: {
309
+ [ToolCallDecision.ACCEPT]: 0,
310
+ [ToolCallDecision.REJECT]: 0,
311
+ [ToolCallDecision.MODIFY]: 0,
312
+ [ToolCallDecision.AUTO_ACCEPT]: 0,
313
+ },
314
+ byName: {},
315
+ },
316
+ files: {
317
+ totalLinesAdded: 0,
318
+ totalLinesRemoved: 0,
319
+ },
320
+ };
321
+ const result = formatter.convertToStreamStats(metrics, 100);
322
+ expect(result).toEqual({
323
+ total_tokens: 0,
324
+ input_tokens: 0,
325
+ output_tokens: 0,
326
+ duration_ms: 100,
327
+ tool_calls: 0,
328
+ });
329
+ });
330
+ it('should use session-level tool calls count', () => {
331
+ const metrics = {
332
+ models: {},
333
+ tools: {
334
+ totalCalls: 3,
335
+ totalSuccess: 2,
336
+ totalFail: 1,
337
+ totalDurationMs: 500,
338
+ totalDecisions: {
339
+ [ToolCallDecision.ACCEPT]: 0,
340
+ [ToolCallDecision.REJECT]: 0,
341
+ [ToolCallDecision.MODIFY]: 0,
342
+ [ToolCallDecision.AUTO_ACCEPT]: 3,
343
+ },
344
+ byName: {
345
+ Read: {
346
+ count: 2,
347
+ success: 2,
348
+ fail: 0,
349
+ durationMs: 300,
350
+ decisions: {
351
+ [ToolCallDecision.ACCEPT]: 0,
352
+ [ToolCallDecision.REJECT]: 0,
353
+ [ToolCallDecision.MODIFY]: 0,
354
+ [ToolCallDecision.AUTO_ACCEPT]: 2,
355
+ },
356
+ },
357
+ Glob: {
358
+ count: 1,
359
+ success: 0,
360
+ fail: 1,
361
+ durationMs: 200,
362
+ decisions: {
363
+ [ToolCallDecision.ACCEPT]: 0,
364
+ [ToolCallDecision.REJECT]: 0,
365
+ [ToolCallDecision.MODIFY]: 0,
366
+ [ToolCallDecision.AUTO_ACCEPT]: 1,
367
+ },
368
+ },
369
+ },
370
+ },
371
+ files: {
372
+ totalLinesAdded: 0,
373
+ totalLinesRemoved: 0,
374
+ },
375
+ };
376
+ const result = formatter.convertToStreamStats(metrics, 1000);
377
+ expect(result.tool_calls).toBe(3);
378
+ });
379
+ it('should pass through duration unchanged', () => {
380
+ const metrics = {
381
+ models: {},
382
+ tools: {
383
+ totalCalls: 0,
384
+ totalSuccess: 0,
385
+ totalFail: 0,
386
+ totalDurationMs: 0,
387
+ totalDecisions: {
388
+ [ToolCallDecision.ACCEPT]: 0,
389
+ [ToolCallDecision.REJECT]: 0,
390
+ [ToolCallDecision.MODIFY]: 0,
391
+ [ToolCallDecision.AUTO_ACCEPT]: 0,
392
+ },
393
+ byName: {},
394
+ },
395
+ files: {
396
+ totalLinesAdded: 0,
397
+ totalLinesRemoved: 0,
398
+ },
399
+ };
400
+ const result = formatter.convertToStreamStats(metrics, 5000);
401
+ expect(result.duration_ms).toBe(5000);
402
+ });
403
+ });
404
+ describe('JSON validity', () => {
405
+ it('should produce valid JSON for all event types', () => {
406
+ const events = [
407
+ {
408
+ type: JsonStreamEventType.INIT,
409
+ timestamp: '2025-10-10T12:00:00.000Z',
410
+ session_id: 'test',
411
+ model: 'gemini-2.0-flash',
412
+ },
413
+ {
414
+ type: JsonStreamEventType.MESSAGE,
415
+ timestamp: '2025-10-10T12:00:00.000Z',
416
+ role: 'user',
417
+ content: 'Test',
418
+ },
419
+ {
420
+ type: JsonStreamEventType.TOOL_USE,
421
+ timestamp: '2025-10-10T12:00:00.000Z',
422
+ tool_name: 'Read',
423
+ tool_id: 'read-1',
424
+ parameters: {},
425
+ },
426
+ {
427
+ type: JsonStreamEventType.TOOL_RESULT,
428
+ timestamp: '2025-10-10T12:00:00.000Z',
429
+ tool_id: 'read-1',
430
+ status: 'success',
431
+ },
432
+ {
433
+ type: JsonStreamEventType.ERROR,
434
+ timestamp: '2025-10-10T12:00:00.000Z',
435
+ severity: 'error',
436
+ message: 'Test error',
437
+ },
438
+ {
439
+ type: JsonStreamEventType.RESULT,
440
+ timestamp: '2025-10-10T12:00:00.000Z',
441
+ status: 'success',
442
+ stats: {
443
+ total_tokens: 0,
444
+ input_tokens: 0,
445
+ output_tokens: 0,
446
+ duration_ms: 0,
447
+ tool_calls: 0,
448
+ },
449
+ },
450
+ ];
451
+ events.forEach((event) => {
452
+ const formatted = formatter.formatEvent(event);
453
+ expect(() => JSON.parse(formatted)).not.toThrow();
454
+ });
455
+ });
456
+ it('should preserve field types', () => {
457
+ const event = {
458
+ type: JsonStreamEventType.RESULT,
459
+ timestamp: '2025-10-10T12:00:00.000Z',
460
+ status: 'success',
461
+ stats: {
462
+ total_tokens: 100,
463
+ input_tokens: 50,
464
+ output_tokens: 50,
465
+ duration_ms: 1200,
466
+ tool_calls: 2,
467
+ },
468
+ };
469
+ const formatted = formatter.formatEvent(event);
470
+ const parsed = JSON.parse(formatted.trim());
471
+ expect(typeof parsed.stats.total_tokens).toBe('number');
472
+ expect(typeof parsed.stats.input_tokens).toBe('number');
473
+ expect(typeof parsed.stats.output_tokens).toBe('number');
474
+ expect(typeof parsed.stats.duration_ms).toBe('number');
475
+ expect(typeof parsed.stats.tool_calls).toBe('number');
476
+ });
477
+ });
478
+ });
479
+ //# sourceMappingURL=stream-json-formatter.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream-json-formatter.test.js","sourceRoot":"","sources":["../../../src/output/stream-json-formatter.test.ts"],"names":[],"mappings":"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"}
@@ -6,7 +6,8 @@
6
6
  import type { SessionMetrics } from '../telemetry/uiTelemetry.js';
7
7
  export declare enum OutputFormat {
8
8
  TEXT = "text",
9
- JSON = "json"
9
+ JSON = "json",
10
+ STREAM_JSON = "stream-json"
10
11
  }
11
12
  export interface JsonError {
12
13
  type: string;
@@ -18,3 +19,64 @@ export interface JsonOutput {
18
19
  stats?: SessionMetrics;
19
20
  error?: JsonError;
20
21
  }
22
+ export declare enum JsonStreamEventType {
23
+ INIT = "init",
24
+ MESSAGE = "message",
25
+ TOOL_USE = "tool_use",
26
+ TOOL_RESULT = "tool_result",
27
+ ERROR = "error",
28
+ RESULT = "result"
29
+ }
30
+ export interface BaseJsonStreamEvent {
31
+ type: JsonStreamEventType;
32
+ timestamp: string;
33
+ }
34
+ export interface InitEvent extends BaseJsonStreamEvent {
35
+ type: JsonStreamEventType.INIT;
36
+ session_id: string;
37
+ model: string;
38
+ }
39
+ export interface MessageEvent extends BaseJsonStreamEvent {
40
+ type: JsonStreamEventType.MESSAGE;
41
+ role: 'user' | 'assistant';
42
+ content: string;
43
+ delta?: boolean;
44
+ }
45
+ export interface ToolUseEvent extends BaseJsonStreamEvent {
46
+ type: JsonStreamEventType.TOOL_USE;
47
+ tool_name: string;
48
+ tool_id: string;
49
+ parameters: Record<string, unknown>;
50
+ }
51
+ export interface ToolResultEvent extends BaseJsonStreamEvent {
52
+ type: JsonStreamEventType.TOOL_RESULT;
53
+ tool_id: string;
54
+ status: 'success' | 'error';
55
+ output?: string;
56
+ error?: {
57
+ type: string;
58
+ message: string;
59
+ };
60
+ }
61
+ export interface ErrorEvent extends BaseJsonStreamEvent {
62
+ type: JsonStreamEventType.ERROR;
63
+ severity: 'warning' | 'error';
64
+ message: string;
65
+ }
66
+ export interface StreamStats {
67
+ total_tokens: number;
68
+ input_tokens: number;
69
+ output_tokens: number;
70
+ duration_ms: number;
71
+ tool_calls: number;
72
+ }
73
+ export interface ResultEvent extends BaseJsonStreamEvent {
74
+ type: JsonStreamEventType.RESULT;
75
+ status: 'success' | 'error';
76
+ error?: {
77
+ type: string;
78
+ message: string;
79
+ };
80
+ stats?: StreamStats;
81
+ }
82
+ export type JsonStreamEvent = InitEvent | MessageEvent | ToolUseEvent | ToolResultEvent | ErrorEvent | ResultEvent;
@@ -7,5 +7,16 @@ export var OutputFormat;
7
7
  (function (OutputFormat) {
8
8
  OutputFormat["TEXT"] = "text";
9
9
  OutputFormat["JSON"] = "json";
10
+ OutputFormat["STREAM_JSON"] = "stream-json";
10
11
  })(OutputFormat || (OutputFormat = {}));
12
+ // Streaming JSON event types
13
+ export var JsonStreamEventType;
14
+ (function (JsonStreamEventType) {
15
+ JsonStreamEventType["INIT"] = "init";
16
+ JsonStreamEventType["MESSAGE"] = "message";
17
+ JsonStreamEventType["TOOL_USE"] = "tool_use";
18
+ JsonStreamEventType["TOOL_RESULT"] = "tool_result";
19
+ JsonStreamEventType["ERROR"] = "error";
20
+ JsonStreamEventType["RESULT"] = "result";
21
+ })(JsonStreamEventType || (JsonStreamEventType = {}));
11
22
  //# sourceMappingURL=types.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/output/types.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAIH,MAAM,CAAN,IAAY,YAGX;AAHD,WAAY,YAAY;IACtB,6BAAa,CAAA;IACb,6BAAa,CAAA;AACf,CAAC,EAHW,YAAY,KAAZ,YAAY,QAGvB"}
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/output/types.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAIH,MAAM,CAAN,IAAY,YAIX;AAJD,WAAY,YAAY;IACtB,6BAAa,CAAA;IACb,6BAAa,CAAA;IACb,2CAA2B,CAAA;AAC7B,CAAC,EAJW,YAAY,KAAZ,YAAY,QAIvB;AAcD,6BAA6B;AAC7B,MAAM,CAAN,IAAY,mBAOX;AAPD,WAAY,mBAAmB;IAC7B,oCAAa,CAAA;IACb,0CAAmB,CAAA;IACnB,4CAAqB,CAAA;IACrB,kDAA2B,CAAA;IAC3B,sCAAe,CAAA;IACf,wCAAiB,CAAA;AACnB,CAAC,EAPW,mBAAmB,KAAnB,mBAAmB,QAO9B"}
@@ -9,6 +9,7 @@ import { spawn as cpSpawn } from 'node:child_process';
9
9
  import { TextDecoder } from 'node:util';
10
10
  import os from 'node:os';
11
11
  import { getCachedEncodingForBuffer } from '../utils/systemEncoding.js';
12
+ import { getShellConfiguration } from '../utils/shell-utils.js';
12
13
  import { isBinary } from '../utils/textUtils.js';
13
14
  import pkg from '@xterm/headless';
14
15
  import { serializeTerminalToObject, } from '../utils/terminalSerializer.js';
@@ -81,11 +82,13 @@ export class ShellExecutionService {
81
82
  static childProcessFallback(commandToExecute, cwd, onOutputEvent, abortSignal) {
82
83
  try {
83
84
  const isWindows = os.platform() === 'win32';
84
- const child = cpSpawn(commandToExecute, [], {
85
+ const { executable, argsPrefix } = getShellConfiguration();
86
+ const spawnArgs = [...argsPrefix, commandToExecute];
87
+ const child = cpSpawn(executable, spawnArgs, {
85
88
  cwd,
86
89
  stdio: ['ignore', 'pipe', 'pipe'],
87
- windowsVerbatimArguments: true,
88
- shell: isWindows ? true : 'bash',
90
+ windowsVerbatimArguments: isWindows ? false : undefined,
91
+ shell: false,
89
92
  detached: !isWindows,
90
93
  env: {
91
94
  ...process.env,
@@ -254,12 +257,9 @@ export class ShellExecutionService {
254
257
  try {
255
258
  const cols = shellExecutionConfig.terminalWidth ?? 80;
256
259
  const rows = shellExecutionConfig.terminalHeight ?? 30;
257
- const isWindows = os.platform() === 'win32';
258
- const shell = isWindows ? 'cmd.exe' : 'bash';
259
- const args = isWindows
260
- ? `/c ${commandToExecute}`
261
- : ['-c', commandToExecute];
262
- const ptyProcess = ptyInfo.module.spawn(shell, args, {
260
+ const { executable, argsPrefix } = getShellConfiguration();
261
+ const args = [...argsPrefix, commandToExecute];
262
+ const ptyProcess = ptyInfo.module.spawn(executable, args, {
263
263
  cwd,
264
264
  name: 'xterm',
265
265
  cols,