@pixelbyte-software/pixcode 1.51.2 → 1.51.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (320) hide show
  1. package/CODE_OF_CONDUCT.md +41 -41
  2. package/CONTRIBUTING.md +155 -155
  3. package/LICENSE +718 -718
  4. package/README.de.md +169 -169
  5. package/README.ja.md +167 -167
  6. package/README.ko.md +167 -167
  7. package/README.md +419 -419
  8. package/README.ru.md +169 -169
  9. package/README.tr.md +298 -298
  10. package/README.zh-CN.md +167 -167
  11. package/SECURITY.md +46 -46
  12. package/dist/api-automation.html +110 -110
  13. package/dist/api-docs.html +548 -548
  14. package/dist/assets/{index-EN9ngyxf.js → index-17CwxHSZ.js} +185 -185
  15. package/dist/assets/index-B9N-gfOQ.css +32 -0
  16. package/dist/clear-cache.html +85 -85
  17. package/dist/convert-icons.md +52 -52
  18. package/dist/docs.html +308 -308
  19. package/dist/favicon.svg +8 -8
  20. package/dist/features.html +133 -133
  21. package/dist/generate-icons.js +48 -48
  22. package/dist/humans.txt +15 -15
  23. package/dist/icons/codex-white.svg +3 -3
  24. package/dist/icons/codex.svg +3 -3
  25. package/dist/icons/cursor-white.svg +11 -11
  26. package/dist/icons/icon-128x128.svg +9 -9
  27. package/dist/icons/icon-144x144.svg +9 -9
  28. package/dist/icons/icon-152x152.svg +9 -9
  29. package/dist/icons/icon-192x192.svg +9 -9
  30. package/dist/icons/icon-384x384.svg +9 -9
  31. package/dist/icons/icon-512x512.svg +9 -9
  32. package/dist/icons/icon-72x72.svg +9 -9
  33. package/dist/icons/icon-96x96.svg +9 -9
  34. package/dist/icons/icon-template.svg +9 -9
  35. package/dist/icons/qwen-logo.svg +14 -14
  36. package/dist/index.html +59 -59
  37. package/dist/landing.html +268 -268
  38. package/dist/llms-full.txt +119 -119
  39. package/dist/llms.txt +53 -53
  40. package/dist/logo.svg +12 -12
  41. package/dist/manifest.json +60 -60
  42. package/dist/openapi.yaml +1696 -1696
  43. package/dist/orchestration.html +125 -125
  44. package/dist/robots.txt +4 -4
  45. package/dist/site.css +692 -692
  46. package/dist/sitemap.xml +51 -51
  47. package/dist/sw.js +132 -132
  48. package/dist-server/server/cli.js +96 -96
  49. package/dist-server/server/daemon/manager.js +33 -33
  50. package/dist-server/server/daemon-manager.js +64 -64
  51. package/dist-server/server/index.js +122 -3
  52. package/dist-server/server/index.js.map +1 -1
  53. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
  54. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
  55. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
  56. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
  57. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
  58. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
  59. package/dist-server/server/modules/orchestration/index.js +1 -0
  60. package/dist-server/server/modules/orchestration/index.js.map +1 -1
  61. package/dist-server/server/routes/commands.js +25 -25
  62. package/dist-server/server/routes/git.js +17 -17
  63. package/dist-server/server/routes/live-view.js +46 -46
  64. package/dist-server/server/services/hermes-gateway.js +310 -0
  65. package/dist-server/server/services/hermes-gateway.js.map +1 -1
  66. package/dist-server/server/services/public-api-manifest.js +59 -51
  67. package/dist-server/server/services/public-api-manifest.js.map +1 -1
  68. package/package.json +222 -222
  69. package/scripts/fix-node-pty.js +67 -67
  70. package/scripts/github/create-v1.38-issues.mjs +351 -351
  71. package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
  72. package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
  73. package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
  74. package/scripts/smoke/changes-panel-layout.mjs +48 -48
  75. package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
  76. package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
  77. package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
  78. package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
  79. package/scripts/smoke/chat-session-state.mjs +19 -19
  80. package/scripts/smoke/code-editor-theme.mjs +55 -55
  81. package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
  82. package/scripts/smoke/command-center-agent-writes.mjs +79 -79
  83. package/scripts/smoke/command-center-non-git.mjs +46 -46
  84. package/scripts/smoke/context-packet.mjs +43 -43
  85. package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
  86. package/scripts/smoke/daemon-entrypoint.mjs +20 -20
  87. package/scripts/smoke/default-landing-routing.mjs +33 -33
  88. package/scripts/smoke/desktop-native-notifications.mjs +30 -30
  89. package/scripts/smoke/desktop-tray-icon.mjs +33 -33
  90. package/scripts/smoke/discord-release-workflow.mjs +24 -24
  91. package/scripts/smoke/git-install-update.mjs +255 -255
  92. package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
  93. package/scripts/smoke/hermes-api-install.mjs +56 -56
  94. package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
  95. package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
  96. package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
  97. package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
  98. package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
  99. package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
  100. package/scripts/smoke/hermes-rest-live.mjs +42 -42
  101. package/scripts/smoke/hermes-roundtrip.mjs +167 -167
  102. package/scripts/smoke/hermes-settings-commands.mjs +349 -346
  103. package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
  104. package/scripts/smoke/live-view-diagnostics.mjs +53 -53
  105. package/scripts/smoke/live-view-environment.mjs +92 -92
  106. package/scripts/smoke/live-view-integration.mjs +450 -450
  107. package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
  108. package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
  109. package/scripts/smoke/model-registry.mjs +36 -36
  110. package/scripts/smoke/multi-project-ui.mjs +45 -45
  111. package/scripts/smoke/multi-worker-slots.mjs +42 -42
  112. package/scripts/smoke/notification-center.mjs +87 -87
  113. package/scripts/smoke/notification-inapp-preference.mjs +23 -23
  114. package/scripts/smoke/notification-taxonomy.mjs +58 -58
  115. package/scripts/smoke/orchestration-api.mjs +172 -172
  116. package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
  117. package/scripts/smoke/orchestration-live-run.mjs +176 -176
  118. package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
  119. package/scripts/smoke/orchestration-model-sync.mjs +30 -30
  120. package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
  121. package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
  122. package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
  123. package/scripts/smoke/permission-policy.mjs +50 -50
  124. package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -167
  125. package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
  126. package/scripts/smoke/provider-rest-api.mjs +124 -124
  127. package/scripts/smoke/provider-selection-status.mjs +52 -52
  128. package/scripts/smoke/run-state-refresh.mjs +52 -52
  129. package/scripts/smoke/runtime-manager.mjs +99 -99
  130. package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
  131. package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
  132. package/scripts/smoke/static-root-routing.mjs +21 -21
  133. package/scripts/smoke/strict-handoff-compact.mjs +60 -60
  134. package/scripts/smoke/taskmaster-config.mjs +24 -24
  135. package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
  136. package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
  137. package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
  138. package/scripts/smoke/telegram-control.mjs +242 -242
  139. package/scripts/smoke/tunnel-persistence.mjs +56 -56
  140. package/scripts/smoke/update-issue-progress.mjs +69 -69
  141. package/scripts/smoke/update-ux.mjs +55 -55
  142. package/scripts/smoke/v138-completion.mjs +132 -132
  143. package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
  144. package/scripts/smoke/v138-diagnostics.mjs +63 -63
  145. package/scripts/smoke/v138-issue-planner.mjs +33 -33
  146. package/scripts/smoke/v143-remote-control.mjs +76 -76
  147. package/scripts/smoke/v144-production-loop.mjs +47 -47
  148. package/scripts/smoke/v145-platformization.mjs +46 -46
  149. package/scripts/smoke/v146-control-room-ui.mjs +150 -150
  150. package/scripts/smoke/version-modal-autoshow.mjs +29 -29
  151. package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
  152. package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
  153. package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
  154. package/scripts/smoke/workflow-templates.mjs +43 -43
  155. package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
  156. package/scripts/update-git-install.mjs +293 -293
  157. package/server/claude-sdk.js +920 -920
  158. package/server/cli.js +1039 -1039
  159. package/server/constants/config.js +4 -4
  160. package/server/cursor-cli.js +344 -344
  161. package/server/daemon/manager.js +563 -563
  162. package/server/daemon-manager.js +964 -964
  163. package/server/database/db.js +921 -921
  164. package/server/database/json-store.js +197 -197
  165. package/server/gemini-cli.js +550 -550
  166. package/server/gemini-response-handler.js +79 -79
  167. package/server/index.js +128 -2
  168. package/server/load-env.js +35 -35
  169. package/server/middleware/auth.js +175 -175
  170. package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
  171. package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
  172. package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
  173. package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
  174. package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
  175. package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
  176. package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
  177. package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
  178. package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
  179. package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
  180. package/server/modules/orchestration/a2a/agent-card.ts +55 -55
  181. package/server/modules/orchestration/a2a/routes.ts +590 -590
  182. package/server/modules/orchestration/a2a/task-store.ts +178 -178
  183. package/server/modules/orchestration/a2a/types.ts +126 -126
  184. package/server/modules/orchestration/a2a/validator.ts +113 -113
  185. package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
  186. package/server/modules/orchestration/index.ts +101 -100
  187. package/server/modules/orchestration/preview/port-watcher.ts +112 -112
  188. package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
  189. package/server/modules/orchestration/preview/types.ts +19 -19
  190. package/server/modules/orchestration/security/permission-policy.ts +401 -401
  191. package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
  192. package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
  193. package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
  194. package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
  195. package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
  196. package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
  197. package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
  198. package/server/modules/orchestration/workflows/context-packet.ts +186 -186
  199. package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
  200. package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
  201. package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
  202. package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
  203. package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
  204. package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
  205. package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
  206. package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
  207. package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
  208. package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
  209. package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
  210. package/server/modules/orchestration/workspace/path-safety.ts +55 -55
  211. package/server/modules/orchestration/workspace/types.ts +52 -52
  212. package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
  213. package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
  214. package/server/modules/providers/index.ts +2 -2
  215. package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
  216. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  217. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  218. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  219. package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
  220. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  221. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  222. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  223. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
  224. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  225. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  226. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  227. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
  228. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  229. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  230. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  231. package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
  232. package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
  233. package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
  234. package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
  235. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
  236. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
  237. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
  238. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
  239. package/server/modules/providers/provider.registry.ts +40 -40
  240. package/server/modules/providers/provider.routes.ts +944 -944
  241. package/server/modules/providers/services/mcp.service.ts +86 -86
  242. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  243. package/server/modules/providers/services/sessions.service.ts +45 -45
  244. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  245. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  246. package/server/modules/providers/shared/provider-configs.ts +142 -142
  247. package/server/modules/providers/tests/mcp.test.ts +293 -293
  248. package/server/openai-codex.js +462 -462
  249. package/server/opencode-cli.js +491 -491
  250. package/server/opencode-response-handler.js +111 -111
  251. package/server/projects.js +3008 -3008
  252. package/server/qwen-code-cli.js +410 -410
  253. package/server/qwen-response-handler.js +73 -73
  254. package/server/routes/agent.js +1435 -1435
  255. package/server/routes/auth.js +159 -159
  256. package/server/routes/codex.js +20 -20
  257. package/server/routes/commands.js +570 -570
  258. package/server/routes/cursor.js +61 -61
  259. package/server/routes/diagnostics.js +41 -41
  260. package/server/routes/gemini.js +25 -25
  261. package/server/routes/git.js +1650 -1650
  262. package/server/routes/live-view.js +411 -411
  263. package/server/routes/mcp-utils.js +13 -13
  264. package/server/routes/messages.js +62 -62
  265. package/server/routes/network.js +125 -125
  266. package/server/routes/platformization.js +212 -212
  267. package/server/routes/plugins.js +320 -320
  268. package/server/routes/production-agent-loop.js +90 -90
  269. package/server/routes/projects.js +917 -917
  270. package/server/routes/public-api.js +34 -34
  271. package/server/routes/qwen.js +27 -27
  272. package/server/routes/remote.js +55 -55
  273. package/server/routes/settings.js +321 -321
  274. package/server/routes/telegram.js +140 -140
  275. package/server/routes/user.js +125 -125
  276. package/server/routes/webhooks.js +63 -63
  277. package/server/services/control-room.js +102 -102
  278. package/server/services/diagnostics.js +165 -165
  279. package/server/services/external-access.js +375 -375
  280. package/server/services/hermes-gateway.js +1562 -1247
  281. package/server/services/hermes-install-jobs.js +729 -729
  282. package/server/services/install-jobs.js +715 -715
  283. package/server/services/live-view.js +956 -956
  284. package/server/services/managed-runtimes.js +493 -493
  285. package/server/services/model-registry.js +144 -144
  286. package/server/services/notification-orchestrator.js +365 -365
  287. package/server/services/notification-taxonomy.js +204 -204
  288. package/server/services/platformization.js +815 -815
  289. package/server/services/production-agent-loop.js +248 -248
  290. package/server/services/provider-cli-versions.js +149 -149
  291. package/server/services/provider-credentials.js +189 -189
  292. package/server/services/provider-models.js +396 -396
  293. package/server/services/public-api-manifest.js +190 -182
  294. package/server/services/remote-connection.js +127 -127
  295. package/server/services/runtime-manager.js +323 -323
  296. package/server/services/startup-update.js +234 -234
  297. package/server/services/telegram/bot.js +331 -331
  298. package/server/services/telegram/control-center.js +979 -979
  299. package/server/services/telegram/telegram-http-client.js +151 -151
  300. package/server/services/telegram/translations.js +340 -340
  301. package/server/services/vapid-keys.js +36 -36
  302. package/server/services/webhooks.js +216 -216
  303. package/server/sessionManager.js +225 -225
  304. package/server/shared/interfaces.ts +54 -54
  305. package/server/shared/types.ts +172 -172
  306. package/server/shared/utils.ts +193 -193
  307. package/server/tsconfig.json +36 -36
  308. package/server/utils/colors.js +21 -21
  309. package/server/utils/commandParser.js +305 -305
  310. package/server/utils/frontmatter.js +18 -18
  311. package/server/utils/gitConfig.js +34 -34
  312. package/server/utils/plugin-loader.js +457 -457
  313. package/server/utils/plugin-process-manager.js +185 -185
  314. package/server/utils/port-access.js +209 -209
  315. package/server/utils/runtime-paths.js +37 -37
  316. package/server/utils/url-detection.js +71 -71
  317. package/server/vite-daemon.js +79 -79
  318. package/shared/modelConstants.js +161 -161
  319. package/shared/networkHosts.js +22 -22
  320. package/dist/assets/index-DMz0zv6T.css +0 -32
