@pixelbyte-software/pixcode 1.51.1 → 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.
- package/CODE_OF_CONDUCT.md +41 -41
- package/CONTRIBUTING.md +155 -155
- package/LICENSE +718 -718
- package/README.de.md +169 -169
- package/README.ja.md +167 -167
- package/README.ko.md +167 -167
- package/README.md +419 -419
- package/README.ru.md +169 -169
- package/README.tr.md +298 -298
- package/README.zh-CN.md +167 -167
- package/SECURITY.md +46 -46
- package/dist/api-automation.html +110 -110
- package/dist/api-docs.html +548 -548
- package/dist/assets/{index-DARIZgoD.js → index-17CwxHSZ.js} +185 -185
- package/dist/assets/index-B9N-gfOQ.css +32 -0
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/docs.html +308 -308
- package/dist/favicon.svg +8 -8
- package/dist/features.html +133 -133
- package/dist/generate-icons.js +48 -48
- package/dist/humans.txt +15 -15
- package/dist/icons/codex-white.svg +3 -3
- package/dist/icons/codex.svg +3 -3
- package/dist/icons/cursor-white.svg +11 -11
- package/dist/icons/icon-128x128.svg +9 -9
- package/dist/icons/icon-144x144.svg +9 -9
- package/dist/icons/icon-152x152.svg +9 -9
- package/dist/icons/icon-192x192.svg +9 -9
- package/dist/icons/icon-384x384.svg +9 -9
- package/dist/icons/icon-512x512.svg +9 -9
- package/dist/icons/icon-72x72.svg +9 -9
- package/dist/icons/icon-96x96.svg +9 -9
- package/dist/icons/icon-template.svg +9 -9
- package/dist/icons/qwen-logo.svg +14 -14
- package/dist/index.html +59 -59
- package/dist/landing.html +268 -268
- package/dist/llms-full.txt +119 -119
- package/dist/llms.txt +53 -53
- package/dist/logo.svg +12 -12
- package/dist/manifest.json +60 -60
- package/dist/openapi.yaml +1696 -1696
- package/dist/orchestration.html +125 -125
- package/dist/robots.txt +4 -4
- package/dist/site.css +692 -692
- package/dist/sitemap.xml +51 -51
- package/dist/sw.js +132 -132
- package/dist-server/server/cli.js +96 -96
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +64 -64
- package/dist-server/server/index.js +125 -4
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
- package/dist-server/server/modules/orchestration/index.js +1 -0
- package/dist-server/server/modules/orchestration/index.js.map +1 -1
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +17 -17
- package/dist-server/server/routes/live-view.js +46 -46
- package/dist-server/server/services/hermes-gateway.js +310 -0
- package/dist-server/server/services/hermes-gateway.js.map +1 -1
- package/dist-server/server/services/public-api-manifest.js +59 -51
- package/dist-server/server/services/public-api-manifest.js.map +1 -1
- package/package.json +222 -222
- package/scripts/fix-node-pty.js +67 -67
- package/scripts/github/create-v1.38-issues.mjs +351 -351
- package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
- package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
- package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
- package/scripts/smoke/changes-panel-layout.mjs +48 -48
- package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
- package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
- package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
- package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
- package/scripts/smoke/chat-session-state.mjs +19 -19
- package/scripts/smoke/code-editor-theme.mjs +55 -55
- package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
- package/scripts/smoke/command-center-agent-writes.mjs +79 -79
- package/scripts/smoke/command-center-non-git.mjs +46 -46
- package/scripts/smoke/context-packet.mjs +43 -43
- package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
- package/scripts/smoke/daemon-entrypoint.mjs +20 -20
- package/scripts/smoke/default-landing-routing.mjs +33 -33
- package/scripts/smoke/desktop-native-notifications.mjs +30 -30
- package/scripts/smoke/desktop-tray-icon.mjs +33 -33
- package/scripts/smoke/discord-release-workflow.mjs +24 -24
- package/scripts/smoke/git-install-update.mjs +255 -255
- package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
- package/scripts/smoke/hermes-api-install.mjs +56 -56
- package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
- package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
- package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
- package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
- package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
- package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
- package/scripts/smoke/hermes-rest-live.mjs +42 -42
- package/scripts/smoke/hermes-roundtrip.mjs +167 -167
- package/scripts/smoke/hermes-settings-commands.mjs +349 -346
- package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
- package/scripts/smoke/live-view-diagnostics.mjs +53 -53
- package/scripts/smoke/live-view-environment.mjs +92 -92
- package/scripts/smoke/live-view-integration.mjs +450 -450
- package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
- package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
- package/scripts/smoke/model-registry.mjs +36 -36
- package/scripts/smoke/multi-project-ui.mjs +45 -45
- package/scripts/smoke/multi-worker-slots.mjs +42 -42
- package/scripts/smoke/notification-center.mjs +87 -87
- package/scripts/smoke/notification-inapp-preference.mjs +23 -23
- package/scripts/smoke/notification-taxonomy.mjs +58 -58
- package/scripts/smoke/orchestration-api.mjs +172 -172
- package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
- package/scripts/smoke/orchestration-live-run.mjs +176 -176
- package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
- package/scripts/smoke/orchestration-model-sync.mjs +30 -30
- package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
- package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
- package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
- package/scripts/smoke/permission-policy.mjs +50 -50
- package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -164
- package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
- package/scripts/smoke/provider-rest-api.mjs +124 -124
- package/scripts/smoke/provider-selection-status.mjs +52 -52
- package/scripts/smoke/run-state-refresh.mjs +52 -52
- package/scripts/smoke/runtime-manager.mjs +99 -99
- package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
- package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
- package/scripts/smoke/static-root-routing.mjs +21 -21
- package/scripts/smoke/strict-handoff-compact.mjs +60 -60
- package/scripts/smoke/taskmaster-config.mjs +24 -24
- package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
- package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
- package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
- package/scripts/smoke/telegram-control.mjs +242 -242
- package/scripts/smoke/tunnel-persistence.mjs +56 -56
- package/scripts/smoke/update-issue-progress.mjs +69 -69
- package/scripts/smoke/update-ux.mjs +55 -55
- package/scripts/smoke/v138-completion.mjs +132 -132
- package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
- package/scripts/smoke/v138-diagnostics.mjs +63 -63
- package/scripts/smoke/v138-issue-planner.mjs +33 -33
- package/scripts/smoke/v143-remote-control.mjs +76 -76
- package/scripts/smoke/v144-production-loop.mjs +47 -47
- package/scripts/smoke/v145-platformization.mjs +46 -46
- package/scripts/smoke/v146-control-room-ui.mjs +150 -150
- package/scripts/smoke/version-modal-autoshow.mjs +29 -29
- package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
- package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
- package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
- package/scripts/smoke/workflow-templates.mjs +43 -43
- package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
- package/scripts/update-git-install.mjs +293 -293
- package/server/claude-sdk.js +920 -920
- package/server/cli.js +1039 -1039
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +344 -344
- package/server/daemon/manager.js +563 -563
- package/server/daemon-manager.js +964 -964
- package/server/database/db.js +921 -921
- package/server/database/json-store.js +197 -197
- package/server/gemini-cli.js +550 -550
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +131 -3
- package/server/load-env.js +35 -35
- package/server/middleware/auth.js +175 -175
- package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
- package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
- package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
- package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
- package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
- package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
- package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/agent-card.ts +55 -55
- package/server/modules/orchestration/a2a/routes.ts +590 -590
- package/server/modules/orchestration/a2a/task-store.ts +178 -178
- package/server/modules/orchestration/a2a/types.ts +126 -126
- package/server/modules/orchestration/a2a/validator.ts +113 -113
- package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
- package/server/modules/orchestration/index.ts +101 -100
- package/server/modules/orchestration/preview/port-watcher.ts +112 -112
- package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
- package/server/modules/orchestration/preview/types.ts +19 -19
- package/server/modules/orchestration/security/permission-policy.ts +401 -401
- package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
- package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
- package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
- package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
- package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
- package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
- package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
- package/server/modules/orchestration/workflows/context-packet.ts +186 -186
- package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
- package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
- package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
- package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
- package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
- package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
- package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
- package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
- package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
- package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
- package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
- package/server/modules/orchestration/workspace/path-safety.ts +55 -55
- package/server/modules/orchestration/workspace/types.ts +52 -52
- package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
- package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
- package/server/modules/providers/index.ts +2 -2
- package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
- package/server/modules/providers/list/claude/claude.provider.ts +15 -15
- package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
- package/server/modules/providers/list/codex/codex.provider.ts +15 -15
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
- package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
- package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
- package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
- package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
- package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
- package/server/modules/providers/provider.registry.ts +40 -40
- package/server/modules/providers/provider.routes.ts +944 -944
- package/server/modules/providers/services/mcp.service.ts +86 -86
- package/server/modules/providers/services/provider-auth.service.ts +26 -26
- package/server/modules/providers/services/sessions.service.ts +45 -45
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
- package/server/modules/providers/shared/provider-configs.ts +142 -142
- package/server/modules/providers/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +462 -462
- package/server/opencode-cli.js +491 -491
- package/server/opencode-response-handler.js +111 -111
- package/server/projects.js +3008 -3008
- package/server/qwen-code-cli.js +410 -410
- package/server/qwen-response-handler.js +73 -73
- package/server/routes/agent.js +1435 -1435
- package/server/routes/auth.js +159 -159
- package/server/routes/codex.js +20 -20
- package/server/routes/commands.js +570 -570
- package/server/routes/cursor.js +61 -61
- package/server/routes/diagnostics.js +41 -41
- package/server/routes/gemini.js +25 -25
- package/server/routes/git.js +1650 -1650
- package/server/routes/live-view.js +411 -411
- package/server/routes/mcp-utils.js +13 -13
- package/server/routes/messages.js +62 -62
- package/server/routes/network.js +125 -125
- package/server/routes/platformization.js +212 -212
- package/server/routes/plugins.js +320 -320
- package/server/routes/production-agent-loop.js +90 -90
- package/server/routes/projects.js +917 -917
- package/server/routes/public-api.js +34 -34
- package/server/routes/qwen.js +27 -27
- package/server/routes/remote.js +55 -55
- package/server/routes/settings.js +321 -321
- package/server/routes/telegram.js +140 -140
- package/server/routes/user.js +125 -125
- package/server/routes/webhooks.js +63 -63
- package/server/services/control-room.js +102 -102
- package/server/services/diagnostics.js +165 -165
- package/server/services/external-access.js +375 -375
- package/server/services/hermes-gateway.js +1562 -1247
- package/server/services/hermes-install-jobs.js +729 -729
- package/server/services/install-jobs.js +715 -715
- package/server/services/live-view.js +956 -956
- package/server/services/managed-runtimes.js +493 -493
- package/server/services/model-registry.js +144 -144
- package/server/services/notification-orchestrator.js +365 -365
- package/server/services/notification-taxonomy.js +204 -204
- package/server/services/platformization.js +815 -815
- package/server/services/production-agent-loop.js +248 -248
- package/server/services/provider-cli-versions.js +149 -149
- package/server/services/provider-credentials.js +189 -189
- package/server/services/provider-models.js +396 -396
- package/server/services/public-api-manifest.js +190 -182
- package/server/services/remote-connection.js +127 -127
- package/server/services/runtime-manager.js +323 -323
- package/server/services/startup-update.js +234 -234
- package/server/services/telegram/bot.js +331 -331
- package/server/services/telegram/control-center.js +979 -979
- package/server/services/telegram/telegram-http-client.js +151 -151
- package/server/services/telegram/translations.js +340 -340
- package/server/services/vapid-keys.js +36 -36
- package/server/services/webhooks.js +216 -216
- package/server/sessionManager.js +225 -225
- package/server/shared/interfaces.ts +54 -54
- package/server/shared/types.ts +172 -172
- package/server/shared/utils.ts +193 -193
- package/server/tsconfig.json +36 -36
- package/server/utils/colors.js +21 -21
- package/server/utils/commandParser.js +305 -305
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +185 -185
- package/server/utils/port-access.js +209 -209
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +79 -79
- package/shared/modelConstants.js +161 -161
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-DMz0zv6T.css +0 -32
package/server/qwen-code-cli.js
CHANGED
|
@@ -1,410 +1,410 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Qwen Code CLI adapter.
|
|
3
|
-
*
|
|
4
|
-
* Qwen Code (https://github.com/QwenLM/qwen-code) is Alibaba's fork of Google's
|
|
5
|
-
* Gemini CLI. The command-line surface, stream-json output, session layout
|
|
6
|
-
* (~/.qwen/tmp/<project>/...), and approval flags all mirror Gemini's. This
|
|
7
|
-
* adapter is therefore a structural copy of gemini-cli.js — kept as its own
|
|
8
|
-
* file so future Qwen-specific divergence (different auth flow, different
|
|
9
|
-
* model list) has a clean place to land without touching Gemini's code path.
|
|
10
|
-
*/
|
|
11
|
-
import { spawn } from 'child_process';
|
|
12
|
-
import { promises as fs } from 'fs';
|
|
13
|
-
import path from 'path';
|
|
14
|
-
import os from 'os';
|
|
15
|
-
|
|
16
|
-
import crossSpawn from 'cross-spawn';
|
|
17
|
-
|
|
18
|
-
import sessionManager from './sessionManager.js';
|
|
19
|
-
import QwenResponseHandler from './qwen-response-handler.js';
|
|
20
|
-
import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
|
|
21
|
-
import { buildSpawnEnv } from './services/provider-credentials.js';
|
|
22
|
-
import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
|
|
23
|
-
import { createNormalizedMessage } from './shared/utils.js';
|
|
24
|
-
|
|
25
|
-
// Use cross-spawn on Windows so `qwen.cmd` resolves correctly.
|
|
26
|
-
const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
|
|
27
|
-
|
|
28
|
-
const activeQwenProcesses = new Map();
|
|
29
|
-
const DEFAULT_CLI_IDLE_TIMEOUT_MS = 600000;
|
|
30
|
-
|
|
31
|
-
function readCliIdleTimeoutMs() {
|
|
32
|
-
const configured = Number.parseInt(process.env.PIXCODE_CLI_IDLE_TIMEOUT_MS || '', 10);
|
|
33
|
-
return Number.isFinite(configured) && configured >= 0 ? configured : DEFAULT_CLI_IDLE_TIMEOUT_MS;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
async function spawnQwen(command, options = {}, ws) {
|
|
37
|
-
const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
|
|
38
|
-
let capturedSessionId = sessionId;
|
|
39
|
-
let sessionCreatedSent = false;
|
|
40
|
-
let assistantBlocks = [];
|
|
41
|
-
|
|
42
|
-
const settings = toolsSettings || { allowedTools: [], disallowedTools: [], skipPermissions: false };
|
|
43
|
-
|
|
44
|
-
const args = [];
|
|
45
|
-
if (command && command.trim()) {
|
|
46
|
-
args.push('--prompt', command);
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
if (sessionId) {
|
|
50
|
-
const session = sessionManager.getSession(sessionId);
|
|
51
|
-
if (session && session.cliSessionId) {
|
|
52
|
-
args.push('--resume', session.cliSessionId);
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
|
|
57
|
-
const workingDir = cleanPath;
|
|
58
|
-
|
|
59
|
-
const tempImagePaths = [];
|
|
60
|
-
let tempDir = null;
|
|
61
|
-
if (images && images.length > 0) {
|
|
62
|
-
try {
|
|
63
|
-
tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
|
|
64
|
-
await fs.mkdir(tempDir, { recursive: true });
|
|
65
|
-
|
|
66
|
-
for (const [index, image] of images.entries()) {
|
|
67
|
-
const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
|
|
68
|
-
if (!matches) continue;
|
|
69
|
-
const [, mimeType, base64Data] = matches;
|
|
70
|
-
const extension = mimeType.split('/')[1] || 'png';
|
|
71
|
-
const filename = `image_${index}.${extension}`;
|
|
72
|
-
const filepath = path.join(tempDir, filename);
|
|
73
|
-
await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
|
|
74
|
-
tempImagePaths.push(filepath);
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
if (tempImagePaths.length > 0 && command && command.trim()) {
|
|
78
|
-
const imageNote = `\n\n[Images given: ${tempImagePaths.length} images are located at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
|
|
79
|
-
const modifiedCommand = command + imageNote;
|
|
80
|
-
const promptIndex = args.indexOf('--prompt');
|
|
81
|
-
if (promptIndex !== -1 && args[promptIndex + 1] === command) {
|
|
82
|
-
args[promptIndex + 1] = modifiedCommand;
|
|
83
|
-
} else if (promptIndex !== -1) {
|
|
84
|
-
args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
} catch (error) {
|
|
88
|
-
console.error('Error processing images for Qwen Code:', error);
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
if (options.debug) {
|
|
93
|
-
args.push('--debug');
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
// Qwen's MCP config mirrors Gemini's — per-user settings.json plus optional
|
|
97
|
-
// project override. Pixcode writes to the user-scope file via the provider
|
|
98
|
-
// MCP module, and Qwen Code auto-loads it, so we don't pass --mcp-config
|
|
99
|
-
// explicitly. Left intentionally minimal to avoid double-loading.
|
|
100
|
-
|
|
101
|
-
const modelToUse = options.model || 'qwen3-coder-plus';
|
|
102
|
-
args.push('--model', modelToUse);
|
|
103
|
-
args.push('--output-format', 'stream-json');
|
|
104
|
-
|
|
105
|
-
if (
|
|
106
|
-
settings.skipPermissions ||
|
|
107
|
-
options.skipPermissions ||
|
|
108
|
-
permissionMode === 'yolo' ||
|
|
109
|
-
permissionMode === 'bypassPermissions' ||
|
|
110
|
-
permissionMode === 'acceptEdits'
|
|
111
|
-
) {
|
|
112
|
-
args.push('--yolo');
|
|
113
|
-
} else if (permissionMode === 'auto_edit') {
|
|
114
|
-
args.push('--approval-mode', 'auto_edit');
|
|
115
|
-
} else if (permissionMode === 'plan') {
|
|
116
|
-
args.push('--approval-mode', 'plan');
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
if (settings.allowedTools && settings.allowedTools.length > 0) {
|
|
120
|
-
args.push('--allowed-tools', settings.allowedTools.join(','));
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
const qwenPath = process.env.QWEN_PATH || 'qwen';
|
|
124
|
-
console.log('Spawning Qwen Code CLI:', qwenPath, args.join(' '));
|
|
125
|
-
console.log('Working directory:', workingDir);
|
|
126
|
-
|
|
127
|
-
let spawnCmd = qwenPath;
|
|
128
|
-
let spawnArgs = args;
|
|
129
|
-
|
|
130
|
-
if (os.platform() !== 'win32') {
|
|
131
|
-
spawnCmd = 'sh';
|
|
132
|
-
spawnArgs = ['-c', 'exec "$0" "$@"', qwenPath, ...args];
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
// Credentials stored in ~/.pixcode/provider-credentials.json take
|
|
136
|
-
// precedence over the host shell env, so an API key saved via the
|
|
137
|
-
// Pixcode UI reaches the Qwen subprocess even when the user never
|
|
138
|
-
// exported it in their login shell.
|
|
139
|
-
const spawnEnv = await buildSpawnEnv('qwen');
|
|
140
|
-
|
|
141
|
-
return new Promise((resolve, reject) => {
|
|
142
|
-
const qwenProcess = spawnFunction(spawnCmd, spawnArgs, {
|
|
143
|
-
cwd: workingDir,
|
|
144
|
-
stdio: ['pipe', 'pipe', 'pipe'],
|
|
145
|
-
env: spawnEnv,
|
|
146
|
-
});
|
|
147
|
-
|
|
148
|
-
let terminalNotificationSent = false;
|
|
149
|
-
let terminalFailureReason = null;
|
|
150
|
-
|
|
151
|
-
const notifyTerminalState = ({ code = null, error = null } = {}) => {
|
|
152
|
-
if (terminalNotificationSent) return;
|
|
153
|
-
terminalNotificationSent = true;
|
|
154
|
-
|
|
155
|
-
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
156
|
-
if (code === 0 && !error) {
|
|
157
|
-
notifyRunStopped({
|
|
158
|
-
userId: ws?.userId || null,
|
|
159
|
-
provider: 'qwen',
|
|
160
|
-
sessionId: finalSessionId,
|
|
161
|
-
sessionName: sessionSummary,
|
|
162
|
-
stopReason: 'completed',
|
|
163
|
-
});
|
|
164
|
-
return;
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
notifyRunFailed({
|
|
168
|
-
userId: ws?.userId || null,
|
|
169
|
-
provider: 'qwen',
|
|
170
|
-
sessionId: finalSessionId,
|
|
171
|
-
sessionName: sessionSummary,
|
|
172
|
-
error: error || terminalFailureReason || `Qwen Code CLI exited with code ${code}`,
|
|
173
|
-
});
|
|
174
|
-
};
|
|
175
|
-
|
|
176
|
-
qwenProcess.tempImagePaths = tempImagePaths;
|
|
177
|
-
qwenProcess.tempDir = tempDir;
|
|
178
|
-
|
|
179
|
-
const processKey = capturedSessionId || sessionId || Date.now().toString();
|
|
180
|
-
activeQwenProcesses.set(processKey, qwenProcess);
|
|
181
|
-
qwenProcess.sessionId = processKey;
|
|
182
|
-
|
|
183
|
-
qwenProcess.stdin.end();
|
|
184
|
-
|
|
185
|
-
const timeoutMs = readCliIdleTimeoutMs();
|
|
186
|
-
let timeout;
|
|
187
|
-
const startTimeout = () => {
|
|
188
|
-
if (timeout) clearTimeout(timeout);
|
|
189
|
-
if (timeoutMs === 0) return;
|
|
190
|
-
timeout = setTimeout(() => {
|
|
191
|
-
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
|
|
192
|
-
terminalFailureReason = `Qwen Code CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
|
|
193
|
-
ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'qwen' }));
|
|
194
|
-
try { qwenProcess.kill('SIGTERM'); } catch { /* noop */ }
|
|
195
|
-
}, timeoutMs);
|
|
196
|
-
};
|
|
197
|
-
startTimeout();
|
|
198
|
-
|
|
199
|
-
if (command && capturedSessionId) {
|
|
200
|
-
sessionManager.addMessage(capturedSessionId, 'user', command);
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
let responseHandler;
|
|
204
|
-
if (ws) {
|
|
205
|
-
responseHandler = new QwenResponseHandler(ws, {
|
|
206
|
-
onContentFragment: (content) => {
|
|
207
|
-
if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
|
|
208
|
-
assistantBlocks[assistantBlocks.length - 1].text += content;
|
|
209
|
-
} else {
|
|
210
|
-
assistantBlocks.push({ type: 'text', text: content });
|
|
211
|
-
}
|
|
212
|
-
},
|
|
213
|
-
onToolUse: (event) => {
|
|
214
|
-
assistantBlocks.push({
|
|
215
|
-
type: 'tool_use',
|
|
216
|
-
id: event.tool_id,
|
|
217
|
-
name: event.tool_name,
|
|
218
|
-
input: event.parameters,
|
|
219
|
-
});
|
|
220
|
-
},
|
|
221
|
-
onToolResult: (event) => {
|
|
222
|
-
if (capturedSessionId) {
|
|
223
|
-
if (assistantBlocks.length > 0) {
|
|
224
|
-
sessionManager.addMessage(capturedSessionId, 'assistant', [...assistantBlocks]);
|
|
225
|
-
assistantBlocks = [];
|
|
226
|
-
}
|
|
227
|
-
sessionManager.addMessage(capturedSessionId, 'user', [{
|
|
228
|
-
type: 'tool_result',
|
|
229
|
-
tool_use_id: event.tool_id,
|
|
230
|
-
content: event.output === undefined ? null : event.output,
|
|
231
|
-
is_error: event.status === 'error',
|
|
232
|
-
}]);
|
|
233
|
-
}
|
|
234
|
-
},
|
|
235
|
-
onInit: (event) => {
|
|
236
|
-
if (capturedSessionId) {
|
|
237
|
-
const sess = sessionManager.getSession(capturedSessionId);
|
|
238
|
-
if (sess && !sess.cliSessionId) {
|
|
239
|
-
sess.cliSessionId = event.session_id;
|
|
240
|
-
sessionManager.saveSession(capturedSessionId);
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
},
|
|
244
|
-
});
|
|
245
|
-
}
|
|
246
|
-
|
|
247
|
-
qwenProcess.stdout.on('data', (data) => {
|
|
248
|
-
const rawOutput = data.toString();
|
|
249
|
-
startTimeout();
|
|
250
|
-
|
|
251
|
-
if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
|
|
252
|
-
capturedSessionId = `qwen_${Date.now()}`;
|
|
253
|
-
sessionCreatedSent = true;
|
|
254
|
-
|
|
255
|
-
sessionManager.createSession(capturedSessionId, cwd || process.cwd());
|
|
256
|
-
if (command) {
|
|
257
|
-
sessionManager.addMessage(capturedSessionId, 'user', command);
|
|
258
|
-
}
|
|
259
|
-
if (processKey !== capturedSessionId) {
|
|
260
|
-
activeQwenProcesses.delete(processKey);
|
|
261
|
-
activeQwenProcesses.set(capturedSessionId, qwenProcess);
|
|
262
|
-
}
|
|
263
|
-
|
|
264
|
-
ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
|
|
265
|
-
ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'qwen' }));
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
if (responseHandler) {
|
|
269
|
-
responseHandler.processData(rawOutput);
|
|
270
|
-
} else if (rawOutput) {
|
|
271
|
-
if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
|
|
272
|
-
assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
|
|
273
|
-
} else {
|
|
274
|
-
assistantBlocks.push({ type: 'text', text: rawOutput });
|
|
275
|
-
}
|
|
276
|
-
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
|
|
277
|
-
ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'qwen' }));
|
|
278
|
-
}
|
|
279
|
-
});
|
|
280
|
-
|
|
281
|
-
qwenProcess.stderr.on('data', (data) => {
|
|
282
|
-
const errorMsg = data.toString();
|
|
283
|
-
startTimeout();
|
|
284
|
-
if (errorMsg.includes('[DEP0040]') ||
|
|
285
|
-
errorMsg.includes('DeprecationWarning') ||
|
|
286
|
-
errorMsg.includes('--trace-deprecation') ||
|
|
287
|
-
errorMsg.includes('Loaded cached credentials')) {
|
|
288
|
-
return;
|
|
289
|
-
}
|
|
290
|
-
|
|
291
|
-
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
|
|
292
|
-
ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'qwen' }));
|
|
293
|
-
});
|
|
294
|
-
|
|
295
|
-
qwenProcess.on('close', async (code) => {
|
|
296
|
-
clearTimeout(timeout);
|
|
297
|
-
|
|
298
|
-
if (responseHandler) {
|
|
299
|
-
responseHandler.forceFlush();
|
|
300
|
-
responseHandler.destroy();
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
304
|
-
activeQwenProcesses.delete(finalSessionId);
|
|
305
|
-
|
|
306
|
-
if (finalSessionId && assistantBlocks.length > 0) {
|
|
307
|
-
sessionManager.addMessage(finalSessionId, 'assistant', assistantBlocks);
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'qwen' }));
|
|
311
|
-
|
|
312
|
-
if (qwenProcess.tempImagePaths && qwenProcess.tempImagePaths.length > 0) {
|
|
313
|
-
for (const imagePath of qwenProcess.tempImagePaths) {
|
|
314
|
-
await fs.unlink(imagePath).catch(() => { /* noop */ });
|
|
315
|
-
}
|
|
316
|
-
if (qwenProcess.tempDir) {
|
|
317
|
-
await fs.rm(qwenProcess.tempDir, { recursive: true, force: true }).catch(() => { /* noop */ });
|
|
318
|
-
}
|
|
319
|
-
}
|
|
320
|
-
|
|
321
|
-
if (code === 0) {
|
|
322
|
-
notifyTerminalState({ code });
|
|
323
|
-
resolve();
|
|
324
|
-
} else {
|
|
325
|
-
if (code === 127) {
|
|
326
|
-
const installed = await providerAuthService.isProviderInstalled('qwen');
|
|
327
|
-
if (!installed) {
|
|
328
|
-
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
|
329
|
-
ws.send(createNormalizedMessage({
|
|
330
|
-
kind: 'error',
|
|
331
|
-
content: 'Qwen Code CLI is not installed. Install it first: npm install -g @qwen-code/qwen-code',
|
|
332
|
-
sessionId: socketSessionId,
|
|
333
|
-
provider: 'qwen',
|
|
334
|
-
}));
|
|
335
|
-
}
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
notifyTerminalState({
|
|
339
|
-
code,
|
|
340
|
-
error: code === null ? 'Qwen Code CLI process was terminated or timed out' : null,
|
|
341
|
-
});
|
|
342
|
-
reject(new Error(code === null ? 'Qwen Code CLI process was terminated or timed out' : `Qwen Code CLI exited with code ${code}`));
|
|
343
|
-
}
|
|
344
|
-
});
|
|
345
|
-
|
|
346
|
-
qwenProcess.on('error', async (error) => {
|
|
347
|
-
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
348
|
-
activeQwenProcesses.delete(finalSessionId);
|
|
349
|
-
|
|
350
|
-
const installed = await providerAuthService.isProviderInstalled('qwen');
|
|
351
|
-
const errorContent = !installed
|
|
352
|
-
? 'Qwen Code CLI is not installed. Install it first: npm install -g @qwen-code/qwen-code'
|
|
353
|
-
: (error?.message || String(error));
|
|
354
|
-
|
|
355
|
-
const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
|
356
|
-
ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'qwen' }));
|
|
357
|
-
// Always emit `complete` so the UI's "Processing..." state clears
|
|
358
|
-
// even when spawn fails (ENOENT, EACCES) and `close` never fires.
|
|
359
|
-
ws.send(createNormalizedMessage({ kind: 'complete', exitCode: 1, isNewSession: !sessionId && !!command, sessionId: errorSessionId, provider: 'qwen' }));
|
|
360
|
-
notifyTerminalState({ error });
|
|
361
|
-
|
|
362
|
-
reject(error);
|
|
363
|
-
});
|
|
364
|
-
});
|
|
365
|
-
}
|
|
366
|
-
|
|
367
|
-
function abortQwenSession(sessionId) {
|
|
368
|
-
let qwenProc = activeQwenProcesses.get(sessionId);
|
|
369
|
-
let processKey = sessionId;
|
|
370
|
-
|
|
371
|
-
if (!qwenProc) {
|
|
372
|
-
for (const [key, proc] of activeQwenProcesses.entries()) {
|
|
373
|
-
if (proc.sessionId === sessionId) {
|
|
374
|
-
qwenProc = proc;
|
|
375
|
-
processKey = key;
|
|
376
|
-
break;
|
|
377
|
-
}
|
|
378
|
-
}
|
|
379
|
-
}
|
|
380
|
-
|
|
381
|
-
if (qwenProc) {
|
|
382
|
-
try {
|
|
383
|
-
qwenProc.kill('SIGTERM');
|
|
384
|
-
setTimeout(() => {
|
|
385
|
-
if (activeQwenProcesses.has(processKey)) {
|
|
386
|
-
try { qwenProc.kill('SIGKILL'); } catch { /* noop */ }
|
|
387
|
-
}
|
|
388
|
-
}, 2000);
|
|
389
|
-
return true;
|
|
390
|
-
} catch {
|
|
391
|
-
return false;
|
|
392
|
-
}
|
|
393
|
-
}
|
|
394
|
-
return false;
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
function isQwenSessionActive(sessionId) {
|
|
398
|
-
return activeQwenProcesses.has(sessionId);
|
|
399
|
-
}
|
|
400
|
-
|
|
401
|
-
function getActiveQwenSessions() {
|
|
402
|
-
return Array.from(activeQwenProcesses.keys());
|
|
403
|
-
}
|
|
404
|
-
|
|
405
|
-
export {
|
|
406
|
-
spawnQwen,
|
|
407
|
-
abortQwenSession,
|
|
408
|
-
isQwenSessionActive,
|
|
409
|
-
getActiveQwenSessions,
|
|
410
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* Qwen Code CLI adapter.
|
|
3
|
+
*
|
|
4
|
+
* Qwen Code (https://github.com/QwenLM/qwen-code) is Alibaba's fork of Google's
|
|
5
|
+
* Gemini CLI. The command-line surface, stream-json output, session layout
|
|
6
|
+
* (~/.qwen/tmp/<project>/...), and approval flags all mirror Gemini's. This
|
|
7
|
+
* adapter is therefore a structural copy of gemini-cli.js — kept as its own
|
|
8
|
+
* file so future Qwen-specific divergence (different auth flow, different
|
|
9
|
+
* model list) has a clean place to land without touching Gemini's code path.
|
|
10
|
+
*/
|
|
11
|
+
import { spawn } from 'child_process';
|
|
12
|
+
import { promises as fs } from 'fs';
|
|
13
|
+
import path from 'path';
|
|
14
|
+
import os from 'os';
|
|
15
|
+
|
|
16
|
+
import crossSpawn from 'cross-spawn';
|
|
17
|
+
|
|
18
|
+
import sessionManager from './sessionManager.js';
|
|
19
|
+
import QwenResponseHandler from './qwen-response-handler.js';
|
|
20
|
+
import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
|
|
21
|
+
import { buildSpawnEnv } from './services/provider-credentials.js';
|
|
22
|
+
import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
|
|
23
|
+
import { createNormalizedMessage } from './shared/utils.js';
|
|
24
|
+
|
|
25
|
+
// Use cross-spawn on Windows so `qwen.cmd` resolves correctly.
|
|
26
|
+
const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
|
|
27
|
+
|
|
28
|
+
const activeQwenProcesses = new Map();
|
|
29
|
+
const DEFAULT_CLI_IDLE_TIMEOUT_MS = 600000;
|
|
30
|
+
|
|
31
|
+
function readCliIdleTimeoutMs() {
|
|
32
|
+
const configured = Number.parseInt(process.env.PIXCODE_CLI_IDLE_TIMEOUT_MS || '', 10);
|
|
33
|
+
return Number.isFinite(configured) && configured >= 0 ? configured : DEFAULT_CLI_IDLE_TIMEOUT_MS;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
async function spawnQwen(command, options = {}, ws) {
|
|
37
|
+
const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
|
|
38
|
+
let capturedSessionId = sessionId;
|
|
39
|
+
let sessionCreatedSent = false;
|
|
40
|
+
let assistantBlocks = [];
|
|
41
|
+
|
|
42
|
+
const settings = toolsSettings || { allowedTools: [], disallowedTools: [], skipPermissions: false };
|
|
43
|
+
|
|
44
|
+
const args = [];
|
|
45
|
+
if (command && command.trim()) {
|
|
46
|
+
args.push('--prompt', command);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
if (sessionId) {
|
|
50
|
+
const session = sessionManager.getSession(sessionId);
|
|
51
|
+
if (session && session.cliSessionId) {
|
|
52
|
+
args.push('--resume', session.cliSessionId);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const cleanPath = (cwd || projectPath || process.cwd()).replace(/[^\x20-\x7E]/g, '').trim();
|
|
57
|
+
const workingDir = cleanPath;
|
|
58
|
+
|
|
59
|
+
const tempImagePaths = [];
|
|
60
|
+
let tempDir = null;
|
|
61
|
+
if (images && images.length > 0) {
|
|
62
|
+
try {
|
|
63
|
+
tempDir = path.join(workingDir, '.tmp', 'images', Date.now().toString());
|
|
64
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
65
|
+
|
|
66
|
+
for (const [index, image] of images.entries()) {
|
|
67
|
+
const matches = image.data.match(/^data:([^;]+);base64,(.+)$/);
|
|
68
|
+
if (!matches) continue;
|
|
69
|
+
const [, mimeType, base64Data] = matches;
|
|
70
|
+
const extension = mimeType.split('/')[1] || 'png';
|
|
71
|
+
const filename = `image_${index}.${extension}`;
|
|
72
|
+
const filepath = path.join(tempDir, filename);
|
|
73
|
+
await fs.writeFile(filepath, Buffer.from(base64Data, 'base64'));
|
|
74
|
+
tempImagePaths.push(filepath);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (tempImagePaths.length > 0 && command && command.trim()) {
|
|
78
|
+
const imageNote = `\n\n[Images given: ${tempImagePaths.length} images are located at the following paths:]\n${tempImagePaths.map((p, i) => `${i + 1}. ${p}`).join('\n')}`;
|
|
79
|
+
const modifiedCommand = command + imageNote;
|
|
80
|
+
const promptIndex = args.indexOf('--prompt');
|
|
81
|
+
if (promptIndex !== -1 && args[promptIndex + 1] === command) {
|
|
82
|
+
args[promptIndex + 1] = modifiedCommand;
|
|
83
|
+
} else if (promptIndex !== -1) {
|
|
84
|
+
args[promptIndex + 1] = args[promptIndex + 1] + imageNote;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
} catch (error) {
|
|
88
|
+
console.error('Error processing images for Qwen Code:', error);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
if (options.debug) {
|
|
93
|
+
args.push('--debug');
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// Qwen's MCP config mirrors Gemini's — per-user settings.json plus optional
|
|
97
|
+
// project override. Pixcode writes to the user-scope file via the provider
|
|
98
|
+
// MCP module, and Qwen Code auto-loads it, so we don't pass --mcp-config
|
|
99
|
+
// explicitly. Left intentionally minimal to avoid double-loading.
|
|
100
|
+
|
|
101
|
+
const modelToUse = options.model || 'qwen3-coder-plus';
|
|
102
|
+
args.push('--model', modelToUse);
|
|
103
|
+
args.push('--output-format', 'stream-json');
|
|
104
|
+
|
|
105
|
+
if (
|
|
106
|
+
settings.skipPermissions ||
|
|
107
|
+
options.skipPermissions ||
|
|
108
|
+
permissionMode === 'yolo' ||
|
|
109
|
+
permissionMode === 'bypassPermissions' ||
|
|
110
|
+
permissionMode === 'acceptEdits'
|
|
111
|
+
) {
|
|
112
|
+
args.push('--yolo');
|
|
113
|
+
} else if (permissionMode === 'auto_edit') {
|
|
114
|
+
args.push('--approval-mode', 'auto_edit');
|
|
115
|
+
} else if (permissionMode === 'plan') {
|
|
116
|
+
args.push('--approval-mode', 'plan');
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
if (settings.allowedTools && settings.allowedTools.length > 0) {
|
|
120
|
+
args.push('--allowed-tools', settings.allowedTools.join(','));
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
const qwenPath = process.env.QWEN_PATH || 'qwen';
|
|
124
|
+
console.log('Spawning Qwen Code CLI:', qwenPath, args.join(' '));
|
|
125
|
+
console.log('Working directory:', workingDir);
|
|
126
|
+
|
|
127
|
+
let spawnCmd = qwenPath;
|
|
128
|
+
let spawnArgs = args;
|
|
129
|
+
|
|
130
|
+
if (os.platform() !== 'win32') {
|
|
131
|
+
spawnCmd = 'sh';
|
|
132
|
+
spawnArgs = ['-c', 'exec "$0" "$@"', qwenPath, ...args];
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Credentials stored in ~/.pixcode/provider-credentials.json take
|
|
136
|
+
// precedence over the host shell env, so an API key saved via the
|
|
137
|
+
// Pixcode UI reaches the Qwen subprocess even when the user never
|
|
138
|
+
// exported it in their login shell.
|
|
139
|
+
const spawnEnv = await buildSpawnEnv('qwen');
|
|
140
|
+
|
|
141
|
+
return new Promise((resolve, reject) => {
|
|
142
|
+
const qwenProcess = spawnFunction(spawnCmd, spawnArgs, {
|
|
143
|
+
cwd: workingDir,
|
|
144
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
145
|
+
env: spawnEnv,
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
let terminalNotificationSent = false;
|
|
149
|
+
let terminalFailureReason = null;
|
|
150
|
+
|
|
151
|
+
const notifyTerminalState = ({ code = null, error = null } = {}) => {
|
|
152
|
+
if (terminalNotificationSent) return;
|
|
153
|
+
terminalNotificationSent = true;
|
|
154
|
+
|
|
155
|
+
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
156
|
+
if (code === 0 && !error) {
|
|
157
|
+
notifyRunStopped({
|
|
158
|
+
userId: ws?.userId || null,
|
|
159
|
+
provider: 'qwen',
|
|
160
|
+
sessionId: finalSessionId,
|
|
161
|
+
sessionName: sessionSummary,
|
|
162
|
+
stopReason: 'completed',
|
|
163
|
+
});
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
notifyRunFailed({
|
|
168
|
+
userId: ws?.userId || null,
|
|
169
|
+
provider: 'qwen',
|
|
170
|
+
sessionId: finalSessionId,
|
|
171
|
+
sessionName: sessionSummary,
|
|
172
|
+
error: error || terminalFailureReason || `Qwen Code CLI exited with code ${code}`,
|
|
173
|
+
});
|
|
174
|
+
};
|
|
175
|
+
|
|
176
|
+
qwenProcess.tempImagePaths = tempImagePaths;
|
|
177
|
+
qwenProcess.tempDir = tempDir;
|
|
178
|
+
|
|
179
|
+
const processKey = capturedSessionId || sessionId || Date.now().toString();
|
|
180
|
+
activeQwenProcesses.set(processKey, qwenProcess);
|
|
181
|
+
qwenProcess.sessionId = processKey;
|
|
182
|
+
|
|
183
|
+
qwenProcess.stdin.end();
|
|
184
|
+
|
|
185
|
+
const timeoutMs = readCliIdleTimeoutMs();
|
|
186
|
+
let timeout;
|
|
187
|
+
const startTimeout = () => {
|
|
188
|
+
if (timeout) clearTimeout(timeout);
|
|
189
|
+
if (timeoutMs === 0) return;
|
|
190
|
+
timeout = setTimeout(() => {
|
|
191
|
+
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId || processKey);
|
|
192
|
+
terminalFailureReason = `Qwen Code CLI timeout - no response received for ${timeoutMs / 1000} seconds`;
|
|
193
|
+
ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'qwen' }));
|
|
194
|
+
try { qwenProcess.kill('SIGTERM'); } catch { /* noop */ }
|
|
195
|
+
}, timeoutMs);
|
|
196
|
+
};
|
|
197
|
+
startTimeout();
|
|
198
|
+
|
|
199
|
+
if (command && capturedSessionId) {
|
|
200
|
+
sessionManager.addMessage(capturedSessionId, 'user', command);
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
let responseHandler;
|
|
204
|
+
if (ws) {
|
|
205
|
+
responseHandler = new QwenResponseHandler(ws, {
|
|
206
|
+
onContentFragment: (content) => {
|
|
207
|
+
if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
|
|
208
|
+
assistantBlocks[assistantBlocks.length - 1].text += content;
|
|
209
|
+
} else {
|
|
210
|
+
assistantBlocks.push({ type: 'text', text: content });
|
|
211
|
+
}
|
|
212
|
+
},
|
|
213
|
+
onToolUse: (event) => {
|
|
214
|
+
assistantBlocks.push({
|
|
215
|
+
type: 'tool_use',
|
|
216
|
+
id: event.tool_id,
|
|
217
|
+
name: event.tool_name,
|
|
218
|
+
input: event.parameters,
|
|
219
|
+
});
|
|
220
|
+
},
|
|
221
|
+
onToolResult: (event) => {
|
|
222
|
+
if (capturedSessionId) {
|
|
223
|
+
if (assistantBlocks.length > 0) {
|
|
224
|
+
sessionManager.addMessage(capturedSessionId, 'assistant', [...assistantBlocks]);
|
|
225
|
+
assistantBlocks = [];
|
|
226
|
+
}
|
|
227
|
+
sessionManager.addMessage(capturedSessionId, 'user', [{
|
|
228
|
+
type: 'tool_result',
|
|
229
|
+
tool_use_id: event.tool_id,
|
|
230
|
+
content: event.output === undefined ? null : event.output,
|
|
231
|
+
is_error: event.status === 'error',
|
|
232
|
+
}]);
|
|
233
|
+
}
|
|
234
|
+
},
|
|
235
|
+
onInit: (event) => {
|
|
236
|
+
if (capturedSessionId) {
|
|
237
|
+
const sess = sessionManager.getSession(capturedSessionId);
|
|
238
|
+
if (sess && !sess.cliSessionId) {
|
|
239
|
+
sess.cliSessionId = event.session_id;
|
|
240
|
+
sessionManager.saveSession(capturedSessionId);
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
},
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
qwenProcess.stdout.on('data', (data) => {
|
|
248
|
+
const rawOutput = data.toString();
|
|
249
|
+
startTimeout();
|
|
250
|
+
|
|
251
|
+
if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
|
|
252
|
+
capturedSessionId = `qwen_${Date.now()}`;
|
|
253
|
+
sessionCreatedSent = true;
|
|
254
|
+
|
|
255
|
+
sessionManager.createSession(capturedSessionId, cwd || process.cwd());
|
|
256
|
+
if (command) {
|
|
257
|
+
sessionManager.addMessage(capturedSessionId, 'user', command);
|
|
258
|
+
}
|
|
259
|
+
if (processKey !== capturedSessionId) {
|
|
260
|
+
activeQwenProcesses.delete(processKey);
|
|
261
|
+
activeQwenProcesses.set(capturedSessionId, qwenProcess);
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
|
|
265
|
+
ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'qwen' }));
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
if (responseHandler) {
|
|
269
|
+
responseHandler.processData(rawOutput);
|
|
270
|
+
} else if (rawOutput) {
|
|
271
|
+
if (assistantBlocks.length > 0 && assistantBlocks[assistantBlocks.length - 1].type === 'text') {
|
|
272
|
+
assistantBlocks[assistantBlocks.length - 1].text += rawOutput;
|
|
273
|
+
} else {
|
|
274
|
+
assistantBlocks.push({ type: 'text', text: rawOutput });
|
|
275
|
+
}
|
|
276
|
+
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
|
|
277
|
+
ws.send(createNormalizedMessage({ kind: 'stream_delta', content: rawOutput, sessionId: socketSessionId, provider: 'qwen' }));
|
|
278
|
+
}
|
|
279
|
+
});
|
|
280
|
+
|
|
281
|
+
qwenProcess.stderr.on('data', (data) => {
|
|
282
|
+
const errorMsg = data.toString();
|
|
283
|
+
startTimeout();
|
|
284
|
+
if (errorMsg.includes('[DEP0040]') ||
|
|
285
|
+
errorMsg.includes('DeprecationWarning') ||
|
|
286
|
+
errorMsg.includes('--trace-deprecation') ||
|
|
287
|
+
errorMsg.includes('Loaded cached credentials')) {
|
|
288
|
+
return;
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : (capturedSessionId || sessionId);
|
|
292
|
+
ws.send(createNormalizedMessage({ kind: 'error', content: errorMsg, sessionId: socketSessionId, provider: 'qwen' }));
|
|
293
|
+
});
|
|
294
|
+
|
|
295
|
+
qwenProcess.on('close', async (code) => {
|
|
296
|
+
clearTimeout(timeout);
|
|
297
|
+
|
|
298
|
+
if (responseHandler) {
|
|
299
|
+
responseHandler.forceFlush();
|
|
300
|
+
responseHandler.destroy();
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
304
|
+
activeQwenProcesses.delete(finalSessionId);
|
|
305
|
+
|
|
306
|
+
if (finalSessionId && assistantBlocks.length > 0) {
|
|
307
|
+
sessionManager.addMessage(finalSessionId, 'assistant', assistantBlocks);
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
ws.send(createNormalizedMessage({ kind: 'complete', exitCode: code, isNewSession: !sessionId && !!command, sessionId: finalSessionId, provider: 'qwen' }));
|
|
311
|
+
|
|
312
|
+
if (qwenProcess.tempImagePaths && qwenProcess.tempImagePaths.length > 0) {
|
|
313
|
+
for (const imagePath of qwenProcess.tempImagePaths) {
|
|
314
|
+
await fs.unlink(imagePath).catch(() => { /* noop */ });
|
|
315
|
+
}
|
|
316
|
+
if (qwenProcess.tempDir) {
|
|
317
|
+
await fs.rm(qwenProcess.tempDir, { recursive: true, force: true }).catch(() => { /* noop */ });
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
if (code === 0) {
|
|
322
|
+
notifyTerminalState({ code });
|
|
323
|
+
resolve();
|
|
324
|
+
} else {
|
|
325
|
+
if (code === 127) {
|
|
326
|
+
const installed = await providerAuthService.isProviderInstalled('qwen');
|
|
327
|
+
if (!installed) {
|
|
328
|
+
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
|
329
|
+
ws.send(createNormalizedMessage({
|
|
330
|
+
kind: 'error',
|
|
331
|
+
content: 'Qwen Code CLI is not installed. Install it first: npm install -g @qwen-code/qwen-code',
|
|
332
|
+
sessionId: socketSessionId,
|
|
333
|
+
provider: 'qwen',
|
|
334
|
+
}));
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
notifyTerminalState({
|
|
339
|
+
code,
|
|
340
|
+
error: code === null ? 'Qwen Code CLI process was terminated or timed out' : null,
|
|
341
|
+
});
|
|
342
|
+
reject(new Error(code === null ? 'Qwen Code CLI process was terminated or timed out' : `Qwen Code CLI exited with code ${code}`));
|
|
343
|
+
}
|
|
344
|
+
});
|
|
345
|
+
|
|
346
|
+
qwenProcess.on('error', async (error) => {
|
|
347
|
+
const finalSessionId = capturedSessionId || sessionId || processKey;
|
|
348
|
+
activeQwenProcesses.delete(finalSessionId);
|
|
349
|
+
|
|
350
|
+
const installed = await providerAuthService.isProviderInstalled('qwen');
|
|
351
|
+
const errorContent = !installed
|
|
352
|
+
? 'Qwen Code CLI is not installed. Install it first: npm install -g @qwen-code/qwen-code'
|
|
353
|
+
: (error?.message || String(error));
|
|
354
|
+
|
|
355
|
+
const errorSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
|
356
|
+
ws.send(createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: errorSessionId, provider: 'qwen' }));
|
|
357
|
+
// Always emit `complete` so the UI's "Processing..." state clears
|
|
358
|
+
// even when spawn fails (ENOENT, EACCES) and `close` never fires.
|
|
359
|
+
ws.send(createNormalizedMessage({ kind: 'complete', exitCode: 1, isNewSession: !sessionId && !!command, sessionId: errorSessionId, provider: 'qwen' }));
|
|
360
|
+
notifyTerminalState({ error });
|
|
361
|
+
|
|
362
|
+
reject(error);
|
|
363
|
+
});
|
|
364
|
+
});
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
function abortQwenSession(sessionId) {
|
|
368
|
+
let qwenProc = activeQwenProcesses.get(sessionId);
|
|
369
|
+
let processKey = sessionId;
|
|
370
|
+
|
|
371
|
+
if (!qwenProc) {
|
|
372
|
+
for (const [key, proc] of activeQwenProcesses.entries()) {
|
|
373
|
+
if (proc.sessionId === sessionId) {
|
|
374
|
+
qwenProc = proc;
|
|
375
|
+
processKey = key;
|
|
376
|
+
break;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
if (qwenProc) {
|
|
382
|
+
try {
|
|
383
|
+
qwenProc.kill('SIGTERM');
|
|
384
|
+
setTimeout(() => {
|
|
385
|
+
if (activeQwenProcesses.has(processKey)) {
|
|
386
|
+
try { qwenProc.kill('SIGKILL'); } catch { /* noop */ }
|
|
387
|
+
}
|
|
388
|
+
}, 2000);
|
|
389
|
+
return true;
|
|
390
|
+
} catch {
|
|
391
|
+
return false;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
return false;
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
function isQwenSessionActive(sessionId) {
|
|
398
|
+
return activeQwenProcesses.has(sessionId);
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
function getActiveQwenSessions() {
|
|
402
|
+
return Array.from(activeQwenProcesses.keys());
|
|
403
|
+
}
|
|
404
|
+
|
|
405
|
+
export {
|
|
406
|
+
spawnQwen,
|
|
407
|
+
abortQwenSession,
|
|
408
|
+
isQwenSessionActive,
|
|
409
|
+
getActiveQwenSessions,
|
|
410
|
+
};
|