@pixelbyte-software/pixcode 1.51.2 → 1.51.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- 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-EN9ngyxf.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 +122 -3
- 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 -167
- 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 +128 -2
- 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
|
@@ -1,56 +1,56 @@
|
|
|
1
|
-
import assert from 'node:assert/strict';
|
|
2
|
-
import fs from 'node:fs';
|
|
3
|
-
import path from 'node:path';
|
|
4
|
-
import { fileURLToPath } from 'node:url';
|
|
5
|
-
|
|
6
|
-
const repoRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), '..', '..');
|
|
7
|
-
|
|
8
|
-
const read = (relativePath) => fs.readFileSync(path.join(repoRoot, relativePath), 'utf8');
|
|
9
|
-
|
|
10
|
-
const hermesRoutes = read('server/modules/orchestration/hermes/hermes.routes.ts');
|
|
11
|
-
const hermesInstallJobs = fs.existsSync(path.join(repoRoot, 'server/services/hermes-install-jobs.js'))
|
|
12
|
-
? read('server/services/hermes-install-jobs.js')
|
|
13
|
-
: '';
|
|
14
|
-
const workbench = read('src/components/vscode-workbench/view/VSCodeWorkbench.tsx');
|
|
15
|
-
const serverIndex = read('server/index.js');
|
|
16
|
-
const smoke = read('scripts/smoke/pixcode-workbench-1-48.mjs');
|
|
17
|
-
|
|
18
|
-
assert.match(hermesRoutes, /createHermesInstallJob/, 'Hermes API should start backend install jobs.');
|
|
19
|
-
assert.match(hermesRoutes, /router\.post\('\/install'/, 'Hermes should expose POST /api/orchestration/hermes/install.');
|
|
20
|
-
assert.match(hermesRoutes, /router\.get\('\/install\/:jobId\/stream'/, 'Hermes install jobs should expose an EventSource stream.');
|
|
21
|
-
assert.match(hermesRoutes, /router\.delete\('\/install\/:jobId'/, 'Hermes install jobs should be cancellable.');
|
|
22
|
-
assert.match(hermesInstallJobs, /downloadHermesInstaller/, 'Hermes installer should be downloaded by backend code, not pasted into the terminal.');
|
|
23
|
-
assert.match(hermesInstallJobs, /--skip-setup/, 'POSIX Hermes API install should skip the interactive setup wizard.');
|
|
24
|
-
assert.match(hermesInstallJobs, /--skip-browser/, 'Hermes API install should skip browser downloads by default for reliable headless installs.');
|
|
25
|
-
assert.doesNotMatch(hermesInstallJobs, /curl -fsSL .* \| bash/, 'Hermes API install must not pipe curl directly into bash.');
|
|
26
|
-
assert.match(workbench, /\/api\/orchestration\/hermes\/install/, 'Workbench Hermes install button should call the Hermes install API.');
|
|
27
|
-
assert.doesNotMatch(workbench, /HERMES_AGENT_INSTALL_COMMAND/, 'Workbench should not launch Hermes install through a terminal command.');
|
|
28
|
-
assert.match(hermesInstallJobs, /formatHermesVersionOutput/, 'Hermes install status should collapse multi-line --version output before showing it in UI badges.');
|
|
29
|
-
assert.match(hermesInstallJobs, /repairHermesCommandLaunchers/, 'Hermes installer should repair stale or text launcher shims after install/status checks.');
|
|
30
|
-
assert.match(hermesInstallJobs, /hermes\.cmd/, 'Windows Hermes repair should create or prefer a hermes.cmd shim so typing hermes does not open the Python launcher as text.');
|
|
31
|
-
assert.match(hermesInstallJobs, /isUsableHermesCommand/, 'Hermes status should verify candidates before treating them as installed.');
|
|
32
|
-
assert.match(hermesInstallJobs, /isHermesSmokeCommandOutput/, 'Hermes status must reject smoke-test launchers before treating Hermes as installed.');
|
|
33
|
-
assert.match(hermesInstallJobs, /shouldRepairHermesLauncher/, 'Hermes status must not persist temporary HERMES_CLI_PATH launchers into user PATH shims.');
|
|
34
|
-
assert.match(hermesInstallJobs, /export function buildHermesPathEnv/, 'Hermes should expose a PATH-only shell env helper so normal project terminals can run hermes after repair.');
|
|
35
|
-
assert.match(hermesInstallJobs, /isWindowsEpermSpawnError/, 'Hermes install should recognize Windows spawn EPERM as a launcher failure.');
|
|
36
|
-
assert.match(hermesInstallJobs, /windowsCmdFallbackCommand/, 'Hermes install should retry Windows PowerShell installer launches through cmd.exe without requiring elevation.');
|
|
37
|
-
assert.match(hermesInstallJobs, /retrying through cmd\.exe without elevation/, 'Hermes install logs should explain the no-admin Windows EPERM fallback.');
|
|
38
|
-
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /platform_toolsets/, 'Pixcode MCP config should enable the Pixcode toolset for Hermes API server runs.');
|
|
39
|
-
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /api_server:[\s\S]+pixcode/, 'Hermes API server should see Pixcode MCP tools during /v1/runs.');
|
|
40
|
-
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /const cliToolsets = \['hermes-cli', 'mcp-pixcode'\]/, 'Hermes CLI profile should keep the full Hermes toolset while adding Pixcode MCP.');
|
|
41
|
-
assert.match(serverIndex, /buildHermesPathEnv\(process\.env/, 'Shell PTYs should inherit Hermes bin directories so typing hermes in Pixcode terminal works on Windows.');
|
|
42
|
-
assert.match(serverIndex, /env: shellEnv/, 'Shell PTYs should use the augmented shell environment instead of raw process.env.');
|
|
43
|
-
assert.doesNotMatch(serverIndex, /pixcode:hermes:start/, 'Hermes H button should not depend on a shell sentinel.');
|
|
44
|
-
assert.doesNotMatch(serverIndex, /Hermes already installed:/, 'Hermes H launch should not leave a version/status banner stuck before the interactive prompt.');
|
|
45
|
-
assert.doesNotMatch(serverIndex, /if command -v hermes >\/dev\/null 2>&1; then command -v hermes; return 0; fi;/, 'POSIX Hermes start must not accept a stale PATH shim without testing it.');
|
|
46
|
-
assert.match(workbench, /HermesActivityButton/, 'Workbench activity rail should expose a dedicated Hermes H button under Terminal.');
|
|
47
|
-
assert.match(workbench, /command=\{hermesCommand\}/, 'Opening the Hermes H button should show the real Hermes terminal/TUI.');
|
|
48
|
-
assert.match(workbench, /bottomTerminalCommand/, 'Hermes settings should be able to launch Hermes subcommands in the bottom terminal.');
|
|
49
|
-
assert.match(workbench, /openBottomTerminal\('hermes'\)/, 'Opening Hermes should restore the Hermes terminal.');
|
|
50
|
-
assert.match(workbench, /openBottomTerminal\('hermes', \{[\s\S]*command: HERMES_DEFAULT_COMMAND,[\s\S]*forceNewSession: true,[\s\S]*\}\)/, 'The Hermes new-session action should reset the Hermes terminal session with the guarded default command.');
|
|
51
|
-
assert.match(workbench, /installLogRef/, 'Hermes install log panel should keep a scroll ref.');
|
|
52
|
-
assert.match(workbench, /scrollTop = installLogRef\.current\.scrollHeight/, 'Hermes install logs should auto-scroll to the latest line.');
|
|
53
|
-
assert.doesNotMatch(workbench, /suspendAutoConnect/, 'Right CLI launches should not be blocked by an open Hermes bottom terminal.');
|
|
54
|
-
assert.match(smoke, /hermes-api-install\.mjs/, 'Main workbench smoke should mention the dedicated Hermes API install smoke.');
|
|
55
|
-
|
|
56
|
-
console.log('hermes API install smoke passed');
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
import path from 'node:path';
|
|
4
|
+
import { fileURLToPath } from 'node:url';
|
|
5
|
+
|
|
6
|
+
const repoRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), '..', '..');
|
|
7
|
+
|
|
8
|
+
const read = (relativePath) => fs.readFileSync(path.join(repoRoot, relativePath), 'utf8');
|
|
9
|
+
|
|
10
|
+
const hermesRoutes = read('server/modules/orchestration/hermes/hermes.routes.ts');
|
|
11
|
+
const hermesInstallJobs = fs.existsSync(path.join(repoRoot, 'server/services/hermes-install-jobs.js'))
|
|
12
|
+
? read('server/services/hermes-install-jobs.js')
|
|
13
|
+
: '';
|
|
14
|
+
const workbench = read('src/components/vscode-workbench/view/VSCodeWorkbench.tsx');
|
|
15
|
+
const serverIndex = read('server/index.js');
|
|
16
|
+
const smoke = read('scripts/smoke/pixcode-workbench-1-48.mjs');
|
|
17
|
+
|
|
18
|
+
assert.match(hermesRoutes, /createHermesInstallJob/, 'Hermes API should start backend install jobs.');
|
|
19
|
+
assert.match(hermesRoutes, /router\.post\('\/install'/, 'Hermes should expose POST /api/orchestration/hermes/install.');
|
|
20
|
+
assert.match(hermesRoutes, /router\.get\('\/install\/:jobId\/stream'/, 'Hermes install jobs should expose an EventSource stream.');
|
|
21
|
+
assert.match(hermesRoutes, /router\.delete\('\/install\/:jobId'/, 'Hermes install jobs should be cancellable.');
|
|
22
|
+
assert.match(hermesInstallJobs, /downloadHermesInstaller/, 'Hermes installer should be downloaded by backend code, not pasted into the terminal.');
|
|
23
|
+
assert.match(hermesInstallJobs, /--skip-setup/, 'POSIX Hermes API install should skip the interactive setup wizard.');
|
|
24
|
+
assert.match(hermesInstallJobs, /--skip-browser/, 'Hermes API install should skip browser downloads by default for reliable headless installs.');
|
|
25
|
+
assert.doesNotMatch(hermesInstallJobs, /curl -fsSL .* \| bash/, 'Hermes API install must not pipe curl directly into bash.');
|
|
26
|
+
assert.match(workbench, /\/api\/orchestration\/hermes\/install/, 'Workbench Hermes install button should call the Hermes install API.');
|
|
27
|
+
assert.doesNotMatch(workbench, /HERMES_AGENT_INSTALL_COMMAND/, 'Workbench should not launch Hermes install through a terminal command.');
|
|
28
|
+
assert.match(hermesInstallJobs, /formatHermesVersionOutput/, 'Hermes install status should collapse multi-line --version output before showing it in UI badges.');
|
|
29
|
+
assert.match(hermesInstallJobs, /repairHermesCommandLaunchers/, 'Hermes installer should repair stale or text launcher shims after install/status checks.');
|
|
30
|
+
assert.match(hermesInstallJobs, /hermes\.cmd/, 'Windows Hermes repair should create or prefer a hermes.cmd shim so typing hermes does not open the Python launcher as text.');
|
|
31
|
+
assert.match(hermesInstallJobs, /isUsableHermesCommand/, 'Hermes status should verify candidates before treating them as installed.');
|
|
32
|
+
assert.match(hermesInstallJobs, /isHermesSmokeCommandOutput/, 'Hermes status must reject smoke-test launchers before treating Hermes as installed.');
|
|
33
|
+
assert.match(hermesInstallJobs, /shouldRepairHermesLauncher/, 'Hermes status must not persist temporary HERMES_CLI_PATH launchers into user PATH shims.');
|
|
34
|
+
assert.match(hermesInstallJobs, /export function buildHermesPathEnv/, 'Hermes should expose a PATH-only shell env helper so normal project terminals can run hermes after repair.');
|
|
35
|
+
assert.match(hermesInstallJobs, /isWindowsEpermSpawnError/, 'Hermes install should recognize Windows spawn EPERM as a launcher failure.');
|
|
36
|
+
assert.match(hermesInstallJobs, /windowsCmdFallbackCommand/, 'Hermes install should retry Windows PowerShell installer launches through cmd.exe without requiring elevation.');
|
|
37
|
+
assert.match(hermesInstallJobs, /retrying through cmd\.exe without elevation/, 'Hermes install logs should explain the no-admin Windows EPERM fallback.');
|
|
38
|
+
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /platform_toolsets/, 'Pixcode MCP config should enable the Pixcode toolset for Hermes API server runs.');
|
|
39
|
+
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /api_server:[\s\S]+pixcode/, 'Hermes API server should see Pixcode MCP tools during /v1/runs.');
|
|
40
|
+
assert.match(read('scripts/hermes/configure-pixcode-mcp.mjs'), /const cliToolsets = \['hermes-cli', 'mcp-pixcode'\]/, 'Hermes CLI profile should keep the full Hermes toolset while adding Pixcode MCP.');
|
|
41
|
+
assert.match(serverIndex, /buildHermesPathEnv\(process\.env/, 'Shell PTYs should inherit Hermes bin directories so typing hermes in Pixcode terminal works on Windows.');
|
|
42
|
+
assert.match(serverIndex, /env: shellEnv/, 'Shell PTYs should use the augmented shell environment instead of raw process.env.');
|
|
43
|
+
assert.doesNotMatch(serverIndex, /pixcode:hermes:start/, 'Hermes H button should not depend on a shell sentinel.');
|
|
44
|
+
assert.doesNotMatch(serverIndex, /Hermes already installed:/, 'Hermes H launch should not leave a version/status banner stuck before the interactive prompt.');
|
|
45
|
+
assert.doesNotMatch(serverIndex, /if command -v hermes >\/dev\/null 2>&1; then command -v hermes; return 0; fi;/, 'POSIX Hermes start must not accept a stale PATH shim without testing it.');
|
|
46
|
+
assert.match(workbench, /HermesActivityButton/, 'Workbench activity rail should expose a dedicated Hermes H button under Terminal.');
|
|
47
|
+
assert.match(workbench, /command=\{hermesCommand\}/, 'Opening the Hermes H button should show the real Hermes terminal/TUI.');
|
|
48
|
+
assert.match(workbench, /bottomTerminalCommand/, 'Hermes settings should be able to launch Hermes subcommands in the bottom terminal.');
|
|
49
|
+
assert.match(workbench, /openBottomTerminal\('hermes'\)/, 'Opening Hermes should restore the Hermes terminal.');
|
|
50
|
+
assert.match(workbench, /openBottomTerminal\('hermes', \{[\s\S]*command: HERMES_DEFAULT_COMMAND,[\s\S]*forceNewSession: true,[\s\S]*\}\)/, 'The Hermes new-session action should reset the Hermes terminal session with the guarded default command.');
|
|
51
|
+
assert.match(workbench, /installLogRef/, 'Hermes install log panel should keep a scroll ref.');
|
|
52
|
+
assert.match(workbench, /scrollTop = installLogRef\.current\.scrollHeight/, 'Hermes install logs should auto-scroll to the latest line.');
|
|
53
|
+
assert.doesNotMatch(workbench, /suspendAutoConnect/, 'Right CLI launches should not be blocked by an open Hermes bottom terminal.');
|
|
54
|
+
assert.match(smoke, /hermes-api-install\.mjs/, 'Main workbench smoke should mention the dedicated Hermes API install smoke.');
|
|
55
|
+
|
|
56
|
+
console.log('hermes API install smoke passed');
|
|
@@ -1,104 +1,104 @@
|
|
|
1
|
-
import assert from 'node:assert/strict';
|
|
2
|
-
import fs from 'node:fs/promises';
|
|
3
|
-
import os from 'node:os';
|
|
4
|
-
import path from 'node:path';
|
|
5
|
-
import { fileURLToPath } from 'node:url';
|
|
6
|
-
|
|
7
|
-
import {
|
|
8
|
-
ensureHermesGateway,
|
|
9
|
-
stopHermesGateway,
|
|
10
|
-
} from '../../server/services/hermes-gateway.js';
|
|
11
|
-
|
|
12
|
-
const repoRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), '..', '..');
|
|
13
|
-
const tempRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'pixcode-hermes-persist-'));
|
|
14
|
-
const fakeHermes = path.join(tempRoot, 'hermes');
|
|
15
|
-
const projectPath = path.join(tempRoot, 'project');
|
|
16
|
-
const hermesHome = path.join(tempRoot, 'home');
|
|
17
|
-
const startCountFile = path.join(tempRoot, 'starts.txt');
|
|
18
|
-
const failProbeFile = path.join(tempRoot, 'fail-probe');
|
|
19
|
-
|
|
20
|
-
await fs.mkdir(projectPath, { recursive: true });
|
|
21
|
-
await fs.writeFile(fakeHermes, `#!/usr/bin/env node
|
|
22
|
-
import fs from 'node:fs';
|
|
23
|
-
import http from 'node:http';
|
|
24
|
-
|
|
25
|
-
if (process.argv.includes('--version')) {
|
|
26
|
-
console.log('Hermes Agent v0.0.0 smoke');
|
|
27
|
-
process.exit(0);
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
fs.appendFileSync(${JSON.stringify(startCountFile)}, '1\\n');
|
|
31
|
-
|
|
32
|
-
const host = process.env.API_SERVER_HOST || '127.0.0.1';
|
|
33
|
-
const port = Number(process.env.API_SERVER_PORT || 8642);
|
|
34
|
-
const key = process.env.API_SERVER_KEY || '';
|
|
35
|
-
const failProbeFile = ${JSON.stringify(failProbeFile)};
|
|
36
|
-
|
|
37
|
-
const server = http.createServer(async (req, res) => {
|
|
38
|
-
const url = new URL(req.url || '/', 'http://127.0.0.1');
|
|
39
|
-
res.setHeader('content-type', 'application/json');
|
|
40
|
-
if (url.pathname !== '/health' && req.headers.authorization !== \`Bearer \${key}\`) {
|
|
41
|
-
res.statusCode = 401;
|
|
42
|
-
res.end(JSON.stringify({ error: 'bad auth' }));
|
|
43
|
-
return;
|
|
44
|
-
}
|
|
45
|
-
if (req.method === 'GET' && url.pathname === '/health') {
|
|
46
|
-
res.end(JSON.stringify({ ok: true }));
|
|
47
|
-
return;
|
|
48
|
-
}
|
|
49
|
-
if (req.method === 'GET' && url.pathname === '/v1/capabilities') {
|
|
50
|
-
if (fs.existsSync(failProbeFile)) {
|
|
51
|
-
res.statusCode = 503;
|
|
52
|
-
res.end(JSON.stringify({ error: 'temporary probe failure' }));
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
res.end(JSON.stringify({ capabilities: ['chat'] }));
|
|
56
|
-
return;
|
|
57
|
-
}
|
|
58
|
-
if (req.method === 'GET' && url.pathname === '/v1/models') {
|
|
59
|
-
res.end(JSON.stringify({ data: [{ id: 'hermes-agent' }] }));
|
|
60
|
-
return;
|
|
61
|
-
}
|
|
62
|
-
res.statusCode = 404;
|
|
63
|
-
res.end(JSON.stringify({ error: url.pathname }));
|
|
64
|
-
});
|
|
65
|
-
|
|
66
|
-
server.listen(port, host);
|
|
67
|
-
`, { mode: 0o755 });
|
|
68
|
-
|
|
69
|
-
process.env.HERMES_CLI_PATH = fakeHermes;
|
|
70
|
-
|
|
71
|
-
try {
|
|
72
|
-
const first = await ensureHermesGateway({
|
|
73
|
-
appRoot: repoRoot,
|
|
74
|
-
projectPath,
|
|
75
|
-
hermesHome,
|
|
76
|
-
pixcodeBaseUrl: 'http://127.0.0.1:9',
|
|
77
|
-
pixcodeApiKey: 'px_persist_smoke_key',
|
|
78
|
-
port: 18772,
|
|
79
|
-
allowSmokeHermes: true,
|
|
80
|
-
repairLaunchers: false,
|
|
81
|
-
});
|
|
82
|
-
assert.equal(first.running, true, 'first gateway should start');
|
|
83
|
-
|
|
84
|
-
await fs.writeFile(failProbeFile, '1');
|
|
85
|
-
const second = await ensureHermesGateway({
|
|
86
|
-
appRoot: repoRoot,
|
|
87
|
-
projectPath,
|
|
88
|
-
hermesHome,
|
|
89
|
-
pixcodeBaseUrl: 'http://127.0.0.1:9',
|
|
90
|
-
pixcodeApiKey: 'px_persist_smoke_key',
|
|
91
|
-
port: 18772,
|
|
92
|
-
allowSmokeHermes: true,
|
|
93
|
-
repairLaunchers: false,
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
const starts = (await fs.readFile(startCountFile, 'utf8')).trim().split('\\n').filter(Boolean).length;
|
|
97
|
-
assert.equal(starts, 1, 'existing Hermes gateway must not be killed and relaunched for a transient probe failure');
|
|
98
|
-
assert.equal(second.baseUrl, first.baseUrl, 'existing Hermes gateway base URL should be reused');
|
|
99
|
-
assert.equal(second.running, true, 'existing Hermes gateway should still be reported as running');
|
|
100
|
-
console.log('hermes gateway persistence smoke passed');
|
|
101
|
-
} finally {
|
|
102
|
-
stopHermesGateway(projectPath);
|
|
103
|
-
delete process.env.HERMES_CLI_PATH;
|
|
104
|
-
}
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import fs from 'node:fs/promises';
|
|
3
|
+
import os from 'node:os';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
import { fileURLToPath } from 'node:url';
|
|
6
|
+
|
|
7
|
+
import {
|
|
8
|
+
ensureHermesGateway,
|
|
9
|
+
stopHermesGateway,
|
|
10
|
+
} from '../../server/services/hermes-gateway.js';
|
|
11
|
+
|
|
12
|
+
const repoRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), '..', '..');
|
|
13
|
+
const tempRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'pixcode-hermes-persist-'));
|
|
14
|
+
const fakeHermes = path.join(tempRoot, 'hermes');
|
|
15
|
+
const projectPath = path.join(tempRoot, 'project');
|
|
16
|
+
const hermesHome = path.join(tempRoot, 'home');
|
|
17
|
+
const startCountFile = path.join(tempRoot, 'starts.txt');
|
|
18
|
+
const failProbeFile = path.join(tempRoot, 'fail-probe');
|
|
19
|
+
|
|
20
|
+
await fs.mkdir(projectPath, { recursive: true });
|
|
21
|
+
await fs.writeFile(fakeHermes, `#!/usr/bin/env node
|
|
22
|
+
import fs from 'node:fs';
|
|
23
|
+
import http from 'node:http';
|
|
24
|
+
|
|
25
|
+
if (process.argv.includes('--version')) {
|
|
26
|
+
console.log('Hermes Agent v0.0.0 smoke');
|
|
27
|
+
process.exit(0);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
fs.appendFileSync(${JSON.stringify(startCountFile)}, '1\\n');
|
|
31
|
+
|
|
32
|
+
const host = process.env.API_SERVER_HOST || '127.0.0.1';
|
|
33
|
+
const port = Number(process.env.API_SERVER_PORT || 8642);
|
|
34
|
+
const key = process.env.API_SERVER_KEY || '';
|
|
35
|
+
const failProbeFile = ${JSON.stringify(failProbeFile)};
|
|
36
|
+
|
|
37
|
+
const server = http.createServer(async (req, res) => {
|
|
38
|
+
const url = new URL(req.url || '/', 'http://127.0.0.1');
|
|
39
|
+
res.setHeader('content-type', 'application/json');
|
|
40
|
+
if (url.pathname !== '/health' && req.headers.authorization !== \`Bearer \${key}\`) {
|
|
41
|
+
res.statusCode = 401;
|
|
42
|
+
res.end(JSON.stringify({ error: 'bad auth' }));
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
if (req.method === 'GET' && url.pathname === '/health') {
|
|
46
|
+
res.end(JSON.stringify({ ok: true }));
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
if (req.method === 'GET' && url.pathname === '/v1/capabilities') {
|
|
50
|
+
if (fs.existsSync(failProbeFile)) {
|
|
51
|
+
res.statusCode = 503;
|
|
52
|
+
res.end(JSON.stringify({ error: 'temporary probe failure' }));
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
res.end(JSON.stringify({ capabilities: ['chat'] }));
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
if (req.method === 'GET' && url.pathname === '/v1/models') {
|
|
59
|
+
res.end(JSON.stringify({ data: [{ id: 'hermes-agent' }] }));
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
res.statusCode = 404;
|
|
63
|
+
res.end(JSON.stringify({ error: url.pathname }));
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
server.listen(port, host);
|
|
67
|
+
`, { mode: 0o755 });
|
|
68
|
+
|
|
69
|
+
process.env.HERMES_CLI_PATH = fakeHermes;
|
|
70
|
+
|
|
71
|
+
try {
|
|
72
|
+
const first = await ensureHermesGateway({
|
|
73
|
+
appRoot: repoRoot,
|
|
74
|
+
projectPath,
|
|
75
|
+
hermesHome,
|
|
76
|
+
pixcodeBaseUrl: 'http://127.0.0.1:9',
|
|
77
|
+
pixcodeApiKey: 'px_persist_smoke_key',
|
|
78
|
+
port: 18772,
|
|
79
|
+
allowSmokeHermes: true,
|
|
80
|
+
repairLaunchers: false,
|
|
81
|
+
});
|
|
82
|
+
assert.equal(first.running, true, 'first gateway should start');
|
|
83
|
+
|
|
84
|
+
await fs.writeFile(failProbeFile, '1');
|
|
85
|
+
const second = await ensureHermesGateway({
|
|
86
|
+
appRoot: repoRoot,
|
|
87
|
+
projectPath,
|
|
88
|
+
hermesHome,
|
|
89
|
+
pixcodeBaseUrl: 'http://127.0.0.1:9',
|
|
90
|
+
pixcodeApiKey: 'px_persist_smoke_key',
|
|
91
|
+
port: 18772,
|
|
92
|
+
allowSmokeHermes: true,
|
|
93
|
+
repairLaunchers: false,
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
const starts = (await fs.readFile(startCountFile, 'utf8')).trim().split('\\n').filter(Boolean).length;
|
|
97
|
+
assert.equal(starts, 1, 'existing Hermes gateway must not be killed and relaunched for a transient probe failure');
|
|
98
|
+
assert.equal(second.baseUrl, first.baseUrl, 'existing Hermes gateway base URL should be reused');
|
|
99
|
+
assert.equal(second.running, true, 'existing Hermes gateway should still be reported as running');
|
|
100
|
+
console.log('hermes gateway persistence smoke passed');
|
|
101
|
+
} finally {
|
|
102
|
+
stopHermesGateway(projectPath);
|
|
103
|
+
delete process.env.HERMES_CLI_PATH;
|
|
104
|
+
}
|