@pixelbyte-software/pixcode 1.51.2 → 1.51.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (331) 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-B9N-gfOQ.css +32 -0
  15. package/dist/assets/{index-EN9ngyxf.js → index-HfGHXhD6.js} +175 -175
  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/database/db.js +14 -2
  52. package/dist-server/server/database/db.js.map +1 -1
  53. package/dist-server/server/index.js +191 -31
  54. package/dist-server/server/index.js.map +1 -1
  55. package/dist-server/server/middleware/auth.js +16 -5
  56. package/dist-server/server/middleware/auth.js.map +1 -1
  57. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
  58. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
  59. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
  60. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
  61. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
  62. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
  63. package/dist-server/server/modules/orchestration/index.js +1 -0
  64. package/dist-server/server/modules/orchestration/index.js.map +1 -1
  65. package/dist-server/server/routes/auth.js +12 -5
  66. package/dist-server/server/routes/auth.js.map +1 -1
  67. package/dist-server/server/routes/commands.js +25 -25
  68. package/dist-server/server/routes/git.js +29 -17
  69. package/dist-server/server/routes/git.js.map +1 -1
  70. package/dist-server/server/routes/live-view.js +46 -46
  71. package/dist-server/server/routes/platformization.js +7 -6
  72. package/dist-server/server/routes/platformization.js.map +1 -1
  73. package/dist-server/server/services/hermes-gateway.js +310 -0
  74. package/dist-server/server/services/hermes-gateway.js.map +1 -1
  75. package/dist-server/server/services/platformization.js +58 -2
  76. package/dist-server/server/services/platformization.js.map +1 -1
  77. package/dist-server/server/services/public-api-manifest.js +59 -51
  78. package/dist-server/server/services/public-api-manifest.js.map +1 -1
  79. package/package.json +222 -222
  80. package/scripts/fix-node-pty.js +67 -67
  81. package/scripts/github/create-v1.38-issues.mjs +351 -351
  82. package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
  83. package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
  84. package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
  85. package/scripts/smoke/changes-panel-layout.mjs +48 -48
  86. package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
  87. package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
  88. package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
  89. package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
  90. package/scripts/smoke/chat-session-state.mjs +19 -19
  91. package/scripts/smoke/code-editor-theme.mjs +55 -55
  92. package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
  93. package/scripts/smoke/command-center-agent-writes.mjs +79 -79
  94. package/scripts/smoke/command-center-non-git.mjs +46 -46
  95. package/scripts/smoke/context-packet.mjs +43 -43
  96. package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
  97. package/scripts/smoke/daemon-entrypoint.mjs +20 -20
  98. package/scripts/smoke/default-landing-routing.mjs +33 -33
  99. package/scripts/smoke/desktop-native-notifications.mjs +30 -30
  100. package/scripts/smoke/desktop-tray-icon.mjs +33 -33
  101. package/scripts/smoke/discord-release-workflow.mjs +24 -24
  102. package/scripts/smoke/git-install-update.mjs +255 -255
  103. package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
  104. package/scripts/smoke/hermes-api-install.mjs +56 -56
  105. package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
  106. package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
  107. package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
  108. package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
  109. package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
  110. package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
  111. package/scripts/smoke/hermes-rest-live.mjs +42 -42
  112. package/scripts/smoke/hermes-roundtrip.mjs +167 -167
  113. package/scripts/smoke/hermes-settings-commands.mjs +349 -346
  114. package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
  115. package/scripts/smoke/live-view-diagnostics.mjs +53 -53
  116. package/scripts/smoke/live-view-environment.mjs +92 -92
  117. package/scripts/smoke/live-view-integration.mjs +450 -450
  118. package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
  119. package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
  120. package/scripts/smoke/model-registry.mjs +36 -36
  121. package/scripts/smoke/multi-project-ui.mjs +45 -45
  122. package/scripts/smoke/multi-worker-slots.mjs +42 -42
  123. package/scripts/smoke/notification-center.mjs +87 -87
  124. package/scripts/smoke/notification-inapp-preference.mjs +23 -23
  125. package/scripts/smoke/notification-taxonomy.mjs +58 -58
  126. package/scripts/smoke/orchestration-api.mjs +172 -172
  127. package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
  128. package/scripts/smoke/orchestration-live-run.mjs +176 -176
  129. package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
  130. package/scripts/smoke/orchestration-model-sync.mjs +30 -30
  131. package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
  132. package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
  133. package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
  134. package/scripts/smoke/permission-policy.mjs +50 -50
  135. package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -167
  136. package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
  137. package/scripts/smoke/provider-rest-api.mjs +124 -124
  138. package/scripts/smoke/provider-selection-status.mjs +52 -52
  139. package/scripts/smoke/run-state-refresh.mjs +52 -52
  140. package/scripts/smoke/runtime-manager.mjs +99 -99
  141. package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
  142. package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
  143. package/scripts/smoke/static-root-routing.mjs +21 -21
  144. package/scripts/smoke/strict-handoff-compact.mjs +60 -60
  145. package/scripts/smoke/taskmaster-config.mjs +24 -24
  146. package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
  147. package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
  148. package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
  149. package/scripts/smoke/telegram-control.mjs +242 -242
  150. package/scripts/smoke/tunnel-persistence.mjs +56 -56
  151. package/scripts/smoke/update-issue-progress.mjs +69 -69
  152. package/scripts/smoke/update-ux.mjs +55 -55
  153. package/scripts/smoke/v138-completion.mjs +132 -132
  154. package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
  155. package/scripts/smoke/v138-diagnostics.mjs +63 -63
  156. package/scripts/smoke/v138-issue-planner.mjs +33 -33
  157. package/scripts/smoke/v143-remote-control.mjs +76 -76
  158. package/scripts/smoke/v144-production-loop.mjs +47 -47
  159. package/scripts/smoke/v145-platformization.mjs +46 -46
  160. package/scripts/smoke/v146-control-room-ui.mjs +150 -150
  161. package/scripts/smoke/version-modal-autoshow.mjs +29 -29
  162. package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
  163. package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
  164. package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
  165. package/scripts/smoke/workflow-templates.mjs +43 -43
  166. package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
  167. package/scripts/update-git-install.mjs +293 -293
  168. package/server/claude-sdk.js +920 -920
  169. package/server/cli.js +1039 -1039
  170. package/server/constants/config.js +4 -4
  171. package/server/cursor-cli.js +344 -344
  172. package/server/daemon/manager.js +563 -563
  173. package/server/daemon-manager.js +964 -964
  174. package/server/database/db.js +908 -895
  175. package/server/database/json-store.js +197 -197
  176. package/server/gemini-cli.js +550 -550
  177. package/server/gemini-response-handler.js +79 -79
  178. package/server/index.js +201 -30
  179. package/server/load-env.js +35 -35
  180. package/server/middleware/auth.js +171 -156
  181. package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
  182. package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
  183. package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
  184. package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
  185. package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
  186. package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
  187. package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
  188. package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
  189. package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
  190. package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
  191. package/server/modules/orchestration/a2a/agent-card.ts +55 -55
  192. package/server/modules/orchestration/a2a/routes.ts +590 -590
  193. package/server/modules/orchestration/a2a/task-store.ts +178 -178
  194. package/server/modules/orchestration/a2a/types.ts +126 -126
  195. package/server/modules/orchestration/a2a/validator.ts +113 -113
  196. package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
  197. package/server/modules/orchestration/index.ts +101 -100
  198. package/server/modules/orchestration/preview/port-watcher.ts +112 -112
  199. package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
  200. package/server/modules/orchestration/preview/types.ts +19 -19
  201. package/server/modules/orchestration/security/permission-policy.ts +401 -401
  202. package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
  203. package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
  204. package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
  205. package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
  206. package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
  207. package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
  208. package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
  209. package/server/modules/orchestration/workflows/context-packet.ts +186 -186
  210. package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
  211. package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
  212. package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
  213. package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
  214. package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
  215. package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
  216. package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
  217. package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
  218. package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
  219. package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
  220. package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
  221. package/server/modules/orchestration/workspace/path-safety.ts +55 -55
  222. package/server/modules/orchestration/workspace/types.ts +52 -52
  223. package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
  224. package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
  225. package/server/modules/providers/index.ts +2 -2
  226. package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
  227. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  228. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  229. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  230. package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
  231. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  232. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  233. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  234. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
  235. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  236. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  237. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  238. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
  239. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  240. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  241. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  242. package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
  243. package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
  244. package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
  245. package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
  246. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
  247. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
  248. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
  249. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
  250. package/server/modules/providers/provider.registry.ts +40 -40
  251. package/server/modules/providers/provider.routes.ts +944 -944
  252. package/server/modules/providers/services/mcp.service.ts +86 -86
  253. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  254. package/server/modules/providers/services/sessions.service.ts +45 -45
  255. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  256. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  257. package/server/modules/providers/shared/provider-configs.ts +142 -142
  258. package/server/modules/providers/tests/mcp.test.ts +293 -293
  259. package/server/openai-codex.js +462 -462
  260. package/server/opencode-cli.js +491 -491
  261. package/server/opencode-response-handler.js +111 -111
  262. package/server/projects.js +3008 -3008
  263. package/server/qwen-code-cli.js +410 -410
  264. package/server/qwen-response-handler.js +73 -73
  265. package/server/routes/agent.js +1435 -1435
  266. package/server/routes/auth.js +154 -146
  267. package/server/routes/codex.js +20 -20
  268. package/server/routes/commands.js +570 -570
  269. package/server/routes/cursor.js +61 -61
  270. package/server/routes/diagnostics.js +41 -41
  271. package/server/routes/gemini.js +25 -25
  272. package/server/routes/git.js +1650 -1635
  273. package/server/routes/live-view.js +411 -411
  274. package/server/routes/mcp-utils.js +13 -13
  275. package/server/routes/messages.js +62 -62
  276. package/server/routes/network.js +125 -125
  277. package/server/routes/platformization.js +198 -197
  278. package/server/routes/plugins.js +320 -320
  279. package/server/routes/production-agent-loop.js +90 -90
  280. package/server/routes/projects.js +917 -917
  281. package/server/routes/public-api.js +34 -34
  282. package/server/routes/qwen.js +27 -27
  283. package/server/routes/remote.js +55 -55
  284. package/server/routes/settings.js +321 -321
  285. package/server/routes/telegram.js +140 -140
  286. package/server/routes/user.js +125 -125
  287. package/server/routes/webhooks.js +63 -63
  288. package/server/services/control-room.js +102 -102
  289. package/server/services/diagnostics.js +165 -165
  290. package/server/services/external-access.js +375 -375
  291. package/server/services/hermes-gateway.js +1562 -1247
  292. package/server/services/hermes-install-jobs.js +729 -729
  293. package/server/services/install-jobs.js +715 -715
  294. package/server/services/live-view.js +956 -956
  295. package/server/services/managed-runtimes.js +493 -493
  296. package/server/services/model-registry.js +144 -144
  297. package/server/services/notification-orchestrator.js +365 -365
  298. package/server/services/notification-taxonomy.js +204 -204
  299. package/server/services/platformization.js +844 -779
  300. package/server/services/production-agent-loop.js +248 -248
  301. package/server/services/provider-cli-versions.js +149 -149
  302. package/server/services/provider-credentials.js +189 -189
  303. package/server/services/provider-models.js +396 -396
  304. package/server/services/public-api-manifest.js +190 -182
  305. package/server/services/remote-connection.js +127 -127
  306. package/server/services/runtime-manager.js +323 -323
  307. package/server/services/startup-update.js +234 -234
  308. package/server/services/telegram/bot.js +331 -331
  309. package/server/services/telegram/control-center.js +979 -979
  310. package/server/services/telegram/telegram-http-client.js +151 -151
  311. package/server/services/telegram/translations.js +340 -340
  312. package/server/services/vapid-keys.js +36 -36
  313. package/server/services/webhooks.js +216 -216
  314. package/server/sessionManager.js +225 -225
  315. package/server/shared/interfaces.ts +54 -54
  316. package/server/shared/types.ts +172 -172
  317. package/server/shared/utils.ts +193 -193
  318. package/server/tsconfig.json +36 -36
  319. package/server/utils/colors.js +21 -21
  320. package/server/utils/commandParser.js +305 -305
  321. package/server/utils/frontmatter.js +18 -18
  322. package/server/utils/gitConfig.js +34 -34
  323. package/server/utils/plugin-loader.js +457 -457
  324. package/server/utils/plugin-process-manager.js +185 -185
  325. package/server/utils/port-access.js +209 -209
  326. package/server/utils/runtime-paths.js +37 -37
  327. package/server/utils/url-detection.js +71 -71
  328. package/server/vite-daemon.js +79 -79
  329. package/shared/modelConstants.js +161 -161
  330. package/shared/networkHosts.js +22 -22
  331. 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
+ }