@pixelbyte-software/pixcode 1.30.1 → 1.31.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (205) hide show
  1. package/LICENSE +718 -718
  2. package/README.de.md +248 -248
  3. package/README.ja.md +240 -240
  4. package/README.ko.md +240 -240
  5. package/README.md +295 -285
  6. package/README.ru.md +248 -248
  7. package/README.tr.md +250 -250
  8. package/README.zh-CN.md +240 -240
  9. package/dist/api-docs.html +879 -879
  10. package/dist/assets/index-BRRJ47XQ.css +32 -0
  11. package/dist/assets/index-EQohwyiC.js +837 -0
  12. package/dist/clear-cache.html +85 -85
  13. package/dist/convert-icons.md +52 -52
  14. package/dist/favicon.png +0 -0
  15. package/dist/favicon.svg +7 -8
  16. package/dist/generate-icons.js +48 -48
  17. package/dist/icons/codex-white.svg +3 -3
  18. package/dist/icons/codex.svg +3 -3
  19. package/dist/icons/cursor-white.svg +11 -11
  20. package/dist/icons/icon-128x128.png +0 -0
  21. package/dist/icons/icon-128x128.svg +9 -12
  22. package/dist/icons/icon-144x144.png +0 -0
  23. package/dist/icons/icon-144x144.svg +9 -12
  24. package/dist/icons/icon-152x152.png +0 -0
  25. package/dist/icons/icon-152x152.svg +9 -12
  26. package/dist/icons/icon-192x192.png +0 -0
  27. package/dist/icons/icon-192x192.svg +9 -12
  28. package/dist/icons/icon-384x384.png +0 -0
  29. package/dist/icons/icon-384x384.svg +9 -12
  30. package/dist/icons/icon-512x512.png +0 -0
  31. package/dist/icons/icon-512x512.svg +9 -12
  32. package/dist/icons/icon-72x72.png +0 -0
  33. package/dist/icons/icon-72x72.svg +9 -12
  34. package/dist/icons/icon-96x96.png +0 -0
  35. package/dist/icons/icon-96x96.svg +9 -12
  36. package/dist/icons/icon-template.svg +9 -12
  37. package/dist/icons/qwen-ai-icon.png +0 -0
  38. package/dist/index.html +59 -49
  39. package/dist/logo.png +0 -0
  40. package/dist/logo.svg +11 -16
  41. package/dist/manifest.json +60 -60
  42. package/dist/sw.js +124 -124
  43. package/dist-server/server/cli.js +100 -97
  44. package/dist-server/server/cli.js.map +1 -1
  45. package/dist-server/server/daemon/manager.js +33 -33
  46. package/dist-server/server/daemon-manager.js +62 -62
  47. package/dist-server/server/database/db.js +114 -22
  48. package/dist-server/server/database/db.js.map +1 -1
  49. package/dist-server/server/database/schema.js +122 -89
  50. package/dist-server/server/database/schema.js.map +1 -1
  51. package/dist-server/server/gemini-cli.js +6 -1
  52. package/dist-server/server/gemini-cli.js.map +1 -1
  53. package/dist-server/server/index.js +234 -65
  54. package/dist-server/server/index.js.map +1 -1
  55. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +29 -2
  56. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -1
  57. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +22 -2
  58. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -1
  59. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +2 -2
  60. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -1
  61. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +14 -2
  62. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -1
  63. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js +132 -0
  64. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js.map +1 -0
  65. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js +87 -0
  66. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js.map +1 -0
  67. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js +201 -0
  68. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js.map +1 -0
  69. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js +19 -0
  70. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js.map +1 -0
  71. package/dist-server/server/modules/providers/provider.registry.js +2 -0
  72. package/dist-server/server/modules/providers/provider.registry.js.map +1 -1
  73. package/dist-server/server/modules/providers/provider.routes.js +310 -1
  74. package/dist-server/server/modules/providers/provider.routes.js.map +1 -1
  75. package/dist-server/server/projects.js +197 -6
  76. package/dist-server/server/projects.js.map +1 -1
  77. package/dist-server/server/qwen-code-cli.js +350 -0
  78. package/dist-server/server/qwen-code-cli.js.map +1 -0
  79. package/dist-server/server/qwen-response-handler.js +70 -0
  80. package/dist-server/server/qwen-response-handler.js.map +1 -0
  81. package/dist-server/server/routes/commands.js +25 -25
  82. package/dist-server/server/routes/git.js +17 -17
  83. package/dist-server/server/routes/network.js +116 -0
  84. package/dist-server/server/routes/network.js.map +1 -0
  85. package/dist-server/server/routes/projects.js +43 -0
  86. package/dist-server/server/routes/projects.js.map +1 -1
  87. package/dist-server/server/routes/qwen.js +23 -0
  88. package/dist-server/server/routes/qwen.js.map +1 -0
  89. package/dist-server/server/routes/taskmaster.js +419 -419
  90. package/dist-server/server/routes/telegram.js +119 -0
  91. package/dist-server/server/routes/telegram.js.map +1 -0
  92. package/dist-server/server/services/external-access.js +228 -0
  93. package/dist-server/server/services/external-access.js.map +1 -0
  94. package/dist-server/server/services/install-jobs.js +394 -0
  95. package/dist-server/server/services/install-jobs.js.map +1 -0
  96. package/dist-server/server/services/notification-orchestrator.js +19 -5
  97. package/dist-server/server/services/notification-orchestrator.js.map +1 -1
  98. package/dist-server/server/services/provider-credentials.js +154 -0
  99. package/dist-server/server/services/provider-credentials.js.map +1 -0
  100. package/dist-server/server/services/provider-models.js +218 -0
  101. package/dist-server/server/services/provider-models.js.map +1 -0
  102. package/dist-server/server/services/telegram/bot.js +259 -0
  103. package/dist-server/server/services/telegram/bot.js.map +1 -0
  104. package/dist-server/server/services/telegram/translations.js +160 -0
  105. package/dist-server/server/services/telegram/translations.js.map +1 -0
  106. package/dist-server/server/utils/port-access.js +196 -0
  107. package/dist-server/server/utils/port-access.js.map +1 -0
  108. package/dist-server/shared/modelConstants.js +18 -0
  109. package/dist-server/shared/modelConstants.js.map +1 -1
  110. package/package.json +177 -168
  111. package/scripts/fix-node-pty.js +67 -67
  112. package/server/claude-sdk.js +834 -834
  113. package/server/cli.js +940 -937
  114. package/server/constants/config.js +4 -4
  115. package/server/cursor-cli.js +342 -342
  116. package/server/daemon/manager.js +564 -564
  117. package/server/daemon-manager.js +920 -920
  118. package/server/database/db.js +696 -593
  119. package/server/database/schema.js +138 -102
  120. package/server/gemini-cli.js +475 -469
  121. package/server/gemini-response-handler.js +79 -79
  122. package/server/index.js +2730 -2557
  123. package/server/load-env.js +34 -34
  124. package/server/middleware/auth.js +132 -132
  125. package/server/modules/providers/list/claude/claude-auth.provider.ts +145 -123
  126. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  127. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  128. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  129. package/server/modules/providers/list/codex/codex-auth.provider.ts +115 -100
  130. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  131. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  132. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  133. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -143
  134. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  135. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  136. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  137. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +163 -151
  138. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  139. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  140. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  141. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +145 -0
  142. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -0
  143. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +218 -0
  144. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -0
  145. package/server/modules/providers/provider.registry.ts +38 -36
  146. package/server/modules/providers/provider.routes.ts +583 -217
  147. package/server/modules/providers/services/mcp.service.ts +94 -94
  148. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  149. package/server/modules/providers/services/sessions.service.ts +45 -45
  150. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  151. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  152. package/server/modules/providers/tests/mcp.test.ts +293 -293
  153. package/server/openai-codex.js +426 -426
  154. package/server/projects.js +2993 -2792
  155. package/server/qwen-code-cli.js +392 -0
  156. package/server/qwen-response-handler.js +73 -0
  157. package/server/routes/agent.js +1245 -1245
  158. package/server/routes/auth.js +134 -134
  159. package/server/routes/codex.js +19 -19
  160. package/server/routes/commands.js +554 -554
  161. package/server/routes/cursor.js +52 -52
  162. package/server/routes/gemini.js +24 -24
  163. package/server/routes/git.js +1488 -1488
  164. package/server/routes/mcp-utils.js +31 -31
  165. package/server/routes/messages.js +61 -61
  166. package/server/routes/network.js +128 -0
  167. package/server/routes/plugins.js +307 -307
  168. package/server/routes/projects.js +675 -627
  169. package/server/routes/qwen.js +27 -0
  170. package/server/routes/settings.js +286 -286
  171. package/server/routes/taskmaster.js +1471 -1471
  172. package/server/routes/telegram.js +125 -0
  173. package/server/routes/user.js +123 -123
  174. package/server/services/external-access.js +240 -0
  175. package/server/services/install-jobs.js +410 -0
  176. package/server/services/notification-orchestrator.js +242 -227
  177. package/server/services/provider-credentials.js +151 -0
  178. package/server/services/provider-models.js +225 -0
  179. package/server/services/telegram/bot.js +280 -0
  180. package/server/services/telegram/translations.js +170 -0
  181. package/server/services/vapid-keys.js +35 -35
  182. package/server/sessionManager.js +225 -225
  183. package/server/shared/interfaces.ts +54 -54
  184. package/server/shared/types.ts +172 -172
  185. package/server/shared/utils.ts +193 -193
  186. package/server/tsconfig.json +36 -36
  187. package/server/utils/colors.js +21 -21
  188. package/server/utils/commandParser.js +303 -303
  189. package/server/utils/frontmatter.js +18 -18
  190. package/server/utils/gitConfig.js +34 -34
  191. package/server/utils/mcp-detector.js +147 -147
  192. package/server/utils/plugin-loader.js +457 -457
  193. package/server/utils/plugin-process-manager.js +184 -184
  194. package/server/utils/port-access.js +209 -0
  195. package/server/utils/runtime-paths.js +37 -37
  196. package/server/utils/taskmaster-websocket.js +128 -128
  197. package/server/utils/url-detection.js +71 -71
  198. package/server/vite-daemon.js +78 -78
  199. package/shared/modelConstants.js +117 -97
  200. package/shared/networkHosts.js +22 -22
  201. package/dist/assets/index-C2c9QNwK.css +0 -32
  202. package/dist/assets/index-DyXDZED-.js +0 -1277
  203. package/dist-server/server/routes/cli-auth.js +0 -25
  204. package/dist-server/server/routes/cli-auth.js.map +0 -1
  205. package/server/routes/cli-auth.js +0 -27