@@ -1,715 +1,715 @@
1
- /**
2
- * In-memory install-job registry + sandboxed local CLI installer.
3
- *
4
- * Why not `npm install -g`:
5
- * - Requires admin/sudo on Windows and most Linux distros. When Pixcode
6
- * runs as a non-privileged daemon, -g fails with EACCES and the user
7
- * sees a blank log with no actionable error.
8
- * - Even on Windows desktop, npm's global prefix is sometimes broken
9
- * (AppData permissions, antivirus quarantining node_modules/.bin).
10
- * - CI/docker/VPS setups often don't have `npm` on the daemon's PATH at
11
- * all, even when the user's interactive shell does.
12
- *
13
- * What we do instead:
14
- * - Install targets go into `~/.pixcode/cli-bin/` as LOCAL dependencies
15
- * of a pixcode-owned package.json (no -g, no sudo, no UAC).
16
- * - Resolve `npm` from the same Node install that's running the server
17
- * (sibling file to `process.execPath`) so PATH environment doesn't
18
- * matter.
19
- * - On server boot, `~/.pixcode/cli-bin/node_modules/.bin` is prepended
20
- * to `process.env.PATH`. Every existing `cross-spawn(binary)` call
21
- * (in claude-auth, gemini-cli, qwen-code-cli, etc.) then resolves to
22
- * the locally installed binary without any change to the adapter code.
23
- *
24
- * The HTTP/stream side of the API is the same as before:
25
- * - POST /install → spawns the child, returns { jobId }
26
- * - GET /install/:jobId/stream → EventSource that replays the buffered
27
- * transcript and then streams live chunks.
28
- * - DELETE /install/:jobId → cancels an in-flight install.
29
- *
30
- * Jobs linger 10 minutes after completion so late subscribers still see
31
- * the outcome.
32
- */
33
- import { execFileSync } from 'node:child_process';
34
- import { EventEmitter } from 'node:events';
35
- import { randomUUID } from 'node:crypto';
36
- import fs from 'node:fs';
37
- import os from 'node:os';
38
- import path from 'node:path';
39
-
40
- // Use cross-spawn instead of node:child_process.spawn. On Windows, node's
41
- // spawn cannot invoke `.cmd` / `.bat` files without `shell: true`, and with
42
- // `shell: true` it tokenises on spaces — so a valid npm path like
43
- // `C:\Program Files\nodejs\npm.cmd` gets split into "C:\Program" + "Files...".
44
- // cross-spawn shells out through cmd.exe with proper quoting transparently
45
- // and is already a transitive dependency we can safely re-use.
46
- import spawn from 'cross-spawn';
47
-
48
- const jobs = new Map();
49
- const FINISHED_TTL_MS = 10 * 60 * 1000;
50
- const HARD_TIMEOUT_MS = 10 * 60 * 1000;
51
- const USER_SHELL_PATH_CACHE_TTL_MS = 5 * 60 * 1000;
52
- const userShellPathCache = {
53
- value: null,
54
- readAt: 0,
55
- };
56
-
57
- export const CLI_HOME = path.join(os.homedir(), '.pixcode', 'cli-bin');
58
- export const CLI_BIN_DIR = path.join(CLI_HOME, 'node_modules', '.bin');
59
-
60
- /**
61
- * npm package → the binary name it installs. Used to verify the install
62
- * actually dropped an executable we can run, since npm can exit(0) even
63
- * when a package has no `bin` entry or our PATH wiring is wrong.
64
- */
65
- const PACKAGE_BINARIES = {
66
- '@anthropic-ai/claude-code': 'claude',
67
- '@openai/codex': 'codex',
68
- '@google/gemini-cli': 'gemini',
69
- '@qwen-code/qwen-code': 'qwen',
70
- 'opencode-ai': 'opencode',
71
- 'task-master': 'task-master',
72
- };
73
-
74
- /**
75
- * Make sure `CLI_HOME` exists with a minimal package.json so `npm install`
76
- * doesn't walk up to some unrelated parent and pollute it.
77
- */
78
- function ensureCliHome() {
79
- fs.mkdirSync(CLI_HOME, { recursive: true });
80
- const pkgPath = path.join(CLI_HOME, 'package.json');
81
- if (!fs.existsSync(pkgPath)) {
82
- const pkg = {
83
- name: 'pixcode-cli-bin',
84
- private: true,
85
- version: '0.0.0',
86
- description:
87
- 'Pixcode-managed sandbox for provider CLIs (claude/codex/gemini/qwen). '
88
- + 'Safe to delete; Pixcode will re-create it on next install.',
89
- };
90
- fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2));
91
- }
92
- }
93
-
94
- /**
95
- * Prepend the pixcode-managed bin dir to PATH. Called at server boot so
96
- * every subsequent `spawn('claude'|'gemini'|'codex'|'qwen', …)` in the
97
- * provider adapters (which use cross-spawn with bare names) resolves to
98
- * the locally installed binary without any per-adapter change.
99
- */
100
- export function primeCliBinPath(env = process.env) {
101
- ensureCliHome();
102
- const augmentedEnv = buildCliSpawnEnv(env);
103
- env.PATH = augmentedEnv.PATH;
104
- if ('Path' in env || augmentedEnv.Path) env.Path = augmentedEnv.Path || augmentedEnv.PATH;
105
- // Once PATH is ready, resolve any well-known provider binaries to absolute
106
- // paths and export them as *_CLI_PATH env vars. This side-steps a Windows
107
- // gotcha: `child_process.spawn('claude', …)` does NOT auto-resolve .cmd /
108
- // .bat extensions, and the Claude Agent SDK calls spawn directly instead
109
- // of via cross-spawn — so a bare "claude" on PATH works in a shell but
110
- // fails inside the SDK. Pinning the full path side-steps it entirely.
111
- resolveProviderExecutables(env);
112
- }
113
-
114
- /**
115
- * Scan PATH (plus known native-installer locations) for every provider
116
- * binary we ship support for, and export *_CLI_PATH env vars pointing to
117
- * the absolute executable. Existing vars are left alone so users can
118
- * override detection.
119
- */
120
- export function resolveProviderExecutables(env = process.env) {
121
- // Claude is intentionally omitted. The Claude Agent SDK ships a bundled
122
- // native binary per platform (@anthropic-ai/claude-agent-sdk-<os>-<arch>)
123
- // and resolves it automatically. Exporting CLAUDE_CLI_PATH here would
124
- // override that and hand a `.cmd` shim to Node's spawn on Windows,
125
- // which then throws EINVAL (spawn can't exec .cmd files directly).
126
- //
127
- // The other providers use cross-spawn in our own adapters, which
128
- // handles .cmd/.bat resolution on Windows. Forcing an absolute path
129
- // there is still helpful because cross-spawn.sync without quoting
130
- // can hit edge cases when PATH contains spaces.
131
- const providers = [
132
- { name: 'codex', envKey: 'CODEX_CLI_PATH' },
133
- { name: 'gemini', envKey: 'GEMINI_CLI_PATH' },
134
- { name: 'qwen', envKey: 'QWEN_CLI_PATH' },
135
- { name: 'opencode', envKey: 'OPENCODE_CLI_PATH' },
136
- { name: 'cursor-agent', envKey: 'CURSOR_CLI_PATH' },
137
- ];
138
- for (const { name, envKey } of providers) {
139
- if (env[envKey]) continue;
140
- const resolved = findExecutableOnPath(name, env);
141
- if (resolved) env[envKey] = resolved;
142
- }
143
- }
144
-
145
- function pathSeparator() {
146
- return process.platform === 'win32' ? ';' : ':';
147
- }
148
-
149
- function splitPathList(value) {
150
- return String(value || '').split(pathSeparator()).map((entry) => entry.trim()).filter(Boolean);
151
- }
152
-
153
- function collectNvmNodeBins(home) {
154
- const versionsDir = path.join(home, '.nvm', 'versions', 'node');
155
- try {
156
- return fs.readdirSync(versionsDir)
157
- .map((version) => path.join(versionsDir, version, 'bin'))
158
- .filter((candidate) => {
159
- try {
160
- return fs.statSync(candidate).isDirectory();
161
- } catch {
162
- return false;
163
- }
164
- })
165
- .sort()
166
- .reverse();
167
- } catch {
168
- return [];
169
- }
170
- }
171
-
172
- function collectKnownUserBinDirs(env = process.env) {
173
- const home = os.homedir();
174
- if (process.platform === 'win32') {
175
- return [
176
- path.join(env.APPDATA || path.join(home, 'AppData', 'Roaming'), 'npm'),
177
- path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'Programs', 'nodejs'),
178
- ];
179
- }
180
-
181
- return [
182
- CLI_BIN_DIR,
183
- path.dirname(process.execPath),
184
- ...collectNvmNodeBins(home),
185
- path.join(home, '.volta', 'bin'),
186
- path.join(home, '.asdf', 'shims'),
187
- path.join(home, '.bun', 'bin'),
188
- path.join(home, '.local', 'bin'),
189
- path.join(home, '.npm-global', 'bin'),
190
- '/opt/homebrew/bin',
191
- '/opt/homebrew/sbin',
192
- '/usr/local/bin',
193
- '/usr/local/sbin',
194
- '/usr/bin',
195
- '/bin',
196
- ];
197
- }
198
-
199
- export function collectUserShellPath(env = process.env) {
200
- if (process.platform === 'win32') return [];
201
-
202
- const now = Date.now();
203
- if (userShellPathCache.value && now - userShellPathCache.readAt < USER_SHELL_PATH_CACHE_TTL_MS) {
204
- return userShellPathCache.value;
205
- }
206
-
207
- const shells = [env.SHELL, '/bin/zsh', '/bin/bash']
208
- .filter(Boolean)
209
- .filter((candidate, index, list) => list.indexOf(candidate) === index)
210
- .filter((candidate) => {
211
- try {
212
- return fs.existsSync(candidate);
213
- } catch {
214
- return false;
215
- }
216
- });
217
-
218
- const marker = '__PIXCODE_LOGIN_PATH__=';
219
- for (const shell of shells) {
220
- try {
221
- const output = execFileSync(shell, ['-lc', `printf '\\n${marker}%s\\n' "$PATH"`], {
222
- encoding: 'utf8',
223
- env,
224
- timeout: 2500,
225
- stdio: ['ignore', 'pipe', 'ignore'],
226
- });
227
- const line = output.split(/\r?\n/).reverse().find((part) => part.startsWith(marker));
228
- const shellPath = line?.slice(marker.length);
229
- if (shellPath) {
230
- const entries = splitPathList(shellPath);
231
- userShellPathCache.value = entries;
232
- userShellPathCache.readAt = now;
233
- return entries;
234
- }
235
- } catch {
236
- // GUI-launched macOS apps often have a tiny PATH. If the user's
237
- // shell startup files are noisy or slow, fall back to known bins.
238
- }
239
- }
240
-
241
- userShellPathCache.value = [];
242
- userShellPathCache.readAt = now;
243
- return [];
244
- }
245
-
246
- function mergePathEntries(env, preferredEntries) {
247
- const existing = splitPathList(env.PATH || env.Path || '');
248
- const seen = new Set();
249
- const merged = [];
250
-
251
- for (const entry of [...preferredEntries, ...existing]) {
252
- if (!entry) continue;
253
- const key = path.resolve(entry);
254
- if (seen.has(key)) continue;
255
- seen.add(key);
256
- merged.push(entry);
257
- }
258
-
259
- const nextPath = merged.join(pathSeparator());
260
- env.PATH = nextPath;
261
- if ('Path' in env) env.Path = nextPath;
262
- return env;
263
- }
264
-
265
- export function buildCliSpawnEnv(baseEnv = process.env) {
266
- const env = { ...baseEnv };
267
- return mergePathEntries(env, [
268
- CLI_BIN_DIR,
269
- ...collectUserShellPath(baseEnv),
270
- ...collectKnownUserBinDirs(baseEnv),
271
- ]);
272
- }
273
-
274
- /**
275
- * Cross-platform lookup for the Claude Code CLI executable. The
276
- * @anthropic-ai/claude-agent-sdk SDK spawns its target with plain
277
- * `child_process.spawn(command, args)` — no shell, no cross-spawn — which
278
- * means:
279
- * - On Unix, `"claude"` resolves via PATH + shebang. Works out of the box.
280
- * - On Windows, `"claude"` does NOT resolve (Node doesn't traverse PATHEXT
281
- * for bare names), and spawning a `.cmd` shim directly throws EINVAL
282
- * after Node 20.12's CVE-2024-27980 fix. We have to hand the SDK the
283
- * real `.exe` target instead.
284
- *
285
- * We use the OS's own `where`/`which` so we stay consistent with whatever
286
- * the user sees in their shell. When `where` yields a `.cmd` shim, we
287
- * peek inside it (npm-generated shims quote the underlying `.exe` path)
288
- * and return that real binary.
289
- *
290
- * Returns the absolute path, or `null` if nothing turned up — callers
291
- * should leave `pathToClaudeCodeExecutable` unset so the SDK falls back
292
- * to its own bundled native binary.
293
- */
294
- export function resolveClaudeExecutable() {
295
- const isWindows = process.platform === 'win32';
296
- try {
297
- if (isWindows) {
298
- // `where.exe` returns one path per line. Prefer `.exe` over any
299
- // `.cmd` or `.ps1` shim because Node's spawn can exec .exe
300
- // directly — .cmd needs shell:true which the SDK doesn't set.
301
- const stdout = execFileSync('where', ['claude'], {
302
- encoding: 'utf8',
303
- stdio: ['ignore', 'pipe', 'ignore'],
304
- }).trim();
305
- const candidates = stdout.split(/\r?\n/).map((s) => s.trim()).filter(Boolean);
306
- const exe = candidates.find((p) => p.toLowerCase().endsWith('.exe'));
307
- if (exe && fs.existsSync(exe)) return exe;
308
- // Only a `.cmd` shim found. Parse it for the real .exe target.
309
- for (const candidate of candidates) {
310
- if (candidate.toLowerCase().endsWith('.cmd')) {
311
- const underlying = parseNpmCmdShim(candidate);
312
- if (underlying) return underlying;
313
- }
314
- }
315
- return candidates[0] || null;
316
- }
317
- const stdout = execFileSync('which', ['claude'], {
318
- encoding: 'utf8',
319
- stdio: ['ignore', 'pipe', 'ignore'],
320
- }).trim();
321
- return stdout || null;
322
- } catch {
323
- // `where`/`which` returns non-zero when nothing matches. Fall back
324
- // to null so the SDK uses its own resolver.
325
- return null;
326
- }
327
- }
328
-
329
- /**
330
- * Cross-platform lookup for a POSIX `bash` the Claude CLI can drive. On
331
- * Windows, `claude.exe` hard-requires a `bash.exe` (typically from Git
332
- * for Windows) and exits with code 1 + a guidance message if it can't
333
- * find one. The CLI reads the path from `CLAUDE_CODE_GIT_BASH_PATH`
334
- * when set, otherwise probes a short list of known install locations —
335
- * which are exactly the ones we try below.
336
- *
337
- * Returns the absolute path or null. On non-Windows platforms we skip
338
- * the probe entirely and rely on the system `bash` that Claude expects
339
- * to already be on PATH.
340
- */
341
- export function resolveGitBashPath() {
342
- if (process.platform !== 'win32') return null;
343
-
344
- if (process.env.CLAUDE_CODE_GIT_BASH_PATH
345
- && fs.existsSync(process.env.CLAUDE_CODE_GIT_BASH_PATH)) {
346
- return process.env.CLAUDE_CODE_GIT_BASH_PATH;
347
- }
348
-
349
- // 1. `where.exe bash` first — the user already has it on PATH if any
350
- // shell launcher (VS Code, etc.) set it up. Prefer this over our
351
- // hard-coded list because it reflects their actual install.
352
- try {
353
- const stdout = execFileSync('where', ['bash'], {
354
- encoding: 'utf8',
355
- stdio: ['ignore', 'pipe', 'ignore'],
356
- }).trim();
357
- const first = stdout.split(/\r?\n/)[0]?.trim();
358
- if (first && fs.existsSync(first)) return first;
359
- } catch { /* fall through to hard-coded probes */ }
360
-
361
- // 2. Known Git-for-Windows install locations. Covers system-wide,
362
- // per-user, scoop, and chocolatey defaults.
363
- const home = os.homedir();
364
- const programFiles = process.env['ProgramFiles'] || 'C:\\Program Files';
365
- const programFilesX86 = process.env['ProgramFiles(x86)'] || 'C:\\Program Files (x86)';
366
- const localAppData = process.env.LOCALAPPDATA || path.join(home, 'AppData', 'Local');
367
-
368
- const candidates = [
369
- path.join(programFiles, 'Git', 'bin', 'bash.exe'),
370
- path.join(programFiles, 'Git', 'usr', 'bin', 'bash.exe'),
371
- path.join(programFilesX86, 'Git', 'bin', 'bash.exe'),
372
- path.join(programFilesX86, 'Git', 'usr', 'bin', 'bash.exe'),
373
- path.join(localAppData, 'Programs', 'Git', 'bin', 'bash.exe'),
374
- path.join(localAppData, 'Programs', 'Git', 'usr', 'bin', 'bash.exe'),
375
- // Scoop's default install path for the git package
376
- path.join(home, 'scoop', 'apps', 'git', 'current', 'bin', 'bash.exe'),
377
- ];
378
-
379
- for (const candidate of candidates) {
380
- try {
381
- if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {
382
- return candidate;
383
- }
384
- } catch { /* ignore */ }
385
- }
386
-
387
- return null;
388
- }
389
-
390
- /**
391
- * Extract the real .exe target from an npm-generated Windows .cmd shim.
392
- *
393
- * The shim looks like:
394
- * @"%_prog%" "%dp0%\node_modules\@anthropic-ai\claude-code\bin\claude.exe" %*
395
- * We capture the first quoted `.exe` path, then expand `%~dp0` / `%dp0`
396
- * to the shim's own directory so the returned path is absolute.
397
- */
398
- function parseNpmCmdShim(cmdPath) {
399
- try {
400
- const content = fs.readFileSync(cmdPath, 'utf8');
401
- const match = content.match(/"([^"]+\.exe)"/i);
402
- if (!match) return null;
403
- const rel = match[1];
404
- const dir = path.dirname(cmdPath);
405
- const resolved = rel
406
- .replace(/%~?dp0%?\\?/gi, `${dir}${path.sep}`)
407
- .replace(/%~dp0/gi, dir);
408
- return fs.existsSync(resolved) ? resolved : null;
409
- } catch {
410
- return null;
411
- }
412
- }
413
-
414
- /**
415
- * Search PATH for an executable, including the Windows extension variants.
416
- * Returns the absolute path or null. Plain Node has no cross-platform
417
- * equivalent of `which`, so we roll our own — it's small enough to not be
418
- * worth an extra dependency.
419
- */
420
- export function findExecutableOnPath(name, env = process.env) {
421
- const isWindows = process.platform === 'win32';
422
- const sep = isWindows ? ';' : ':';
423
- const paths = (env.PATH || env.Path || '').split(sep).filter(Boolean);
424
-
425
- // Common native-installer / per-user fallback paths that aren't always on
426
- // the daemon's PATH but are on the user's interactive shell PATH. We
427
- // union them in so "pixcode --no-daemon" and "pixcode daemon" agree.
428
- const home = os.homedir();
429
- if (isWindows) {
430
- paths.push(path.join(env.APPDATA || path.join(home, 'AppData', 'Roaming'), 'npm'));
431
- paths.push(path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'Programs', `${name}-code`));
432
- paths.push(path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'AnthropicClaude'));
433
- } else {
434
- paths.push(...collectUserShellPath(env));
435
- paths.push(...collectKnownUserBinDirs(env));
436
- }
437
-
438
- const exts = isWindows
439
- ? ['.cmd', '.exe', '.bat', '.ps1', '']
440
- : [''];
441
-
442
- for (const dir of paths) {
443
- for (const ext of exts) {
444
- const candidate = path.join(dir, name + ext);
445
- try {
446
- if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {
447
- return candidate;
448
- }
449
- } catch {
450
- // Permission denied / broken symlink — ignore and keep looking.
451
- }
452
- }
453
- }
454
- return null;
455
- }
456
-
457
- /**
458
- * Resolve `npm` next to the currently-running `node` binary. This is
459
- * more reliable than trusting PATH — when Pixcode runs as a daemon, PATH
460
- * is often minimal and doesn't include the user's node install.
461
- */
462
- export function resolveNpmCommand(env = process.env) {
463
- const nodeDir = path.dirname(process.execPath);
464
- const isWindows = process.platform === 'win32';
465
- const candidates = isWindows
466
- ? ['npm.cmd', 'npm.exe']
467
- : ['npm'];
468
- for (const c of candidates) {
469
- const full = path.join(nodeDir, c);
470
- if (fs.existsSync(full)) return full;
471
- }
472
- // Windows sometimes ships npm in a sibling "npm" directory.
473
- if (isWindows) {
474
- const siblingNpm = path.join(nodeDir, 'node_modules', 'npm', 'bin', 'npm-cli.js');
475
- if (fs.existsSync(siblingNpm)) {
476
- return siblingNpm; // we'll invoke `node <npm-cli.js>`
477
- }
478
- }
479
-
480
- const resolvedFromPath = findExecutableOnPath('npm', env);
481
- if (resolvedFromPath) return resolvedFromPath;
482
-
483
- return null;
484
- }
485
-
486
- function packageFromCommand(installCmd) {
487
- // Legacy callers still pass `npm install -g <pkg>` strings — extract
488
- // the @scope/name so the local installer can reuse the same input.
489
- const match = String(installCmd).match(/@[^\s]+\/[^\s]+|[\w.-]+(?:@[\w.-]+)?$/);
490
- return match ? match[0] : installCmd;
491
- }
492
-
493
- export function createInstallJob({ provider, installCmd, packageName }) {
494
- const pkg = packageName || packageFromCommand(installCmd);
495
- const id = randomUUID();
496
- const emitter = new EventEmitter();
497
- emitter.setMaxListeners(20);
498
-
499
- const job = {
500
- id,
501
- provider,
502
- installCmd,
503
- package: pkg,
504
- status: 'running',
505
- startedAt: new Date().toISOString(),
506
- finishedAt: null,
507
- exitCode: null,
508
- error: null,
509
- logs: [],
510
- emitter,
511
- child: null,
512
- timer: null,
513
- };
514
-
515
- const appendLog = (stream, chunk) => {
516
- const entry = { stream, chunk, at: Date.now() };
517
- job.logs.push(entry);
518
- if (job.logs.length > 2000) {
519
- job.logs.splice(0, job.logs.length - 2000);
520
- }
521
- emitter.emit('log', entry);
522
- };
523
-
524
- try {
525
- ensureCliHome();
526
- } catch (err) {
527
- job.status = 'error';
528
- job.error = `Could not create ${CLI_HOME}: ${err?.message || err}`;
529
- job.finishedAt = new Date().toISOString();
530
- appendLog('stderr', job.error + '\n');
531
- emitter.emit('done', buildDonePayload(job));
532
- scheduleCleanup(job);
533
- jobs.set(id, job);
534
- return job;
535
- }
536
-
537
- appendLog('meta', `Installing ${pkg} into ${CLI_HOME}\n`);
538
- appendLog('meta', `(sandboxed — no sudo / admin required)\n`);
539
-
540
- const installEnv = buildCliSpawnEnv(process.env);
541
- const npmCmd = resolveNpmCommand(installEnv);
542
- if (!npmCmd) {
543
- job.status = 'error';
544
- job.error = 'npm was not found. Install Node.js/npm or add it to your macOS login shell PATH, then click Refresh.';
545
- job.finishedAt = new Date().toISOString();
546
- appendLog('stderr', job.error + '\n');
547
- emitter.emit('done', buildDonePayload(job));
548
- scheduleCleanup(job);
549
- jobs.set(id, job);
550
- return job;
551
- }
552
-
553
- const useNodeRunner = npmCmd.endsWith('.js');
554
-
555
- const cmd = useNodeRunner ? process.execPath : npmCmd;
556
- const args = useNodeRunner
557
- ? [npmCmd, 'install', pkg, '--no-audit', '--no-fund', '--loglevel=http']
558
- : ['install', pkg, '--no-audit', '--no-fund', '--loglevel=http'];
559
-
560
- appendLog('meta', `$ ${cmd} ${args.join(' ')}\n`);
561
-
562
- let child;
563
- try {
564
- child = spawn(cmd, args, {
565
- cwd: CLI_HOME,
566
- env: { ...installEnv, npm_config_yes: 'true' },
567
- stdio: ['ignore', 'pipe', 'pipe'],
568
- windowsHide: true,
569
- // cross-spawn handles .cmd/.bat resolution itself — no shell
570
- // needed. Passing `shell: true` here would re-introduce the
571
- // space-in-path tokenisation bug that caused "'C:\Program' is
572
- // not recognized" on Windows installs of Node.
573
- });
574
- } catch (err) {
575
- const message = err?.message || String(err);
576
- console.error(`[install-job:${provider}:${id}] Spawn failed:`, message);
577
- job.status = 'error';
578
- job.error = `Failed to launch npm: ${message}`;
579
- job.finishedAt = new Date().toISOString();
580
- appendLog('stderr', job.error + '\n');
581
- emitter.emit('done', buildDonePayload(job));
582
- scheduleCleanup(job);
583
- jobs.set(id, job);
584
- return job;
585
- }
586
-
587
- job.child = child;
588
- child.stdout.on('data', (buf) => appendLog('stdout', buf.toString()));
589
- child.stderr.on('data', (buf) => appendLog('stderr', buf.toString()));
590
-
591
- child.on('error', (err) => {
592
- if (job.status !== 'running') return;
593
- job.status = 'error';
594
- job.error = `npm process error: ${err.message}`;
595
- job.finishedAt = new Date().toISOString();
596
- appendLog('stderr', job.error + '\n');
597
- emitter.emit('done', buildDonePayload(job));
598
- scheduleCleanup(job);
599
- });
600
-
601
- child.on('close', (code, signal) => {
602
- if (job.status !== 'running') return;
603
- job.exitCode = code ?? null;
604
- job.finishedAt = new Date().toISOString();
605
-
606
- if (code !== 0) {
607
- job.status = 'error';
608
- job.error = signal
609
- ? `Install killed by signal ${signal}`
610
- : `npm exited with code ${code}`;
611
- emitter.emit('done', buildDonePayload(job));
612
- scheduleCleanup(job);
613
- return;
614
- }
615
-
616
- // Verify the binary actually landed. If we don't check, a package
617
- // without a `bin` entry (or a half-extracted tarball) would still
618
- // read as "success" and the user would be confused when auth
619
- // status stays red.
620
- const binName = PACKAGE_BINARIES[pkg] || provider;
621
- const binaryPath = findInstalledBinary(binName);
622
- if (!binaryPath) {
623
- job.status = 'error';
624
- job.error = `npm exited cleanly but ${binName} was not found in ${CLI_BIN_DIR}`;
625
- appendLog('stderr', job.error + '\n');
626
- emitter.emit('done', buildDonePayload(job));
627
- scheduleCleanup(job);
628
- return;
629
- }
630
-
631
- // Make sure our live server process can resolve the new binary
632
- // from this moment on, without a restart. primeCliBinPath is
633
- // idempotent so re-calling after each install is cheap.
634
- primeCliBinPath();
635
-
636
- appendLog('meta', `✓ Installed ${binName} → ${binaryPath}\n`);
637
- job.status = 'done';
638
- job.binaryPath = binaryPath;
639
- emitter.emit('done', buildDonePayload(job));
640
- scheduleCleanup(job);
641
- });
642
-
643
- job.timer = setTimeout(() => {
644
- if (job.status !== 'running') return;
645
- try { child.kill('SIGKILL'); } catch { /* noop */ }
646
- job.status = 'error';
647
- job.error = 'Install timed out after 10 minutes';
648
- job.finishedAt = new Date().toISOString();
649
- appendLog('stderr', job.error + '\n');
650
- emitter.emit('done', buildDonePayload(job));
651
- scheduleCleanup(job);
652
- }, HARD_TIMEOUT_MS);
653
-
654
- jobs.set(id, job);
655
- return job;
656
- }
657
-
658
- function findInstalledBinary(name) {
659
- const isWindows = process.platform === 'win32';
660
- const candidates = isWindows
661
- ? [`${name}.cmd`, `${name}.exe`, name]
662
- : [name];
663
- for (const c of candidates) {
664
- const full = path.join(CLI_BIN_DIR, c);
665
- if (fs.existsSync(full)) return full;
666
- }
667
- return null;
668
- }
669
-
670
- function buildDonePayload(job) {
671
- if (job.status === 'done') {
672
- return {
673
- success: true,
674
- exitCode: job.exitCode,
675
- binaryPath: job.binaryPath,
676
- message: `${job.provider} installed. Refreshing auth status…`,
677
- };
678
- }
679
- return {
680
- success: false,
681
- exitCode: job.exitCode,
682
- error: job.error || 'Install failed',
683
- };
684
- }
685
-
686
- function scheduleCleanup(job) {
687
- if (job.timer) {
688
- clearTimeout(job.timer);
689
- job.timer = null;
690
- }
691
- setTimeout(() => {
692
- jobs.delete(job.id);
693
- }, FINISHED_TTL_MS);
694
- }
695
-
696
- export function getInstallJob(id) {
697
- return jobs.get(id) || null;
698
- }
699
-
700
- export function cancelInstallJob(id) {
701
- const job = jobs.get(id);
702
- if (!job) return false;
703
- if (job.status !== 'running') return false;
704
- try { job.child?.kill(); } catch { /* noop */ }
705
- job.status = 'error';
706
- job.error = 'Install cancelled';
707
- job.finishedAt = new Date().toISOString();
708
- job.emitter.emit('done', buildDonePayload(job));
709
- scheduleCleanup(job);
710
- return true;
711
- }
712
-
713
- export function snapshotDonePayload(job) {
714
- return buildDonePayload(job);
715
- }
1
+ /**
2
+ * In-memory install-job registry + sandboxed local CLI installer.
3
+ *
4
+ * Why not `npm install -g`:
5
+ * - Requires admin/sudo on Windows and most Linux distros. When Pixcode
6
+ * runs as a non-privileged daemon, -g fails with EACCES and the user
7
+ * sees a blank log with no actionable error.
8
+ * - Even on Windows desktop, npm's global prefix is sometimes broken
9
+ * (AppData permissions, antivirus quarantining node_modules/.bin).
10
+ * - CI/docker/VPS setups often don't have `npm` on the daemon's PATH at
11
+ * all, even when the user's interactive shell does.
12
+ *
13
+ * What we do instead:
14
+ * - Install targets go into `~/.pixcode/cli-bin/` as LOCAL dependencies
15
+ * of a pixcode-owned package.json (no -g, no sudo, no UAC).
16
+ * - Resolve `npm` from the same Node install that's running the server
17
+ * (sibling file to `process.execPath`) so PATH environment doesn't
18
+ * matter.
19
+ * - On server boot, `~/.pixcode/cli-bin/node_modules/.bin` is prepended
20
+ * to `process.env.PATH`. Every existing `cross-spawn(binary)` call
21
+ * (in claude-auth, gemini-cli, qwen-code-cli, etc.) then resolves to
22
+ * the locally installed binary without any change to the adapter code.
23
+ *
24
+ * The HTTP/stream side of the API is the same as before:
25
+ * - POST /install → spawns the child, returns { jobId }
26
+ * - GET /install/:jobId/stream → EventSource that replays the buffered
27
+ * transcript and then streams live chunks.
28
+ * - DELETE /install/:jobId → cancels an in-flight install.
29
+ *
30
+ * Jobs linger 10 minutes after completion so late subscribers still see
31
+ * the outcome.
32
+ */
33
+ import { execFileSync } from 'node:child_process';
34
+ import { EventEmitter } from 'node:events';
35
+ import { randomUUID } from 'node:crypto';
36
+ import fs from 'node:fs';
37
+ import os from 'node:os';
38
+ import path from 'node:path';
39
+
40
+ // Use cross-spawn instead of node:child_process.spawn. On Windows, node's
41
+ // spawn cannot invoke `.cmd` / `.bat` files without `shell: true`, and with
42
+ // `shell: true` it tokenises on spaces — so a valid npm path like
43
+ // `C:\Program Files\nodejs\npm.cmd` gets split into "C:\Program" + "Files...".
44
+ // cross-spawn shells out through cmd.exe with proper quoting transparently
45
+ // and is already a transitive dependency we can safely re-use.
46
+ import spawn from 'cross-spawn';
47
+
48
+ const jobs = new Map();
49
+ const FINISHED_TTL_MS = 10 * 60 * 1000;
50
+ const HARD_TIMEOUT_MS = 10 * 60 * 1000;
51
+ const USER_SHELL_PATH_CACHE_TTL_MS = 5 * 60 * 1000;
52
+ const userShellPathCache = {
53
+ value: null,
54
+ readAt: 0,
55
+ };
56
+
57
+ export const CLI_HOME = path.join(os.homedir(), '.pixcode', 'cli-bin');
58
+ export const CLI_BIN_DIR = path.join(CLI_HOME, 'node_modules', '.bin');
59
+
60
+ /**
61
+ * npm package → the binary name it installs. Used to verify the install
62
+ * actually dropped an executable we can run, since npm can exit(0) even
63
+ * when a package has no `bin` entry or our PATH wiring is wrong.
64
+ */
65
+ const PACKAGE_BINARIES = {
66
+ '@anthropic-ai/claude-code': 'claude',
67
+ '@openai/codex': 'codex',
68
+ '@google/gemini-cli': 'gemini',
69
+ '@qwen-code/qwen-code': 'qwen',
70
+ 'opencode-ai': 'opencode',
71
+ 'task-master': 'task-master',
72
+ };
73
+
74
+ /**
75
+ * Make sure `CLI_HOME` exists with a minimal package.json so `npm install`
76
+ * doesn't walk up to some unrelated parent and pollute it.
77
+ */
78
+ function ensureCliHome() {
79
+ fs.mkdirSync(CLI_HOME, { recursive: true });
80
+ const pkgPath = path.join(CLI_HOME, 'package.json');
81
+ if (!fs.existsSync(pkgPath)) {
82
+ const pkg = {
83
+ name: 'pixcode-cli-bin',
84
+ private: true,
85
+ version: '0.0.0',
86
+ description:
87
+ 'Pixcode-managed sandbox for provider CLIs (claude/codex/gemini/qwen). '
88
+ + 'Safe to delete; Pixcode will re-create it on next install.',
89
+ };
90
+ fs.writeFileSync(pkgPath, JSON.stringify(pkg, null, 2));
91
+ }
92
+ }
93
+
94
+ /**
95
+ * Prepend the pixcode-managed bin dir to PATH. Called at server boot so
96
+ * every subsequent `spawn('claude'|'gemini'|'codex'|'qwen', …)` in the
97
+ * provider adapters (which use cross-spawn with bare names) resolves to
98
+ * the locally installed binary without any per-adapter change.
99
+ */
100
+ export function primeCliBinPath(env = process.env) {
101
+ ensureCliHome();
102
+ const augmentedEnv = buildCliSpawnEnv(env);
103
+ env.PATH = augmentedEnv.PATH;
104
+ if ('Path' in env || augmentedEnv.Path) env.Path = augmentedEnv.Path || augmentedEnv.PATH;
105
+ // Once PATH is ready, resolve any well-known provider binaries to absolute
106
+ // paths and export them as *_CLI_PATH env vars. This side-steps a Windows
107
+ // gotcha: `child_process.spawn('claude', …)` does NOT auto-resolve .cmd /
108
+ // .bat extensions, and the Claude Agent SDK calls spawn directly instead
109
+ // of via cross-spawn — so a bare "claude" on PATH works in a shell but
110
+ // fails inside the SDK. Pinning the full path side-steps it entirely.
111
+ resolveProviderExecutables(env);
112
+ }
113
+
114
+ /**
115
+ * Scan PATH (plus known native-installer locations) for every provider
116
+ * binary we ship support for, and export *_CLI_PATH env vars pointing to
117
+ * the absolute executable. Existing vars are left alone so users can
118
+ * override detection.
119
+ */
120
+ export function resolveProviderExecutables(env = process.env) {
121
+ // Claude is intentionally omitted. The Claude Agent SDK ships a bundled
122
+ // native binary per platform (@anthropic-ai/claude-agent-sdk-<os>-<arch>)
123
+ // and resolves it automatically. Exporting CLAUDE_CLI_PATH here would
124
+ // override that and hand a `.cmd` shim to Node's spawn on Windows,
125
+ // which then throws EINVAL (spawn can't exec .cmd files directly).
126
+ //
127
+ // The other providers use cross-spawn in our own adapters, which
128
+ // handles .cmd/.bat resolution on Windows. Forcing an absolute path
129
+ // there is still helpful because cross-spawn.sync without quoting
130
+ // can hit edge cases when PATH contains spaces.
131
+ const providers = [
132
+ { name: 'codex', envKey: 'CODEX_CLI_PATH' },
133
+ { name: 'gemini', envKey: 'GEMINI_CLI_PATH' },
134
+ { name: 'qwen', envKey: 'QWEN_CLI_PATH' },
135
+ { name: 'opencode', envKey: 'OPENCODE_CLI_PATH' },
136
+ { name: 'cursor-agent', envKey: 'CURSOR_CLI_PATH' },
137
+ ];
138
+ for (const { name, envKey } of providers) {
139
+ if (env[envKey]) continue;
140
+ const resolved = findExecutableOnPath(name, env);
141
+ if (resolved) env[envKey] = resolved;
142
+ }
143
+ }
144
+
145
+ function pathSeparator() {
146
+ return process.platform === 'win32' ? ';' : ':';
147
+ }
148
+
149
+ function splitPathList(value) {
150
+ return String(value || '').split(pathSeparator()).map((entry) => entry.trim()).filter(Boolean);
151
+ }
152
+
153
+ function collectNvmNodeBins(home) {
154
+ const versionsDir = path.join(home, '.nvm', 'versions', 'node');
155
+ try {
156
+ return fs.readdirSync(versionsDir)
157
+ .map((version) => path.join(versionsDir, version, 'bin'))
158
+ .filter((candidate) => {
159
+ try {
160
+ return fs.statSync(candidate).isDirectory();
161
+ } catch {
162
+ return false;
163
+ }
164
+ })
165
+ .sort()
166
+ .reverse();
167
+ } catch {
168
+ return [];
169
+ }
170
+ }
171
+
172
+ function collectKnownUserBinDirs(env = process.env) {
173
+ const home = os.homedir();
174
+ if (process.platform === 'win32') {
175
+ return [
176
+ path.join(env.APPDATA || path.join(home, 'AppData', 'Roaming'), 'npm'),
177
+ path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'Programs', 'nodejs'),
178
+ ];
179
+ }
180
+
181
+ return [
182
+ CLI_BIN_DIR,
183
+ path.dirname(process.execPath),
184
+ ...collectNvmNodeBins(home),
185
+ path.join(home, '.volta', 'bin'),
186
+ path.join(home, '.asdf', 'shims'),
187
+ path.join(home, '.bun', 'bin'),
188
+ path.join(home, '.local', 'bin'),
189
+ path.join(home, '.npm-global', 'bin'),
190
+ '/opt/homebrew/bin',
191
+ '/opt/homebrew/sbin',
192
+ '/usr/local/bin',
193
+ '/usr/local/sbin',
194
+ '/usr/bin',
195
+ '/bin',
196
+ ];
197
+ }
198
+
199
+ export function collectUserShellPath(env = process.env) {
200
+ if (process.platform === 'win32') return [];
201
+
202
+ const now = Date.now();
203
+ if (userShellPathCache.value && now - userShellPathCache.readAt < USER_SHELL_PATH_CACHE_TTL_MS) {
204
+ return userShellPathCache.value;
205
+ }
206
+
207
+ const shells = [env.SHELL, '/bin/zsh', '/bin/bash']
208
+ .filter(Boolean)
209
+ .filter((candidate, index, list) => list.indexOf(candidate) === index)
210
+ .filter((candidate) => {
211
+ try {
212
+ return fs.existsSync(candidate);
213
+ } catch {
214
+ return false;
215
+ }
216
+ });
217
+
218
+ const marker = '__PIXCODE_LOGIN_PATH__=';
219
+ for (const shell of shells) {
220
+ try {
221
+ const output = execFileSync(shell, ['-lc', `printf '\\n${marker}%s\\n' "$PATH"`], {
222
+ encoding: 'utf8',
223
+ env,
224
+ timeout: 2500,
225
+ stdio: ['ignore', 'pipe', 'ignore'],
226
+ });
227
+ const line = output.split(/\r?\n/).reverse().find((part) => part.startsWith(marker));
228
+ const shellPath = line?.slice(marker.length);
229
+ if (shellPath) {
230
+ const entries = splitPathList(shellPath);
231
+ userShellPathCache.value = entries;
232
+ userShellPathCache.readAt = now;
233
+ return entries;
234
+ }
235
+ } catch {
236
+ // GUI-launched macOS apps often have a tiny PATH. If the user's
237
+ // shell startup files are noisy or slow, fall back to known bins.
238
+ }
239
+ }
240
+
241
+ userShellPathCache.value = [];
242
+ userShellPathCache.readAt = now;
243
+ return [];
244
+ }
245
+
246
+ function mergePathEntries(env, preferredEntries) {
247
+ const existing = splitPathList(env.PATH || env.Path || '');
248
+ const seen = new Set();
249
+ const merged = [];
250
+
251
+ for (const entry of [...preferredEntries, ...existing]) {
252
+ if (!entry) continue;
253
+ const key = path.resolve(entry);
254
+ if (seen.has(key)) continue;
255
+ seen.add(key);
256
+ merged.push(entry);
257
+ }
258
+
259
+ const nextPath = merged.join(pathSeparator());
260
+ env.PATH = nextPath;
261
+ if ('Path' in env) env.Path = nextPath;
262
+ return env;
263
+ }
264
+
265
+ export function buildCliSpawnEnv(baseEnv = process.env) {
266
+ const env = { ...baseEnv };
267
+ return mergePathEntries(env, [
268
+ CLI_BIN_DIR,
269
+ ...collectUserShellPath(baseEnv),
270
+ ...collectKnownUserBinDirs(baseEnv),
271
+ ]);
272
+ }
273
+
274
+ /**
275
+ * Cross-platform lookup for the Claude Code CLI executable. The
276
+ * @anthropic-ai/claude-agent-sdk SDK spawns its target with plain
277
+ * `child_process.spawn(command, args)` — no shell, no cross-spawn — which
278
+ * means:
279
+ * - On Unix, `"claude"` resolves via PATH + shebang. Works out of the box.
280
+ * - On Windows, `"claude"` does NOT resolve (Node doesn't traverse PATHEXT
281
+ * for bare names), and spawning a `.cmd` shim directly throws EINVAL
282
+ * after Node 20.12's CVE-2024-27980 fix. We have to hand the SDK the
283
+ * real `.exe` target instead.
284
+ *
285
+ * We use the OS's own `where`/`which` so we stay consistent with whatever
286
+ * the user sees in their shell. When `where` yields a `.cmd` shim, we
287
+ * peek inside it (npm-generated shims quote the underlying `.exe` path)
288
+ * and return that real binary.
289
+ *
290
+ * Returns the absolute path, or `null` if nothing turned up — callers
291
+ * should leave `pathToClaudeCodeExecutable` unset so the SDK falls back
292
+ * to its own bundled native binary.
293
+ */
294
+ export function resolveClaudeExecutable() {
295
+ const isWindows = process.platform === 'win32';
296
+ try {
297
+ if (isWindows) {
298
+ // `where.exe` returns one path per line. Prefer `.exe` over any
299
+ // `.cmd` or `.ps1` shim because Node's spawn can exec .exe
300
+ // directly — .cmd needs shell:true which the SDK doesn't set.
301
+ const stdout = execFileSync('where', ['claude'], {
302
+ encoding: 'utf8',
303
+ stdio: ['ignore', 'pipe', 'ignore'],
304
+ }).trim();
305
+ const candidates = stdout.split(/\r?\n/).map((s) => s.trim()).filter(Boolean);
306
+ const exe = candidates.find((p) => p.toLowerCase().endsWith('.exe'));
307
+ if (exe && fs.existsSync(exe)) return exe;
308
+ // Only a `.cmd` shim found. Parse it for the real .exe target.
309
+ for (const candidate of candidates) {
310
+ if (candidate.toLowerCase().endsWith('.cmd')) {
311
+ const underlying = parseNpmCmdShim(candidate);
312
+ if (underlying) return underlying;
313
+ }
314
+ }
315
+ return candidates[0] || null;
316
+ }
317
+ const stdout = execFileSync('which', ['claude'], {
318
+ encoding: 'utf8',
319
+ stdio: ['ignore', 'pipe', 'ignore'],
320
+ }).trim();
321
+ return stdout || null;
322
+ } catch {
323
+ // `where`/`which` returns non-zero when nothing matches. Fall back
324
+ // to null so the SDK uses its own resolver.
325
+ return null;
326
+ }
327
+ }
328
+
329
+ /**
330
+ * Cross-platform lookup for a POSIX `bash` the Claude CLI can drive. On
331
+ * Windows, `claude.exe` hard-requires a `bash.exe` (typically from Git
332
+ * for Windows) and exits with code 1 + a guidance message if it can't
333
+ * find one. The CLI reads the path from `CLAUDE_CODE_GIT_BASH_PATH`
334
+ * when set, otherwise probes a short list of known install locations —
335
+ * which are exactly the ones we try below.
336
+ *
337
+ * Returns the absolute path or null. On non-Windows platforms we skip
338
+ * the probe entirely and rely on the system `bash` that Claude expects
339
+ * to already be on PATH.
340
+ */
341
+ export function resolveGitBashPath() {
342
+ if (process.platform !== 'win32') return null;
343
+
344
+ if (process.env.CLAUDE_CODE_GIT_BASH_PATH
345
+ && fs.existsSync(process.env.CLAUDE_CODE_GIT_BASH_PATH)) {
346
+ return process.env.CLAUDE_CODE_GIT_BASH_PATH;
347
+ }
348
+
349
+ // 1. `where.exe bash` first — the user already has it on PATH if any
350
+ // shell launcher (VS Code, etc.) set it up. Prefer this over our
351
+ // hard-coded list because it reflects their actual install.
352
+ try {
353
+ const stdout = execFileSync('where', ['bash'], {
354
+ encoding: 'utf8',
355
+ stdio: ['ignore', 'pipe', 'ignore'],
356
+ }).trim();
357
+ const first = stdout.split(/\r?\n/)[0]?.trim();
358
+ if (first && fs.existsSync(first)) return first;
359
+ } catch { /* fall through to hard-coded probes */ }
360
+
361
+ // 2. Known Git-for-Windows install locations. Covers system-wide,
362
+ // per-user, scoop, and chocolatey defaults.
363
+ const home = os.homedir();
364
+ const programFiles = process.env['ProgramFiles'] || 'C:\\Program Files';
365
+ const programFilesX86 = process.env['ProgramFiles(x86)'] || 'C:\\Program Files (x86)';
366
+ const localAppData = process.env.LOCALAPPDATA || path.join(home, 'AppData', 'Local');
367
+
368
+ const candidates = [
369
+ path.join(programFiles, 'Git', 'bin', 'bash.exe'),
370
+ path.join(programFiles, 'Git', 'usr', 'bin', 'bash.exe'),
371
+ path.join(programFilesX86, 'Git', 'bin', 'bash.exe'),
372
+ path.join(programFilesX86, 'Git', 'usr', 'bin', 'bash.exe'),
373
+ path.join(localAppData, 'Programs', 'Git', 'bin', 'bash.exe'),
374
+ path.join(localAppData, 'Programs', 'Git', 'usr', 'bin', 'bash.exe'),
375
+ // Scoop's default install path for the git package
376
+ path.join(home, 'scoop', 'apps', 'git', 'current', 'bin', 'bash.exe'),
377
+ ];
378
+
379
+ for (const candidate of candidates) {
380
+ try {
381
+ if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {
382
+ return candidate;
383
+ }
384
+ } catch { /* ignore */ }
385
+ }
386
+
387
+ return null;
388
+ }
389
+
390
+ /**
391
+ * Extract the real .exe target from an npm-generated Windows .cmd shim.
392
+ *
393
+ * The shim looks like:
394
+ * @"%_prog%" "%dp0%\node_modules\@anthropic-ai\claude-code\bin\claude.exe" %*
395
+ * We capture the first quoted `.exe` path, then expand `%~dp0` / `%dp0`
396
+ * to the shim's own directory so the returned path is absolute.
397
+ */
398
+ function parseNpmCmdShim(cmdPath) {
399
+ try {
400
+ const content = fs.readFileSync(cmdPath, 'utf8');
401
+ const match = content.match(/"([^"]+\.exe)"/i);
402
+ if (!match) return null;
403
+ const rel = match[1];
404
+ const dir = path.dirname(cmdPath);
405
+ const resolved = rel
406
+ .replace(/%~?dp0%?\\?/gi, `${dir}${path.sep}`)
407
+ .replace(/%~dp0/gi, dir);
408
+ return fs.existsSync(resolved) ? resolved : null;
409
+ } catch {
410
+ return null;
411
+ }
412
+ }
413
+
414
+ /**
415
+ * Search PATH for an executable, including the Windows extension variants.
416
+ * Returns the absolute path or null. Plain Node has no cross-platform
417
+ * equivalent of `which`, so we roll our own — it's small enough to not be
418
+ * worth an extra dependency.
419
+ */
420
+ export function findExecutableOnPath(name, env = process.env) {
421
+ const isWindows = process.platform === 'win32';
422
+ const sep = isWindows ? ';' : ':';
423
+ const paths = (env.PATH || env.Path || '').split(sep).filter(Boolean);
424
+
425
+ // Common native-installer / per-user fallback paths that aren't always on
426
+ // the daemon's PATH but are on the user's interactive shell PATH. We
427
+ // union them in so "pixcode --no-daemon" and "pixcode daemon" agree.
428
+ const home = os.homedir();
429
+ if (isWindows) {
430
+ paths.push(path.join(env.APPDATA || path.join(home, 'AppData', 'Roaming'), 'npm'));
431
+ paths.push(path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'Programs', `${name}-code`));
432
+ paths.push(path.join(env.LOCALAPPDATA || path.join(home, 'AppData', 'Local'), 'AnthropicClaude'));
433
+ } else {
434
+ paths.push(...collectUserShellPath(env));
435
+ paths.push(...collectKnownUserBinDirs(env));
436
+ }
437
+
438
+ const exts = isWindows
439
+ ? ['.cmd', '.exe', '.bat', '.ps1', '']
440
+ : [''];
441
+
442
+ for (const dir of paths) {
443
+ for (const ext of exts) {
444
+ const candidate = path.join(dir, name + ext);
445
+ try {
446
+ if (fs.existsSync(candidate) && fs.statSync(candidate).isFile()) {
447
+ return candidate;
448
+ }
449
+ } catch {
450
+ // Permission denied / broken symlink — ignore and keep looking.
451
+ }
452
+ }
453
+ }
454
+ return null;
455
+ }
456
+
457
+ /**
458
+ * Resolve `npm` next to the currently-running `node` binary. This is
459
+ * more reliable than trusting PATH — when Pixcode runs as a daemon, PATH
460
+ * is often minimal and doesn't include the user's node install.
461
+ */
462
+ export function resolveNpmCommand(env = process.env) {
463
+ const nodeDir = path.dirname(process.execPath);
464
+ const isWindows = process.platform === 'win32';
465
+ const candidates = isWindows
466
+ ? ['npm.cmd', 'npm.exe']
467
+ : ['npm'];
468
+ for (const c of candidates) {
469
+ const full = path.join(nodeDir, c);
470
+ if (fs.existsSync(full)) return full;
471
+ }
472
+ // Windows sometimes ships npm in a sibling "npm" directory.
473
+ if (isWindows) {
474
+ const siblingNpm = path.join(nodeDir, 'node_modules', 'npm', 'bin', 'npm-cli.js');
475
+ if (fs.existsSync(siblingNpm)) {
476
+ return siblingNpm; // we'll invoke `node <npm-cli.js>`
477
+ }
478
+ }
479
+
480
+ const resolvedFromPath = findExecutableOnPath('npm', env);
481
+ if (resolvedFromPath) return resolvedFromPath;
482
+
483
+ return null;
484
+ }
485
+
486
+ function packageFromCommand(installCmd) {
487
+ // Legacy callers still pass `npm install -g <pkg>` strings — extract
488
+ // the @scope/name so the local installer can reuse the same input.
489
+ const match = String(installCmd).match(/@[^\s]+\/[^\s]+|[\w.-]+(?:@[\w.-]+)?$/);
490
+ return match ? match[0] : installCmd;
491
+ }
492
+
493
+ export function createInstallJob({ provider, installCmd, packageName }) {
494
+ const pkg = packageName || packageFromCommand(installCmd);
495
+ const id = randomUUID();
496
+ const emitter = new EventEmitter();
497
+ emitter.setMaxListeners(20);
498
+
499
+ const job = {
500
+ id,
501
+ provider,
502
+ installCmd,
503
+ package: pkg,
504
+ status: 'running',
505
+ startedAt: new Date().toISOString(),
506
+ finishedAt: null,
507
+ exitCode: null,
508
+ error: null,
509
+ logs: [],
510
+ emitter,
511
+ child: null,
512
+ timer: null,
513
+ };
514
+
515
+ const appendLog = (stream, chunk) => {
516
+ const entry = { stream, chunk, at: Date.now() };
517
+ job.logs.push(entry);
518
+ if (job.logs.length > 2000) {
519
+ job.logs.splice(0, job.logs.length - 2000);
520
+ }
521
+ emitter.emit('log', entry);
522
+ };
523
+
524
+ try {
525
+ ensureCliHome();
526
+ } catch (err) {
527
+ job.status = 'error';
528
+ job.error = `Could not create ${CLI_HOME}: ${err?.message || err}`;
529
+ job.finishedAt = new Date().toISOString();
530
+ appendLog('stderr', job.error + '\n');
531
+ emitter.emit('done', buildDonePayload(job));
532
+ scheduleCleanup(job);
533
+ jobs.set(id, job);
534
+ return job;
535
+ }
536
+
537
+ appendLog('meta', `Installing ${pkg} into ${CLI_HOME}\n`);
538
+ appendLog('meta', `(sandboxed — no sudo / admin required)\n`);
539
+
540
+ const installEnv = buildCliSpawnEnv(process.env);
541
+ const npmCmd = resolveNpmCommand(installEnv);
542
+ if (!npmCmd) {
543
+ job.status = 'error';
544
+ job.error = 'npm was not found. Install Node.js/npm or add it to your macOS login shell PATH, then click Refresh.';
545
+ job.finishedAt = new Date().toISOString();
546
+ appendLog('stderr', job.error + '\n');
547
+ emitter.emit('done', buildDonePayload(job));
548
+ scheduleCleanup(job);
549
+ jobs.set(id, job);
550
+ return job;
551
+ }
552
+
553
+ const useNodeRunner = npmCmd.endsWith('.js');
554
+
555
+ const cmd = useNodeRunner ? process.execPath : npmCmd;
556
+ const args = useNodeRunner
557
+ ? [npmCmd, 'install', pkg, '--no-audit', '--no-fund', '--loglevel=http']
558
+ : ['install', pkg, '--no-audit', '--no-fund', '--loglevel=http'];
559
+
560
+ appendLog('meta', `$ ${cmd} ${args.join(' ')}\n`);
561
+
562
+ let child;
563
+ try {
564
+ child = spawn(cmd, args, {
565
+ cwd: CLI_HOME,
566
+ env: { ...installEnv, npm_config_yes: 'true' },
567
+ stdio: ['ignore', 'pipe', 'pipe'],
568
+ windowsHide: true,
569
+ // cross-spawn handles .cmd/.bat resolution itself — no shell
570
+ // needed. Passing `shell: true` here would re-introduce the
571
+ // space-in-path tokenisation bug that caused "'C:\Program' is
572
+ // not recognized" on Windows installs of Node.
573
+ });
574
+ } catch (err) {
575
+ const message = err?.message || String(err);
576
+ console.error(`[install-job:${provider}:${id}] Spawn failed:`, message);
577
+ job.status = 'error';
578
+ job.error = `Failed to launch npm: ${message}`;
579
+ job.finishedAt = new Date().toISOString();
580
+ appendLog('stderr', job.error + '\n');
581
+ emitter.emit('done', buildDonePayload(job));
582
+ scheduleCleanup(job);
583
+ jobs.set(id, job);
584
+ return job;
585
+ }
586
+
587
+ job.child = child;
588
+ child.stdout.on('data', (buf) => appendLog('stdout', buf.toString()));
589
+ child.stderr.on('data', (buf) => appendLog('stderr', buf.toString()));
590
+
591
+ child.on('error', (err) => {
592
+ if (job.status !== 'running') return;
593
+ job.status = 'error';
594
+ job.error = `npm process error: ${err.message}`;
595
+ job.finishedAt = new Date().toISOString();
596
+ appendLog('stderr', job.error + '\n');
597
+ emitter.emit('done', buildDonePayload(job));
598
+ scheduleCleanup(job);
599
+ });
600
+
601
+ child.on('close', (code, signal) => {
602
+ if (job.status !== 'running') return;
603
+ job.exitCode = code ?? null;
604
+ job.finishedAt = new Date().toISOString();
605
+
606
+ if (code !== 0) {
607
+ job.status = 'error';
608
+ job.error = signal
609
+ ? `Install killed by signal ${signal}`
610
+ : `npm exited with code ${code}`;
611
+ emitter.emit('done', buildDonePayload(job));
612
+ scheduleCleanup(job);
613
+ return;
614
+ }
615
+
616
+ // Verify the binary actually landed. If we don't check, a package
617
+ // without a `bin` entry (or a half-extracted tarball) would still
618
+ // read as "success" and the user would be confused when auth
619
+ // status stays red.
620
+ const binName = PACKAGE_BINARIES[pkg] || provider;
621
+ const binaryPath = findInstalledBinary(binName);
622
+ if (!binaryPath) {
623
+ job.status = 'error';
624
+ job.error = `npm exited cleanly but ${binName} was not found in ${CLI_BIN_DIR}`;
625
+ appendLog('stderr', job.error + '\n');
626
+ emitter.emit('done', buildDonePayload(job));
627
+ scheduleCleanup(job);
628
+ return;
629
+ }
630
+
631
+ // Make sure our live server process can resolve the new binary
632
+ // from this moment on, without a restart. primeCliBinPath is
633
+ // idempotent so re-calling after each install is cheap.
634
+ primeCliBinPath();
635
+
636
+ appendLog('meta', `✓ Installed ${binName} → ${binaryPath}\n`);
637
+ job.status = 'done';
638
+ job.binaryPath = binaryPath;
639
+ emitter.emit('done', buildDonePayload(job));
640
+ scheduleCleanup(job);
641
+ });
642
+
643
+ job.timer = setTimeout(() => {
644
+ if (job.status !== 'running') return;
645
+ try { child.kill('SIGKILL'); } catch { /* noop */ }
646
+ job.status = 'error';
647
+ job.error = 'Install timed out after 10 minutes';
648
+ job.finishedAt = new Date().toISOString();
649
+ appendLog('stderr', job.error + '\n');
650
+ emitter.emit('done', buildDonePayload(job));
651
+ scheduleCleanup(job);
652
+ }, HARD_TIMEOUT_MS);
653
+
654
+ jobs.set(id, job);
655
+ return job;
656
+ }
657
+
658
+ function findInstalledBinary(name) {
659
+ const isWindows = process.platform === 'win32';
660
+ const candidates = isWindows
661
+ ? [`${name}.cmd`, `${name}.exe`, name]
662
+ : [name];
663
+ for (const c of candidates) {
664
+ const full = path.join(CLI_BIN_DIR, c);
665
+ if (fs.existsSync(full)) return full;
666
+ }
667
+ return null;
668
+ }
669
+
670
+ function buildDonePayload(job) {
671
+ if (job.status === 'done') {
672
+ return {
673
+ success: true,
674
+ exitCode: job.exitCode,
675
+ binaryPath: job.binaryPath,
676
+ message: `${job.provider} installed. Refreshing auth status…`,
677
+ };
678
+ }
679
+ return {
680
+ success: false,
681
+ exitCode: job.exitCode,
682
+ error: job.error || 'Install failed',
683
+ };
684
+ }
685
+
686
+ function scheduleCleanup(job) {
687
+ if (job.timer) {
688
+ clearTimeout(job.timer);
689
+ job.timer = null;
690
+ }
691
+ setTimeout(() => {
692
+ jobs.delete(job.id);
693
+ }, FINISHED_TTL_MS);
694
+ }
695
+
696
+ export function getInstallJob(id) {
697
+ return jobs.get(id) || null;
698
+ }
699
+
700
+ export function cancelInstallJob(id) {
701
+ const job = jobs.get(id);
702
+ if (!job) return false;
703
+ if (job.status !== 'running') return false;
704
+ try { job.child?.kill(); } catch { /* noop */ }
705
+ job.status = 'error';
706
+ job.error = 'Install cancelled';
707
+ job.finishedAt = new Date().toISOString();
708
+ job.emitter.emit('done', buildDonePayload(job));
709
+ scheduleCleanup(job);
710
+ return true;
711
+ }
712
+
713
+ export function snapshotDonePayload(job) {
714
+ return buildDonePayload(job);
715
+ }