@@ -1,469 +1,475 @@
1
- import { spawn } from 'child_process';
2
- import crossSpawn from 'cross-spawn';
3
-
4
- // Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
5
- const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
6
- import { promises as fs } from 'fs';
7
- import path from 'path';
8
- import os from 'os';
9
- import sessionManager from './sessionManager.js';
10
- import GeminiResponseHandler from './gemini-response-handler.js';
11
- import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
12
- import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
13
- import { createNormalizedMessage } from './shared/utils.js';
14
-
15
- let activeGeminiProcesses = new Map(); // Track active processes by session ID
16
-
17
- async function spawnGemini(command, options = {}, ws) {
18
- const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
19
- let capturedSessionId = sessionId; // Track session ID throughout the process
20
- let sessionCreatedSent = false; // Track if we've already sent session-created event
21
- let assistantBlocks = []; // Accumulate the full response blocks including tools
22
-
23
- // Use tools settings passed from frontend, or defaults
24
- const settings = toolsSettings || {
25
- allowedTools: [],
26
- disallowedTools: [],
27
- skipPermissions: false
28
- };
29
-
30
- // Build Gemini CLI command - start with print/resume flags first
31
- const args = [];
32
-
33
- // Add prompt flag with command if we have a command
34
- if (command && command.trim()) {
35
- args.push('--prompt', command);
36
- }
37
-
38
- // If we have a sessionId, we want to resume
39
- if (sessionId) {
40
- const session = sessionManager.getSession(sessionId);
41
- if (session && session.cliSessionId) {
42
- args.push('--resume', session.cliSessionId);
43
- }
44
- }
45
-
46
- // Use cwd (actual project directory) instead of projectPath (Gemini's metadata directory)
47
- // Clean the path by removing any non-printable characters
48
- const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
49
- const workingDir = cleanPath;
50
-
51
- // Handle images by saving them to temporary files and passing paths to Gemini
52
- const tempImagePaths = [];
53
- let tempDir = null;
54
- if (images && images.length > 0) {
55
- try {
56
- // Create temp directory in the project directory so Gemini can access it
57
- tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
58
- await fs.mkdir(tempDir, { recursive: true });
59
-
60
- // Save each image to a temp file
61
- for (const [index, image] of images.entries()) {
62
- // Extract base64 data and mime type
63
- const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
64
- if (!matches) {
65
- continue;
66
- }
67
-
68
- const [, mimeType, base64Data] = matches;
69
- const extension = mimeType.split('/')[1] || 'png';
70
- const filename = `image_${index}.${extension}`;
71
- const filepath = path.join(tempDir, filename);
72
-
73
- // Write base64 data to file
74
- await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
75
- tempImagePaths.push(filepath);
76
- }
77
-
78
- // Include the full image paths in the prompt for Gemini to reference
79
- // Gemini CLI can read images from file paths in the prompt
80
- if (tempImagePaths.length > 0 && command && command.trim()) {
81
- const imageNote = `\n\n[Images given: ${tempImagePaths.length} images are located at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
82
- const modifiedCommand = command + imageNote;
83
-
84
- // Update the command in args
85
- const promptIndex = args.indexOf('--prompt');
86
- if (promptIndex !== -1 && args[promptIndex + 1] === command) {
87
- args[promptIndex + 1] = modifiedCommand;
88
- } else if (promptIndex !== -1) {
89
- // If we're using context, update the full prompt
90
- args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
91
- }
92
- }
93
- } catch (error) {
94
- console.error('Error processing images for Gemini:', error);
95
- }
96
- }
97
-
98
- // Add basic flags for Gemini
99
- if (options.debug) {
100
- args.push('--debug');
101
- }
102
-
103
- // Add MCP config flag only if MCP servers are configured
104
- try {
105
- const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
106
- let hasMcpServers = false;
107
-
108
- try {
109
- await fs.access(geminiConfigPath);
110
- const geminiConfigRaw = await fs.readFile(geminiConfigPath, 'utf8');
111
- const geminiConfig = JSON.parse(geminiConfigRaw);
112
-
113
- // Check global MCP servers
114
- if (geminiConfig.mcpServers && Object.keys(geminiConfig.mcpServers).length > 0) {
115
- hasMcpServers = true;
116
- }
117
-
118
- // Check project-specific MCP servers
119
- if (!hasMcpServers && geminiConfig.geminiProjects) {
120
- const currentProjectPath = process.cwd();
121
- const projectConfig = geminiConfig.geminiProjects[currentProjectPath];
122
- if (projectConfig && projectConfig.mcpServers && Object.keys(projectConfig.mcpServers).length > 0) {
123
- hasMcpServers = true;
124
- }
125
- }
126
- } catch (e) {
127
- // Ignore if file doesn't exist or isn't parsable
128
- }
129
-
130
- if (hasMcpServers) {
131
- args.push('--mcp-config', geminiConfigPath);
132
- }
133
- } catch (error) {
134
- // Ignore outer errors
135
- }
136
-
137
- // Add model for all sessions (both new and resumed)
138
- let modelToUse = options.model || 'gemini-2.5-flash';
139
- args.push('--model', modelToUse);
140
- args.push('--output-format', 'stream-json');
141
-
142
- // Handle approval modes and allowed tools
143
- if (settings.skipPermissions || options.skipPermissions || permissionMode === 'yolo') {
144
- args.push('--yolo');
145
- } else if (permissionMode === 'auto_edit') {
146
- args.push('--approval-mode', 'auto_edit');
147
- } else if (permissionMode === 'plan') {
148
- args.push('--approval-mode', 'plan');
149
- }
150
-
151
- if (settings.allowedTools && settings.allowedTools.length > 0) {
152
- args.push('--allowed-tools', settings.allowedTools.join(','));
153
- }
154
-
155
- // Try to find gemini in PATH first, then fall back to environment variable
156
- const geminiPath = process.env.GEMINI_PATH || 'gemini';
157
- console.log('Spawning Gemini CLI:', geminiPath, args.join(' '));
158
- console.log('Working directory:', workingDir);
159
-
160
- let spawnCmd = geminiPath;
161
- let spawnArgs = args;
162
-
163
- // On non-Windows platforms, wrap the execution in a shell to avoid ENOEXEC
164
- // which happens when the target is a script lacking a shebang.
165
- if (os.platform() !== 'win32') {
166
- spawnCmd = 'sh';
167
- // Use exec to replace the shell process, ensuring signals hit gemini directly
168
- spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
169
- }
170
-
171
- return new Promise((resolve, reject) => {
172
- const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
173
- cwd: workingDir,
174
- stdio: ['pipe', 'pipe', 'pipe'],
175
- env: { ...process.env } // Inherit all environment variables
176
- });
177
- let terminalNotificationSent = false;
178
- let terminalFailureReason = null;
179
-
180
- const notifyTerminalState = ({ code = null, error = null } = {}) => {
181
- if (terminalNotificationSent) {
182
- return;
183
- }
184
-
185
- terminalNotificationSent = true;
186
-
187
- const finalSessionId = capturedSessionId || sessionId || processKey;
188
- if (code === 0 && !error) {
189
- notifyRunStopped({
190
- userId: ws?.userId || null,
191
- provider: 'gemini',
192
- sessionId: finalSessionId,
193
- sessionName: sessionSummary,
194
- stopReason: 'completed'
195
- });
196
- return;
197
- }
198
-
199
- notifyRunFailed({
200
- userId: ws?.userId || null,
201
- provider: 'gemini',
202
- sessionId: finalSessionId,
203
- sessionName: sessionSummary,
204
- error: error || terminalFailureReason || `Gemini CLI exited with code ${code}`
205
- });
206
- };
207
-
208
- // Attach temp file info to process for cleanup later
209
- geminiProcess.tempImagePaths = tempImagePaths;
210
- geminiProcess.tempDir = tempDir;
211
-
212
- // Store process reference for potential abort
213
- const processKey = capturedSessionId || sessionId || Date.now().toString();
214
- activeGeminiProcesses.set(processKey, geminiProcess);
215
-
216
- // Store sessionId on the process object for debugging
217
- geminiProcess.sessionId = processKey;
218
-
219
- // Close stdin to signal we're done sending input
220
- geminiProcess.stdin.end();
221
-
222
- // Add timeout handler
223
- const timeoutMs = 120000; // 120 seconds for slower models
224
- let timeout;
225
-
226
- const startTimeout = () => {
227
- if (timeout) clearTimeout(timeout);
228
- timeout = setTimeout(() => {
229
- const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
230
- terminalFailureReason = `Gemini CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
231
- ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
232
- try {
233
- geminiProcess.kill('SIGTERM');
234
- } catch (e) { }
235
- }, timeoutMs);
236
- };
237
-
238
- startTimeout();
239
-
240
- // Save user message to session when starting
241
- if (command && capturedSessionId) {
242
- sessionManager.addMessage(capturedSessionId, 'user', command);
243
- }
244
-
245
- // Create response handler for NDJSON buffering
246
- let responseHandler;
247
- if (ws) {
248
- responseHandler = new GeminiResponseHandler(ws, {
249
- onContentFragment: (content) => {
250
- if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
251
- assistantBlocks[assistantBlocks.length - 1].text += content;
252
- } else {
253
- assistantBlocks.push({ type: 'text', text: content });
254
- }
255
- },
256
- onToolUse: (event) => {
257
- assistantBlocks.push({
258
- type: 'tool_use',
259
- id: event.tool_id,
260
- name: event.tool_name,
261
- input: event.parameters
262
- });
263
- },
264
- onToolResult: (event) => {
265
- if (capturedSessionId) {
266
- if (assistantBlocks.length > 0) {
267
- sessionManager.addMessage(capturedSessionId, 'assistant', [...assistantBlocks]);
268
- assistantBlocks = [];
269
- }
270
- sessionManager.addMessage(capturedSessionId, 'user', [{
271
- type: 'tool_result',
272
- tool_use_id: event.tool_id,
273
- content: event.output === undefined ? null : event.output,
274
- is_error: event.status === 'error'
275
- }]);
276
- }
277
- },
278
- onInit: (event) => {
279
- if (capturedSessionId) {
280
- const sess = sessionManager.getSession(capturedSessionId);
281
- if (sess && !sess.cliSessionId) {
282
- sess.cliSessionId = event.session_id;
283
- sessionManager.saveSession(capturedSessionId);
284
- }
285
- }
286
- }
287
- });
288
- }
289
-
290
- // Handle stdout
291
- geminiProcess.stdout.on('data', (data) => {
292
- const rawOutput = data.toString();
293
- startTimeout(); // Re-arm the timeout
294
-
295
- // For new sessions, create a session ID FIRST
296
- if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
297
- capturedSessionId = `gemini_${Date.now()}`;
298
- sessionCreatedSent = true;
299
-
300
- // Create session in session manager
301
- sessionManager.createSession(capturedSessionId, cwd || process.cwd());
302
-
303
- // Save the user message now that we have a session ID
304
- if (command) {
305
- sessionManager.addMessage(capturedSessionId, 'user', command);
306
- }
307
-
308
- // Update process key with captured session ID
309
- if (processKey !== capturedSessionId) {
310
- activeGeminiProcesses.delete(processKey);
311
- activeGeminiProcesses.set(capturedSessionId, geminiProcess);
312
- }
313
-
314
- ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
315
-
316
- ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
317
- }
318
-
319
- if (responseHandler) {
320
- responseHandler.processData(rawOutput);
321
- } else if (rawOutput) {
322
- // Fallback to direct sending for raw CLI mode without WS
323
- if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
324
- assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
325
- } else {
326
- assistantBlocks.push({ type: 'text', text: rawOutput });
327
- }
328
- const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
329
- ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'gemini' }));
330
- }
331
- });
332
-
333
- // Handle stderr
334
- geminiProcess.stderr.on('data', (data) => {
335
- const errorMsg = data.toString();
336
-
337
- // Filter out deprecation warnings and "Loaded cached credentials" message
338
- if (errorMsg.includes('[DEP0040]') ||
339
- errorMsg.includes('DeprecationWarning') ||
340
- errorMsg.includes('--trace-deprecation') ||
341
- errorMsg.includes('Loaded cached credentials')) {
342
- return;
343
- }
344
-
345
- const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
346
- ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'gemini' }));
347
- });
348
-
349
- // Handle process completion
350
- geminiProcess.on('close', async (code) => {
351
- clearTimeout(timeout);
352
-
353
- // Flush any remaining buffered content
354
- if (responseHandler) {
355
- responseHandler.forceFlush();
356
- responseHandler.destroy();
357
- }
358
-
359
- // Clean up process reference
360
- const finalSessionId = capturedSessionId || sessionId || processKey;
361
- activeGeminiProcesses.delete(finalSessionId);
362
-
363
- // Save assistant response to session if we have one
364
- if (finalSessionId && assistantBlocks.length > 0) {
365
- sessionManager.addMessage(finalSessionId, 'assistant', assistantBlocks);
366
- }
367
-
368
- ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'gemini' }));
369
-
370
- // Clean up temporary image files if any
371
- if (geminiProcess.tempImagePaths && geminiProcess.tempImagePaths.length > 0) {
372
- for (const imagePath of geminiProcess.tempImagePaths) {
373
- await fs.unlink(imagePath).catch(err => { });
374
- }
375
- if (geminiProcess.tempDir) {
376
- await fs.rm(geminiProcess.tempDir, { recursive: true, force: true }).catch(err => { });
377
- }
378
- }
379
-
380
- if (code === 0) {
381
- notifyTerminalState({ code });
382
- resolve();
383
- } else {
384
- // code 127 = shell "command not found" — check installation
385
- if (code === 127) {
386
- const installed = await providerAuthService.isProviderInstalled('gemini');
387
- if (!installed) {
388
- const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
389
- ws.send(createNormalizedMessage({ kind: 'error', content: 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli', sessionId: socketSessionId, provider: 'gemini' }));
390
- }
391
- }
392
-
393
- notifyTerminalState({
394
- code,
395
- error: code === null ? 'Gemini CLI process was terminated or timed out' : null
396
- });
397
- reject(new Error(code === null ? 'Gemini CLI process was terminated or timed out' : `Gemini CLI exited with code ${code}`));
398
- }
399
- });
400
-
401
- // Handle process errors
402
- geminiProcess.on('error', async (error) => {
403
- // Clean up process reference on error
404
- const finalSessionId = capturedSessionId || sessionId || processKey;
405
- activeGeminiProcesses.delete(finalSessionId);
406
-
407
- // Check if Gemini CLI is installed for a clearer error message
408
- const installed = await providerAuthService.isProviderInstalled('gemini');
409
- const errorContent = !installed
410
- ? 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli'
411
- : error.message;
412
-
413
- const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
414
- ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'gemini' }));
415
- notifyTerminalState({ error });
416
-
417
- reject(error);
418
- });
419
-
420
- });
421
- }
422
-
423
- function abortGeminiSession(sessionId) {
424
- let geminiProc = activeGeminiProcesses.get(sessionId);
425
- let processKey = sessionId;
426
-
427
- if (!geminiProc) {
428
- for (const [key, proc] of activeGeminiProcesses.entries()) {
429
- if (proc.sessionId === sessionId) {
430
- geminiProc = proc;
431
- processKey = key;
432
- break;
433
- }
434
- }
435
- }
436
-
437
- if (geminiProc) {
438
- try {
439
- geminiProc.kill('SIGTERM');
440
- setTimeout(() => {
441
- if (activeGeminiProcesses.has(processKey)) {
442
- try {
443
- geminiProc.kill('SIGKILL');
444
- } catch (e) { }
445
- }
446
- }, 2000); // Wait 2 seconds before force kill
447
-
448
- return true;
449
- } catch (error) {
450
- return false;
451
- }
452
- }
453
- return false;
454
- }
455
-
456
- function isGeminiSessionActive(sessionId) {
457
- return activeGeminiProcesses.has(sessionId);
458
- }
459
-
460
- function getActiveGeminiSessions() {
461
- return Array.from(activeGeminiProcesses.keys());
462
- }
463
-
464
- export {
465
- spawnGemini,
466
- abortGeminiSession,
467
- isGeminiSessionActive,
468
- getActiveGeminiSessions
469
- };
1
+ import { spawn } from 'child_process';
2
+ import crossSpawn from 'cross-spawn';
3
+
4
+ // Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
5
+ const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
6
+ import { promises as fs } from 'fs';
7
+ import path from 'path';
8
+ import os from 'os';
9
+ import sessionManager from './sessionManager.js';
10
+ import GeminiResponseHandler from './gemini-response-handler.js';
11
+ import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
12
+ import { buildSpawnEnv } from './services/provider-credentials.js';
13
+ import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
14
+ import { createNormalizedMessage } from './shared/utils.js';
15
+
16
+ let activeGeminiProcesses = new Map(); // Track active processes by session ID
17
+
18
+ async function spawnGemini(command, options = {}, ws) {
19
+ const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
20
+ let capturedSessionId = sessionId; // Track session ID throughout the process
21
+ let sessionCreatedSent = false; // Track if we've already sent session-created event
22
+ let assistantBlocks = []; // Accumulate the full response blocks including tools
23
+
24
+ // Use tools settings passed from frontend, or defaults
25
+ const settings = toolsSettings || {
26
+ allowedTools: [],
27
+ disallowedTools: [],
28
+ skipPermissions: false
29
+ };
30
+
31
+ // Build Gemini CLI command - start with print/resume flags first
32
+ const args = [];
33
+
34
+ // Add prompt flag with command if we have a command
35
+ if (command && command.trim()) {
36
+ args.push('--prompt', command);
37
+ }
38
+
39
+ // If we have a sessionId, we want to resume
40
+ if (sessionId) {
41
+ const session = sessionManager.getSession(sessionId);
42
+ if (session && session.cliSessionId) {
43
+ args.push('--resume', session.cliSessionId);
44
+ }
45
+ }
46
+
47
+ // Use cwd (actual project directory) instead of projectPath (Gemini's metadata directory)
48
+ // Clean the path by removing any non-printable characters
49
+ const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
50
+ const workingDir = cleanPath;
51
+
52
+ // Handle images by saving them to temporary files and passing paths to Gemini
53
+ const tempImagePaths = [];
54
+ let tempDir = null;
55
+ if (images && images.length > 0) {
56
+ try {
57
+ // Create temp directory in the project directory so Gemini can access it
58
+ tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
59
+ await fs.mkdir(tempDir, { recursive: true });
60
+
61
+ // Save each image to a temp file
62
+ for (const [index, image] of images.entries()) {
63
+ // Extract base64 data and mime type
64
+ const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
65
+ if (!matches) {
66
+ continue;
67
+ }
68
+
69
+ const [, mimeType, base64Data] = matches;
70
+ const extension = mimeType.split('/')[1] || 'png';
71
+ const filename = `image_${index}.${extension}`;
72
+ const filepath = path.join(tempDir, filename);
73
+
74
+ // Write base64 data to file
75
+ await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
76
+ tempImagePaths.push(filepath);
77
+ }
78
+
79
+ // Include the full image paths in the prompt for Gemini to reference
80
+ // Gemini CLI can read images from file paths in the prompt
81
+ if (tempImagePaths.length > 0 && command && command.trim()) {
82
+ const imageNote = `\n\n[Images given: ${tempImagePaths.length} images are located at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
83
+ const modifiedCommand = command + imageNote;
84
+
85
+ // Update the command in args
86
+ const promptIndex = args.indexOf('--prompt');
87
+ if (promptIndex !== -1 && args[promptIndex + 1] === command) {
88
+ args[promptIndex + 1] = modifiedCommand;
89
+ } else if (promptIndex !== -1) {
90
+ // If we're using context, update the full prompt
91
+ args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
92
+ }
93
+ }
94
+ } catch (error) {
95
+ console.error('Error processing images for Gemini:', error);
96
+ }
97
+ }
98
+
99
+ // Add basic flags for Gemini
100
+ if (options.debug) {
101
+ args.push('--debug');
102
+ }
103
+
104
+ // Add MCP config flag only if MCP servers are configured
105
+ try {
106
+ const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
107
+ let hasMcpServers = false;
108
+
109
+ try {
110
+ await fs.access(geminiConfigPath);
111
+ const geminiConfigRaw = await fs.readFile(geminiConfigPath, 'utf8');
112
+ const geminiConfig = JSON.parse(geminiConfigRaw);
113
+
114
+ // Check global MCP servers
115
+ if (geminiConfig.mcpServers && Object.keys(geminiConfig.mcpServers).length > 0) {
116
+ hasMcpServers = true;
117
+ }
118
+
119
+ // Check project-specific MCP servers
120
+ if (!hasMcpServers && geminiConfig.geminiProjects) {
121
+ const currentProjectPath = process.cwd();
122
+ const projectConfig = geminiConfig.geminiProjects[currentProjectPath];
123
+ if (projectConfig && projectConfig.mcpServers && Object.keys(projectConfig.mcpServers).length > 0) {
124
+ hasMcpServers = true;
125
+ }
126
+ }
127
+ } catch (e) {
128
+ // Ignore if file doesn't exist or isn't parsable
129
+ }
130
+
131
+ if (hasMcpServers) {
132
+ args.push('--mcp-config', geminiConfigPath);
133
+ }
134
+ } catch (error) {
135
+ // Ignore outer errors
136
+ }
137
+
138
+ // Add model for all sessions (both new and resumed)
139
+ let modelToUse = options.model || 'gemini-2.5-flash';
140
+ args.push('--model', modelToUse);
141
+ args.push('--output-format', 'stream-json');
142
+
143
+ // Handle approval modes and allowed tools
144
+ if (settings.skipPermissions || options.skipPermissions || permissionMode === 'yolo') {
145
+ args.push('--yolo');
146
+ } else if (permissionMode === 'auto_edit') {
147
+ args.push('--approval-mode', 'auto_edit');
148
+ } else if (permissionMode === 'plan') {
149
+ args.push('--approval-mode', 'plan');
150
+ }
151
+
152
+ if (settings.allowedTools && settings.allowedTools.length > 0) {
153
+ args.push('--allowed-tools', settings.allowedTools.join(','));
154
+ }
155
+
156
+ // Try to find gemini in PATH first, then fall back to environment variable
157
+ const geminiPath = process.env.GEMINI_PATH || 'gemini';
158
+ console.log('Spawning Gemini CLI:', geminiPath, args.join(' '));
159
+ console.log('Working directory:', workingDir);
160
+
161
+ let spawnCmd = geminiPath;
162
+ let spawnArgs = args;
163
+
164
+ // On non-Windows platforms, wrap the execution in a shell to avoid ENOEXEC
165
+ // which happens when the target is a script lacking a shebang.
166
+ if (os.platform() !== 'win32') {
167
+ spawnCmd = 'sh';
168
+ // Use exec to replace the shell process, ensuring signals hit gemini directly
169
+ spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
170
+ }
171
+
172
+ // Pixcode UI-saved API key (stored in ~/.pixcode/provider-credentials.json)
173
+ // overlays on top of process.env so Gemini picks it up without the user
174
+ // exporting GEMINI_API_KEY in their shell.
175
+ const spawnEnv = await buildSpawnEnv('gemini');
176
+
177
+ return new Promise((resolve, reject) => {
178
+ const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
179
+ cwd: workingDir,
180
+ stdio: ['pipe', 'pipe', 'pipe'],
181
+ env: spawnEnv,
182
+ });
183
+ let terminalNotificationSent = false;
184
+ let terminalFailureReason = null;
185
+
186
+ const notifyTerminalState = ({ code = null, error = null } = {}) => {
187
+ if (terminalNotificationSent) {
188
+ return;
189
+ }
190
+
191
+ terminalNotificationSent = true;
192
+
193
+ const finalSessionId = capturedSessionId || sessionId || processKey;
194
+ if (code === 0 && !error) {
195
+ notifyRunStopped({
196
+ userId: ws?.userId || null,
197
+ provider: 'gemini',
198
+ sessionId: finalSessionId,
199
+ sessionName: sessionSummary,
200
+ stopReason: 'completed'
201
+ });
202
+ return;
203
+ }
204
+
205
+ notifyRunFailed({
206
+ userId: ws?.userId || null,
207
+ provider: 'gemini',
208
+ sessionId: finalSessionId,
209
+ sessionName: sessionSummary,
210
+ error: error || terminalFailureReason || `Gemini CLI exited with code ${code}`
211
+ });
212
+ };
213
+
214
+ // Attach temp file info to process for cleanup later
215
+ geminiProcess.tempImagePaths = tempImagePaths;
216
+ geminiProcess.tempDir = tempDir;
217
+
218
+ // Store process reference for potential abort
219
+ const processKey = capturedSessionId || sessionId || Date.now().toString();
220
+ activeGeminiProcesses.set(processKey, geminiProcess);
221
+
222
+ // Store sessionId on the process object for debugging
223
+ geminiProcess.sessionId = processKey;
224
+
225
+ // Close stdin to signal we're done sending input
226
+ geminiProcess.stdin.end();
227
+
228
+ // Add timeout handler
229
+ const timeoutMs = 120000; // 120 seconds for slower models
230
+ let timeout;
231
+
232
+ const startTimeout = () => {
233
+ if (timeout) clearTimeout(timeout);
234
+ timeout = setTimeout(() => {
235
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
236
+ terminalFailureReason = `Gemini CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
237
+ ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
238
+ try {
239
+ geminiProcess.kill('SIGTERM');
240
+ } catch (e) { }
241
+ }, timeoutMs);
242
+ };
243
+
244
+ startTimeout();
245
+
246
+ // Save user message to session when starting
247
+ if (command && capturedSessionId) {
248
+ sessionManager.addMessage(capturedSessionId, 'user', command);
249
+ }
250
+
251
+ // Create response handler for NDJSON buffering
252
+ let responseHandler;
253
+ if (ws) {
254
+ responseHandler = new GeminiResponseHandler(ws, {
255
+ onContentFragment: (content) => {
256
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
257
+ assistantBlocks[assistantBlocks.length - 1].text += content;
258
+ } else {
259
+ assistantBlocks.push({ type: 'text', text: content });
260
+ }
261
+ },
262
+ onToolUse: (event) => {
263
+ assistantBlocks.push({
264
+ type: 'tool_use',
265
+ id: event.tool_id,
266
+ name: event.tool_name,
267
+ input: event.parameters
268
+ });
269
+ },
270
+ onToolResult: (event) => {
271
+ if (capturedSessionId) {
272
+ if (assistantBlocks.length > 0) {
273
+ sessionManager.addMessage(capturedSessionId, 'assistant', [...assistantBlocks]);
274
+ assistantBlocks = [];
275
+ }
276
+ sessionManager.addMessage(capturedSessionId, 'user', [{
277
+ type: 'tool_result',
278
+ tool_use_id: event.tool_id,
279
+ content: event.output === undefined ? null : event.output,
280
+ is_error: event.status === 'error'
281
+ }]);
282
+ }
283
+ },
284
+ onInit: (event) => {
285
+ if (capturedSessionId) {
286
+ const sess = sessionManager.getSession(capturedSessionId);
287
+ if (sess && !sess.cliSessionId) {
288
+ sess.cliSessionId = event.session_id;
289
+ sessionManager.saveSession(capturedSessionId);
290
+ }
291
+ }
292
+ }
293
+ });
294
+ }
295
+
296
+ // Handle stdout
297
+ geminiProcess.stdout.on('data', (data) => {
298
+ const rawOutput = data.toString();
299
+ startTimeout(); // Re-arm the timeout
300
+
301
+ // For new sessions, create a session ID FIRST
302
+ if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
303
+ capturedSessionId = `gemini_${Date.now()}`;
304
+ sessionCreatedSent = true;
305
+
306
+ // Create session in session manager
307
+ sessionManager.createSession(capturedSessionId, cwd || process.cwd());
308
+
309
+ // Save the user message now that we have a session ID
310
+ if (command) {
311
+ sessionManager.addMessage(capturedSessionId, 'user', command);
312
+ }
313
+
314
+ // Update process key with captured session ID
315
+ if (processKey !== capturedSessionId) {
316
+ activeGeminiProcesses.delete(processKey);
317
+ activeGeminiProcesses.set(capturedSessionId, geminiProcess);
318
+ }
319
+
320
+ ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
321
+
322
+ ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
323
+ }
324
+
325
+ if (responseHandler) {
326
+ responseHandler.processData(rawOutput);
327
+ } else if (rawOutput) {
328
+ // Fallback to direct sending for raw CLI mode without WS
329
+ if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
330
+ assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
331
+ } else {
332
+ assistantBlocks.push({ type: 'text', text: rawOutput });
333
+ }
334
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
335
+ ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'gemini' }));
336
+ }
337
+ });
338
+
339
+ // Handle stderr
340
+ geminiProcess.stderr.on('data', (data) => {
341
+ const errorMsg = data.toString();
342
+
343
+ // Filter out deprecation warnings and "Loaded cached credentials" message
344
+ if (errorMsg.includes('[DEP0040]') ||
345
+ errorMsg.includes('DeprecationWarning') ||
346
+ errorMsg.includes('--trace-deprecation') ||
347
+ errorMsg.includes('Loaded cached credentials')) {
348
+ return;
349
+ }
350
+
351
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
352
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'gemini' }));
353
+ });
354
+
355
+ // Handle process completion
356
+ geminiProcess.on('close', async (code) => {
357
+ clearTimeout(timeout);
358
+
359
+ // Flush any remaining buffered content
360
+ if (responseHandler) {
361
+ responseHandler.forceFlush();
362
+ responseHandler.destroy();
363
+ }
364
+
365
+ // Clean up process reference
366
+ const finalSessionId = capturedSessionId || sessionId || processKey;
367
+ activeGeminiProcesses.delete(finalSessionId);
368
+
369
+ // Save assistant response to session if we have one
370
+ if (finalSessionId && assistantBlocks.length > 0) {
371
+ sessionManager.addMessage(finalSessionId, 'assistant', assistantBlocks);
372
+ }
373
+
374
+ ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'gemini' }));
375
+
376
+ // Clean up temporary image files if any
377
+ if (geminiProcess.tempImagePaths && geminiProcess.tempImagePaths.length > 0) {
378
+ for (const imagePath of geminiProcess.tempImagePaths) {
379
+ await fs.unlink(imagePath).catch(err => { });
380
+ }
381
+ if (geminiProcess.tempDir) {
382
+ await fs.rm(geminiProcess.tempDir, { recursive: true, force: true }).catch(err => { });
383
+ }
384
+ }
385
+
386
+ if (code === 0) {
387
+ notifyTerminalState({ code });
388
+ resolve();
389
+ } else {
390
+ // code 127 = shell "command not found" — check installation
391
+ if (code === 127) {
392
+ const installed = await providerAuthService.isProviderInstalled('gemini');
393
+ if (!installed) {
394
+ const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
395
+ ws.send(createNormalizedMessage({ kind: 'error', content: 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli', sessionId: socketSessionId, provider: 'gemini' }));
396
+ }
397
+ }
398
+
399
+ notifyTerminalState({
400
+ code,
401
+ error: code === null ? 'Gemini CLI process was terminated or timed out' : null
402
+ });
403
+ reject(new Error(code === null ? 'Gemini CLI process was terminated or timed out' : `Gemini CLI exited with code ${code}`));
404
+ }
405
+ });
406
+
407
+ // Handle process errors
408
+ geminiProcess.on('error', async (error) => {
409
+ // Clean up process reference on error
410
+ const finalSessionId = capturedSessionId || sessionId || processKey;
411
+ activeGeminiProcesses.delete(finalSessionId);
412
+
413
+ // Check if Gemini CLI is installed for a clearer error message
414
+ const installed = await providerAuthService.isProviderInstalled('gemini');
415
+ const errorContent = !installed
416
+ ? 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli'
417
+ : error.message;
418
+
419
+ const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
420
+ ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'gemini' }));
421
+ notifyTerminalState({ error });
422
+
423
+ reject(error);
424
+ });
425
+
426
+ });
427
+ }
428
+
429
+ function abortGeminiSession(sessionId) {
430
+ let geminiProc = activeGeminiProcesses.get(sessionId);
431
+ let processKey = sessionId;
432
+
433
+ if (!geminiProc) {
434
+ for (const [key, proc] of activeGeminiProcesses.entries()) {
435
+ if (proc.sessionId === sessionId) {
436
+ geminiProc = proc;
437
+ processKey = key;
438
+ break;
439
+ }
440
+ }
441
+ }
442
+
443
+ if (geminiProc) {
444
+ try {
445
+ geminiProc.kill('SIGTERM');
446
+ setTimeout(() => {
447
+ if (activeGeminiProcesses.has(processKey)) {
448
+ try {
449
+ geminiProc.kill('SIGKILL');
450
+ } catch (e) { }
451
+ }
452
+ }, 2000); // Wait 2 seconds before force kill
453
+
454
+ return true;
455
+ } catch (error) {
456
+ return false;
457
+ }
458
+ }
459
+ return false;
460
+ }
461
+
462
+ function isGeminiSessionActive(sessionId) {
463
+ return activeGeminiProcesses.has(sessionId);
464
+ }
465
+
466
+ function getActiveGeminiSessions() {
467
+ return Array.from(activeGeminiProcesses.keys());
468
+ }
469
+
470
+ export {
471
+ spawnGemini,
472
+ abortGeminiSession,
473
+ isGeminiSessionActive,
474
+ getActiveGeminiSessions
475
+ };