@pixelbyte-software/pixcode 1.51.2 → 1.51.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- 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-B9N-gfOQ.css +32 -0
- package/dist/assets/{index-EN9ngyxf.js → index-HfGHXhD6.js} +175 -175
- 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/database/db.js +14 -2
- package/dist-server/server/database/db.js.map +1 -1
- package/dist-server/server/index.js +191 -31
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/middleware/auth.js +16 -5
- package/dist-server/server/middleware/auth.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/auth.js +12 -5
- package/dist-server/server/routes/auth.js.map +1 -1
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +29 -17
- package/dist-server/server/routes/git.js.map +1 -1
- package/dist-server/server/routes/live-view.js +46 -46
- package/dist-server/server/routes/platformization.js +7 -6
- package/dist-server/server/routes/platformization.js.map +1 -1
- 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/platformization.js +58 -2
- package/dist-server/server/services/platformization.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 +908 -895
- 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 +201 -30
- package/server/load-env.js +35 -35
- package/server/middleware/auth.js +171 -156
- 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 +154 -146
- 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 -1635
- 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 +198 -197
- 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 +844 -779
- 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,493 +1,493 @@
|
|
|
1
|
-
import { spawn } from 'node:child_process';
|
|
2
|
-
import { promises as fs } from 'node:fs';
|
|
3
|
-
import os from 'node:os';
|
|
4
|
-
import path from 'node:path';
|
|
5
|
-
|
|
6
|
-
import * as tar from 'tar';
|
|
7
|
-
|
|
8
|
-
import { buildCliSpawnEnv, findExecutableOnPath, resolveNpmCommand } from './install-jobs.js';
|
|
9
|
-
|
|
10
|
-
const DEFAULT_RUNTIMES_HOME = path.join(os.homedir(), '.pixcode', 'runtimes');
|
|
11
|
-
const MANIFEST_FILE = 'pixcode-runtime.json';
|
|
12
|
-
const installLocks = new Map();
|
|
13
|
-
|
|
14
|
-
function runtimesHome(env = process.env) {
|
|
15
|
-
return env.PIXCODE_MANAGED_RUNTIMES_HOME || DEFAULT_RUNTIMES_HOME;
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
function runtimeDir(id, env = process.env) {
|
|
19
|
-
return path.join(runtimesHome(env), id);
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
function manifestPath(id, env = process.env) {
|
|
23
|
-
return path.join(runtimeDir(id, env), MANIFEST_FILE);
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
async function fileExists(filePath) {
|
|
27
|
-
try {
|
|
28
|
-
const stats = await fs.stat(filePath);
|
|
29
|
-
return stats.isFile();
|
|
30
|
-
} catch {
|
|
31
|
-
return false;
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
async function readManifest(id, env = process.env) {
|
|
36
|
-
try {
|
|
37
|
-
const content = await fs.readFile(manifestPath(id, env), 'utf8');
|
|
38
|
-
const manifest = JSON.parse(content);
|
|
39
|
-
if (manifest?.executablePath && await fileExists(manifest.executablePath)) {
|
|
40
|
-
return manifest;
|
|
41
|
-
}
|
|
42
|
-
} catch {
|
|
43
|
-
// Missing or malformed manifests are treated as not installed.
|
|
44
|
-
}
|
|
45
|
-
return null;
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
function platformTokens() {
|
|
49
|
-
if (process.platform === 'win32') return ['windows'];
|
|
50
|
-
if (process.platform === 'darwin') return ['mac', 'darwin'];
|
|
51
|
-
if (process.platform === 'linux') return ['linux'];
|
|
52
|
-
return [process.platform];
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
function archTokens() {
|
|
56
|
-
if (process.arch === 'x64') return ['x86_64', 'amd64', 'x64'];
|
|
57
|
-
if (process.arch === 'arm64') return ['aarch64', 'arm64'];
|
|
58
|
-
return [process.arch];
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
function scoreFrankenPhpAsset(assetName) {
|
|
62
|
-
const name = assetName.toLowerCase();
|
|
63
|
-
if (!name.includes('frankenphp')) return -1;
|
|
64
|
-
if (name.includes('debug')) return -1;
|
|
65
|
-
if (!platformTokens().some((token) => name.includes(token))) return -1;
|
|
66
|
-
if (!archTokens().some((token) => name.includes(token))) return -1;
|
|
67
|
-
|
|
68
|
-
let score = 10;
|
|
69
|
-
if (process.platform === 'win32' && name.endsWith('.zip')) score += 10;
|
|
70
|
-
if (process.platform !== 'win32' && !name.endsWith('.zip')) score += 10;
|
|
71
|
-
if (!name.includes('gnu')) score += 2;
|
|
72
|
-
if (name.endsWith('.tar.gz') || name.endsWith('.tgz')) score += 1;
|
|
73
|
-
return score;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
function selectFrankenPhpAsset(release) {
|
|
77
|
-
const assets = Array.isArray(release?.assets) ? release.assets : [];
|
|
78
|
-
const candidates = assets
|
|
79
|
-
.map((asset) => ({ asset, score: scoreFrankenPhpAsset(asset.name || '') }))
|
|
80
|
-
.filter((entry) => entry.score >= 0)
|
|
81
|
-
.sort((a, b) => b.score - a.score);
|
|
82
|
-
|
|
83
|
-
return candidates[0]?.asset || null;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
function isGitHubUrl(url) {
|
|
87
|
-
try {
|
|
88
|
-
const hostname = new URL(url).hostname.toLowerCase();
|
|
89
|
-
return hostname === 'github.com' || hostname === 'api.github.com' || hostname.endsWith('.github.com');
|
|
90
|
-
} catch {
|
|
91
|
-
return String(url).includes('github.com');
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
async function fetchJson(url, env = process.env, options = {}) {
|
|
96
|
-
const headers = {
|
|
97
|
-
Accept: options.accept || 'application/json',
|
|
98
|
-
'User-Agent': 'Pixcode Live View',
|
|
99
|
-
};
|
|
100
|
-
if (env.GITHUB_TOKEN && isGitHubUrl(url)) headers.Authorization = `Bearer ${env.GITHUB_TOKEN}`;
|
|
101
|
-
|
|
102
|
-
const response = await fetch(url, { headers });
|
|
103
|
-
if (!response.ok) {
|
|
104
|
-
throw new Error(`Runtime metadata request failed with HTTP ${response.status}`);
|
|
105
|
-
}
|
|
106
|
-
return response.json();
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
async function downloadFile(url, targetFile, env = process.env) {
|
|
110
|
-
const headers = { 'User-Agent': 'Pixcode Live View' };
|
|
111
|
-
if (env.GITHUB_TOKEN && isGitHubUrl(url)) headers.Authorization = `Bearer ${env.GITHUB_TOKEN}`;
|
|
112
|
-
|
|
113
|
-
const response = await fetch(url, { headers });
|
|
114
|
-
if (!response.ok) {
|
|
115
|
-
throw new Error(`Runtime download failed with HTTP ${response.status}`);
|
|
116
|
-
}
|
|
117
|
-
const buffer = Buffer.from(await response.arrayBuffer());
|
|
118
|
-
await fs.writeFile(targetFile, buffer);
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
function runProcess(command, args, options = {}) {
|
|
122
|
-
return new Promise((resolve, reject) => {
|
|
123
|
-
let stderr = '';
|
|
124
|
-
let settled = false;
|
|
125
|
-
const { timeoutMs, ...spawnOptions } = options;
|
|
126
|
-
const child = spawn(command, args, {
|
|
127
|
-
...spawnOptions,
|
|
128
|
-
stdio: ['ignore', 'ignore', 'pipe'],
|
|
129
|
-
windowsHide: true,
|
|
130
|
-
});
|
|
131
|
-
const finish = (callback, value) => {
|
|
132
|
-
if (settled) return;
|
|
133
|
-
settled = true;
|
|
134
|
-
if (timer) clearTimeout(timer);
|
|
135
|
-
callback(value);
|
|
136
|
-
};
|
|
137
|
-
const timer = timeoutMs
|
|
138
|
-
? setTimeout(() => {
|
|
139
|
-
try {
|
|
140
|
-
child.kill();
|
|
141
|
-
} catch {
|
|
142
|
-
// Process may have exited between timeout scheduling and kill.
|
|
143
|
-
}
|
|
144
|
-
finish(reject, new Error(`${command} timed out after ${timeoutMs}ms`));
|
|
145
|
-
}, timeoutMs)
|
|
146
|
-
: null;
|
|
147
|
-
child.stderr.on('data', (chunk) => {
|
|
148
|
-
stderr += chunk.toString();
|
|
149
|
-
});
|
|
150
|
-
child.on('error', (error) => finish(reject, error));
|
|
151
|
-
child.on('close', (code) => {
|
|
152
|
-
if (code === 0) {
|
|
153
|
-
finish(resolve);
|
|
154
|
-
return;
|
|
155
|
-
}
|
|
156
|
-
finish(reject, new Error(`${command} exited with code ${code}${stderr ? `: ${stderr.trim()}` : ''}`));
|
|
157
|
-
});
|
|
158
|
-
});
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
function buildPowerShellExpandArchiveArgs(archivePath, targetDir) {
|
|
162
|
-
return [
|
|
163
|
-
'-NoProfile',
|
|
164
|
-
'-ExecutionPolicy',
|
|
165
|
-
'Bypass',
|
|
166
|
-
'-Command',
|
|
167
|
-
'& { param([string]$archive, [string]$destination) Expand-Archive -Force -LiteralPath $archive -DestinationPath $destination }',
|
|
168
|
-
archivePath,
|
|
169
|
-
targetDir,
|
|
170
|
-
];
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
async function extractZip(archivePath, targetDir, env = process.env) {
|
|
174
|
-
if (process.platform === 'win32') {
|
|
175
|
-
const shell = env.ComSpec || process.env.ComSpec || 'powershell.exe';
|
|
176
|
-
const isCmd = shell.toLowerCase().endsWith('cmd.exe');
|
|
177
|
-
const expandArgs = buildPowerShellExpandArchiveArgs(archivePath, targetDir);
|
|
178
|
-
if (isCmd) {
|
|
179
|
-
await runProcess('powershell.exe', expandArgs, { env });
|
|
180
|
-
return;
|
|
181
|
-
}
|
|
182
|
-
await runProcess(shell, expandArgs, { env });
|
|
183
|
-
return;
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
await runProcess('unzip', ['-q', archivePath, '-d', targetDir], { env });
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
async function extractTarGz(archivePath, targetDir, env = process.env) {
|
|
190
|
-
void env;
|
|
191
|
-
await tar.x({
|
|
192
|
-
file: archivePath,
|
|
193
|
-
cwd: targetDir,
|
|
194
|
-
});
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
async function findRuntimeExecutable(searchRoot, binaryName) {
|
|
198
|
-
const expectedNames = process.platform === 'win32'
|
|
199
|
-
? [`${binaryName}.exe`, binaryName]
|
|
200
|
-
: [binaryName];
|
|
201
|
-
const stack = [searchRoot];
|
|
202
|
-
|
|
203
|
-
while (stack.length > 0) {
|
|
204
|
-
const current = stack.pop();
|
|
205
|
-
const entries = await fs.readdir(current, { withFileTypes: true }).catch(() => []);
|
|
206
|
-
for (const entry of entries) {
|
|
207
|
-
const full = path.join(current, entry.name);
|
|
208
|
-
if (entry.isDirectory()) {
|
|
209
|
-
stack.push(full);
|
|
210
|
-
continue;
|
|
211
|
-
}
|
|
212
|
-
if (entry.isFile() && expectedNames.includes(entry.name)) {
|
|
213
|
-
return full;
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
return null;
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
async function copyRuntimeWithSidecars(executablePath, currentDir) {
|
|
222
|
-
const executableDir = path.dirname(executablePath);
|
|
223
|
-
await fs.rm(currentDir, { recursive: true, force: true });
|
|
224
|
-
await fs.mkdir(currentDir, { recursive: true });
|
|
225
|
-
await fs.cp(executableDir, currentDir, { recursive: true, force: true });
|
|
226
|
-
return path.join(currentDir, path.basename(executablePath));
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
async function validateManagedRuntimeExecutable(id, executablePath, env = process.env) {
|
|
230
|
-
if (id !== 'frankenphp') return true;
|
|
231
|
-
try {
|
|
232
|
-
await runProcess(executablePath, ['version'], {
|
|
233
|
-
env,
|
|
234
|
-
timeoutMs: 5000,
|
|
235
|
-
});
|
|
236
|
-
return true;
|
|
237
|
-
} catch {
|
|
238
|
-
return false;
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
async function installFrankenPhp(env = process.env) {
|
|
243
|
-
const releaseApiUrl = env.PIXCODE_FRANKENPHP_RELEASE_API
|
|
244
|
-
|| 'https://api.github.com/repos/php/frankenphp/releases/latest';
|
|
245
|
-
const release = env.PIXCODE_FRANKENPHP_URL
|
|
246
|
-
? null
|
|
247
|
-
: await fetchJson(releaseApiUrl, env, { accept: 'application/vnd.github+json' });
|
|
248
|
-
const asset = env.PIXCODE_FRANKENPHP_URL
|
|
249
|
-
? {
|
|
250
|
-
name: path.basename(new URL(env.PIXCODE_FRANKENPHP_URL).pathname),
|
|
251
|
-
browser_download_url: env.PIXCODE_FRANKENPHP_URL,
|
|
252
|
-
}
|
|
253
|
-
: selectFrankenPhpAsset(release);
|
|
254
|
-
|
|
255
|
-
if (!asset?.browser_download_url) {
|
|
256
|
-
throw new Error('No FrankenPHP binary is available for this operating system and CPU architecture.');
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
const baseDir = runtimeDir('frankenphp', env);
|
|
260
|
-
const stagingDir = path.join(baseDir, `.staging-${Date.now()}`);
|
|
261
|
-
const currentDir = path.join(baseDir, 'current');
|
|
262
|
-
await fs.mkdir(stagingDir, { recursive: true });
|
|
263
|
-
|
|
264
|
-
const archivePath = path.join(stagingDir, asset.name || 'frankenphp');
|
|
265
|
-
await downloadFile(asset.browser_download_url, archivePath, env);
|
|
266
|
-
|
|
267
|
-
let executablePath = archivePath;
|
|
268
|
-
const assetName = (asset.name || '').toLowerCase();
|
|
269
|
-
if (assetName.endsWith('.zip')) {
|
|
270
|
-
await extractZip(archivePath, stagingDir, env);
|
|
271
|
-
executablePath = await findRuntimeExecutable(stagingDir, 'frankenphp');
|
|
272
|
-
if (!executablePath) {
|
|
273
|
-
throw new Error('Downloaded FrankenPHP archive did not contain a frankenphp executable.');
|
|
274
|
-
}
|
|
275
|
-
} else if (assetName.endsWith('.tar.gz') || assetName.endsWith('.tgz')) {
|
|
276
|
-
await extractTarGz(archivePath, stagingDir, env);
|
|
277
|
-
executablePath = await findRuntimeExecutable(stagingDir, 'frankenphp');
|
|
278
|
-
if (!executablePath) {
|
|
279
|
-
throw new Error('Downloaded FrankenPHP archive did not contain a frankenphp executable.');
|
|
280
|
-
}
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
if (process.platform !== 'win32') {
|
|
284
|
-
await fs.chmod(executablePath, 0o755).catch(() => undefined);
|
|
285
|
-
}
|
|
286
|
-
|
|
287
|
-
const finalExecutable = await copyRuntimeWithSidecars(executablePath, currentDir);
|
|
288
|
-
if (process.platform !== 'win32') {
|
|
289
|
-
await fs.chmod(finalExecutable, 0o755).catch(() => undefined);
|
|
290
|
-
}
|
|
291
|
-
|
|
292
|
-
const manifest = {
|
|
293
|
-
id: 'frankenphp',
|
|
294
|
-
label: 'Pixcode PHP runtime',
|
|
295
|
-
provider: 'FrankenPHP',
|
|
296
|
-
version: release?.tag_name || 'custom',
|
|
297
|
-
executablePath: finalExecutable,
|
|
298
|
-
sourceUrl: asset.browser_download_url,
|
|
299
|
-
installedAt: new Date().toISOString(),
|
|
300
|
-
};
|
|
301
|
-
await fs.writeFile(manifestPath('frankenphp', env), JSON.stringify(manifest, null, 2));
|
|
302
|
-
await fs.rm(stagingDir, { recursive: true, force: true }).catch(() => undefined);
|
|
303
|
-
return manifest;
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
async function installNpmRuntime(env = process.env) {
|
|
307
|
-
const registryUrl = env.PIXCODE_NPM_RUNTIME_REGISTRY
|
|
308
|
-
|| 'https://registry.npmjs.org/npm/latest';
|
|
309
|
-
const metadata = await fetchJson(registryUrl, env, { accept: 'application/json' });
|
|
310
|
-
const tarballUrl = metadata?.dist?.tarball;
|
|
311
|
-
if (!tarballUrl) {
|
|
312
|
-
throw new Error('No npm runtime tarball is available from the npm registry.');
|
|
313
|
-
}
|
|
314
|
-
|
|
315
|
-
const baseDir = runtimeDir('npm', env);
|
|
316
|
-
const stagingDir = path.join(baseDir, `.staging-${Date.now()}`);
|
|
317
|
-
const currentDir = path.join(baseDir, 'current');
|
|
318
|
-
await fs.mkdir(stagingDir, { recursive: true });
|
|
319
|
-
|
|
320
|
-
const archivePath = path.join(stagingDir, 'npm-runtime.tgz');
|
|
321
|
-
await downloadFile(tarballUrl, archivePath, env);
|
|
322
|
-
await extractTarGz(archivePath, stagingDir, env);
|
|
323
|
-
|
|
324
|
-
const packageDir = path.join(stagingDir, 'package');
|
|
325
|
-
const executablePath = path.join(packageDir, 'bin', 'npm-cli.js');
|
|
326
|
-
if (!(await fileExists(executablePath))) {
|
|
327
|
-
throw new Error('Downloaded npm runtime did not contain bin/npm-cli.js.');
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
await fs.rm(currentDir, { recursive: true, force: true });
|
|
331
|
-
await fs.cp(packageDir, currentDir, { recursive: true, force: true });
|
|
332
|
-
|
|
333
|
-
const finalExecutable = path.join(currentDir, 'bin', 'npm-cli.js');
|
|
334
|
-
const manifest = {
|
|
335
|
-
id: 'npm',
|
|
336
|
-
label: 'Pixcode Node package runner',
|
|
337
|
-
provider: 'npm',
|
|
338
|
-
version: metadata?.version || 'latest',
|
|
339
|
-
executablePath: finalExecutable,
|
|
340
|
-
runner: 'node',
|
|
341
|
-
sourceUrl: tarballUrl,
|
|
342
|
-
installedAt: new Date().toISOString(),
|
|
343
|
-
};
|
|
344
|
-
await fs.writeFile(manifestPath('npm', env), JSON.stringify(manifest, null, 2));
|
|
345
|
-
await fs.rm(stagingDir, { recursive: true, force: true }).catch(() => undefined);
|
|
346
|
-
return manifest;
|
|
347
|
-
}
|
|
348
|
-
|
|
349
|
-
export async function getManagedRuntimeStatus(id, options = {}) {
|
|
350
|
-
const env = options.env || process.env;
|
|
351
|
-
const preferManaged = Boolean(options.preferManaged);
|
|
352
|
-
if (id !== 'frankenphp' && id !== 'npm') {
|
|
353
|
-
return {
|
|
354
|
-
id,
|
|
355
|
-
status: 'unsupported',
|
|
356
|
-
installable: false,
|
|
357
|
-
reason: 'Pixcode does not have a managed runtime for this stack yet.',
|
|
358
|
-
};
|
|
359
|
-
}
|
|
360
|
-
|
|
361
|
-
if (id === 'npm') {
|
|
362
|
-
if (!preferManaged) {
|
|
363
|
-
const spawnEnv = buildCliSpawnEnv(env);
|
|
364
|
-
const npmExecutable = resolveNpmCommand(spawnEnv);
|
|
365
|
-
if (npmExecutable) {
|
|
366
|
-
return {
|
|
367
|
-
id,
|
|
368
|
-
label: 'npm',
|
|
369
|
-
status: 'system',
|
|
370
|
-
installable: true,
|
|
371
|
-
executablePath: npmExecutable,
|
|
372
|
-
runner: npmExecutable.endsWith('.js') ? 'node' : undefined,
|
|
373
|
-
};
|
|
374
|
-
}
|
|
375
|
-
}
|
|
376
|
-
|
|
377
|
-
const manifest = await readManifest(id, env);
|
|
378
|
-
if (manifest) {
|
|
379
|
-
return {
|
|
380
|
-
id,
|
|
381
|
-
label: manifest.label || 'Pixcode Node package runner',
|
|
382
|
-
status: 'installed',
|
|
383
|
-
installable: true,
|
|
384
|
-
executablePath: manifest.executablePath,
|
|
385
|
-
runner: manifest.runner || 'node',
|
|
386
|
-
version: manifest.version,
|
|
387
|
-
};
|
|
388
|
-
}
|
|
389
|
-
|
|
390
|
-
return {
|
|
391
|
-
id,
|
|
392
|
-
label: 'Pixcode Node package runner',
|
|
393
|
-
provider: 'npm',
|
|
394
|
-
status: 'missing',
|
|
395
|
-
installable: true,
|
|
396
|
-
reason: 'Pixcode will prepare a local Node package runner automatically before starting this project.',
|
|
397
|
-
};
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
if (!preferManaged) {
|
|
401
|
-
const spawnEnv = buildCliSpawnEnv(env);
|
|
402
|
-
const systemExecutable = findExecutableOnPath('frankenphp', spawnEnv);
|
|
403
|
-
if (systemExecutable) {
|
|
404
|
-
return {
|
|
405
|
-
id,
|
|
406
|
-
label: 'FrankenPHP',
|
|
407
|
-
status: 'system',
|
|
408
|
-
installable: true,
|
|
409
|
-
executablePath: systemExecutable,
|
|
410
|
-
};
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
const manifest = await readManifest(id, env);
|
|
415
|
-
if (manifest) {
|
|
416
|
-
const valid = await validateManagedRuntimeExecutable(id, manifest.executablePath, env);
|
|
417
|
-
if (!valid) {
|
|
418
|
-
return {
|
|
419
|
-
id,
|
|
420
|
-
label: 'Pixcode PHP runtime',
|
|
421
|
-
provider: 'FrankenPHP',
|
|
422
|
-
status: 'missing',
|
|
423
|
-
installable: true,
|
|
424
|
-
reason: 'The existing Pixcode PHP runtime is incomplete or cannot start. Pixcode will reinstall it automatically.',
|
|
425
|
-
};
|
|
426
|
-
}
|
|
427
|
-
|
|
428
|
-
return {
|
|
429
|
-
id,
|
|
430
|
-
label: manifest.label || 'Pixcode PHP runtime',
|
|
431
|
-
status: 'installed',
|
|
432
|
-
installable: true,
|
|
433
|
-
executablePath: manifest.executablePath,
|
|
434
|
-
version: manifest.version,
|
|
435
|
-
};
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
return {
|
|
439
|
-
id,
|
|
440
|
-
label: 'Pixcode PHP runtime',
|
|
441
|
-
provider: 'FrankenPHP',
|
|
442
|
-
status: 'missing',
|
|
443
|
-
installable: true,
|
|
444
|
-
reason: 'Pixcode will prepare a local PHP runtime automatically before starting this project.',
|
|
445
|
-
};
|
|
446
|
-
}
|
|
447
|
-
|
|
448
|
-
export async function ensureManagedRuntime(id, options = {}) {
|
|
449
|
-
const env = options.env || process.env;
|
|
450
|
-
const status = await getManagedRuntimeStatus(id, {
|
|
451
|
-
env,
|
|
452
|
-
preferManaged: options.preferManaged,
|
|
453
|
-
});
|
|
454
|
-
if (status.executablePath) return status;
|
|
455
|
-
if (!status.installable) {
|
|
456
|
-
throw new Error(status.reason || 'This runtime cannot be prepared automatically.');
|
|
457
|
-
}
|
|
458
|
-
|
|
459
|
-
const lockKey = `${runtimesHome(env)}:${id}`;
|
|
460
|
-
if (installLocks.has(lockKey)) return installLocks.get(lockKey);
|
|
461
|
-
|
|
462
|
-
const installPromise = (async () => {
|
|
463
|
-
if (id === 'frankenphp') {
|
|
464
|
-
const manifest = await installFrankenPhp(env);
|
|
465
|
-
return {
|
|
466
|
-
id,
|
|
467
|
-
label: manifest.label,
|
|
468
|
-
status: 'installed',
|
|
469
|
-
installable: true,
|
|
470
|
-
executablePath: manifest.executablePath,
|
|
471
|
-
version: manifest.version,
|
|
472
|
-
};
|
|
473
|
-
}
|
|
474
|
-
if (id === 'npm') {
|
|
475
|
-
const manifest = await installNpmRuntime(env);
|
|
476
|
-
return {
|
|
477
|
-
id,
|
|
478
|
-
label: manifest.label,
|
|
479
|
-
status: 'installed',
|
|
480
|
-
installable: true,
|
|
481
|
-
executablePath: manifest.executablePath,
|
|
482
|
-
runner: manifest.runner || 'node',
|
|
483
|
-
version: manifest.version,
|
|
484
|
-
};
|
|
485
|
-
}
|
|
486
|
-
throw new Error(`Unsupported managed runtime: ${id}`);
|
|
487
|
-
})().finally(() => {
|
|
488
|
-
installLocks.delete(lockKey);
|
|
489
|
-
});
|
|
490
|
-
|
|
491
|
-
installLocks.set(lockKey, installPromise);
|
|
492
|
-
return installPromise;
|
|
493
|
-
}
|
|
1
|
+
import { spawn } from 'node:child_process';
|
|
2
|
+
import { promises as fs } from 'node:fs';
|
|
3
|
+
import os from 'node:os';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
|
|
6
|
+
import * as tar from 'tar';
|
|
7
|
+
|
|
8
|
+
import { buildCliSpawnEnv, findExecutableOnPath, resolveNpmCommand } from './install-jobs.js';
|
|
9
|
+
|
|
10
|
+
const DEFAULT_RUNTIMES_HOME = path.join(os.homedir(), '.pixcode', 'runtimes');
|
|
11
|
+
const MANIFEST_FILE = 'pixcode-runtime.json';
|
|
12
|
+
const installLocks = new Map();
|
|
13
|
+
|
|
14
|
+
function runtimesHome(env = process.env) {
|
|
15
|
+
return env.PIXCODE_MANAGED_RUNTIMES_HOME || DEFAULT_RUNTIMES_HOME;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
function runtimeDir(id, env = process.env) {
|
|
19
|
+
return path.join(runtimesHome(env), id);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function manifestPath(id, env = process.env) {
|
|
23
|
+
return path.join(runtimeDir(id, env), MANIFEST_FILE);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
async function fileExists(filePath) {
|
|
27
|
+
try {
|
|
28
|
+
const stats = await fs.stat(filePath);
|
|
29
|
+
return stats.isFile();
|
|
30
|
+
} catch {
|
|
31
|
+
return false;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
async function readManifest(id, env = process.env) {
|
|
36
|
+
try {
|
|
37
|
+
const content = await fs.readFile(manifestPath(id, env), 'utf8');
|
|
38
|
+
const manifest = JSON.parse(content);
|
|
39
|
+
if (manifest?.executablePath && await fileExists(manifest.executablePath)) {
|
|
40
|
+
return manifest;
|
|
41
|
+
}
|
|
42
|
+
} catch {
|
|
43
|
+
// Missing or malformed manifests are treated as not installed.
|
|
44
|
+
}
|
|
45
|
+
return null;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function platformTokens() {
|
|
49
|
+
if (process.platform === 'win32') return ['windows'];
|
|
50
|
+
if (process.platform === 'darwin') return ['mac', 'darwin'];
|
|
51
|
+
if (process.platform === 'linux') return ['linux'];
|
|
52
|
+
return [process.platform];
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function archTokens() {
|
|
56
|
+
if (process.arch === 'x64') return ['x86_64', 'amd64', 'x64'];
|
|
57
|
+
if (process.arch === 'arm64') return ['aarch64', 'arm64'];
|
|
58
|
+
return [process.arch];
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
function scoreFrankenPhpAsset(assetName) {
|
|
62
|
+
const name = assetName.toLowerCase();
|
|
63
|
+
if (!name.includes('frankenphp')) return -1;
|
|
64
|
+
if (name.includes('debug')) return -1;
|
|
65
|
+
if (!platformTokens().some((token) => name.includes(token))) return -1;
|
|
66
|
+
if (!archTokens().some((token) => name.includes(token))) return -1;
|
|
67
|
+
|
|
68
|
+
let score = 10;
|
|
69
|
+
if (process.platform === 'win32' && name.endsWith('.zip')) score += 10;
|
|
70
|
+
if (process.platform !== 'win32' && !name.endsWith('.zip')) score += 10;
|
|
71
|
+
if (!name.includes('gnu')) score += 2;
|
|
72
|
+
if (name.endsWith('.tar.gz') || name.endsWith('.tgz')) score += 1;
|
|
73
|
+
return score;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function selectFrankenPhpAsset(release) {
|
|
77
|
+
const assets = Array.isArray(release?.assets) ? release.assets : [];
|
|
78
|
+
const candidates = assets
|
|
79
|
+
.map((asset) => ({ asset, score: scoreFrankenPhpAsset(asset.name || '') }))
|
|
80
|
+
.filter((entry) => entry.score >= 0)
|
|
81
|
+
.sort((a, b) => b.score - a.score);
|
|
82
|
+
|
|
83
|
+
return candidates[0]?.asset || null;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function isGitHubUrl(url) {
|
|
87
|
+
try {
|
|
88
|
+
const hostname = new URL(url).hostname.toLowerCase();
|
|
89
|
+
return hostname === 'github.com' || hostname === 'api.github.com' || hostname.endsWith('.github.com');
|
|
90
|
+
} catch {
|
|
91
|
+
return String(url).includes('github.com');
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
async function fetchJson(url, env = process.env, options = {}) {
|
|
96
|
+
const headers = {
|
|
97
|
+
Accept: options.accept || 'application/json',
|
|
98
|
+
'User-Agent': 'Pixcode Live View',
|
|
99
|
+
};
|
|
100
|
+
if (env.GITHUB_TOKEN && isGitHubUrl(url)) headers.Authorization = `Bearer ${env.GITHUB_TOKEN}`;
|
|
101
|
+
|
|
102
|
+
const response = await fetch(url, { headers });
|
|
103
|
+
if (!response.ok) {
|
|
104
|
+
throw new Error(`Runtime metadata request failed with HTTP ${response.status}`);
|
|
105
|
+
}
|
|
106
|
+
return response.json();
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
async function downloadFile(url, targetFile, env = process.env) {
|
|
110
|
+
const headers = { 'User-Agent': 'Pixcode Live View' };
|
|
111
|
+
if (env.GITHUB_TOKEN && isGitHubUrl(url)) headers.Authorization = `Bearer ${env.GITHUB_TOKEN}`;
|
|
112
|
+
|
|
113
|
+
const response = await fetch(url, { headers });
|
|
114
|
+
if (!response.ok) {
|
|
115
|
+
throw new Error(`Runtime download failed with HTTP ${response.status}`);
|
|
116
|
+
}
|
|
117
|
+
const buffer = Buffer.from(await response.arrayBuffer());
|
|
118
|
+
await fs.writeFile(targetFile, buffer);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function runProcess(command, args, options = {}) {
|
|
122
|
+
return new Promise((resolve, reject) => {
|
|
123
|
+
let stderr = '';
|
|
124
|
+
let settled = false;
|
|
125
|
+
const { timeoutMs, ...spawnOptions } = options;
|
|
126
|
+
const child = spawn(command, args, {
|
|
127
|
+
...spawnOptions,
|
|
128
|
+
stdio: ['ignore', 'ignore', 'pipe'],
|
|
129
|
+
windowsHide: true,
|
|
130
|
+
});
|
|
131
|
+
const finish = (callback, value) => {
|
|
132
|
+
if (settled) return;
|
|
133
|
+
settled = true;
|
|
134
|
+
if (timer) clearTimeout(timer);
|
|
135
|
+
callback(value);
|
|
136
|
+
};
|
|
137
|
+
const timer = timeoutMs
|
|
138
|
+
? setTimeout(() => {
|
|
139
|
+
try {
|
|
140
|
+
child.kill();
|
|
141
|
+
} catch {
|
|
142
|
+
// Process may have exited between timeout scheduling and kill.
|
|
143
|
+
}
|
|
144
|
+
finish(reject, new Error(`${command} timed out after ${timeoutMs}ms`));
|
|
145
|
+
}, timeoutMs)
|
|
146
|
+
: null;
|
|
147
|
+
child.stderr.on('data', (chunk) => {
|
|
148
|
+
stderr += chunk.toString();
|
|
149
|
+
});
|
|
150
|
+
child.on('error', (error) => finish(reject, error));
|
|
151
|
+
child.on('close', (code) => {
|
|
152
|
+
if (code === 0) {
|
|
153
|
+
finish(resolve);
|
|
154
|
+
return;
|
|
155
|
+
}
|
|
156
|
+
finish(reject, new Error(`${command} exited with code ${code}${stderr ? `: ${stderr.trim()}` : ''}`));
|
|
157
|
+
});
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
function buildPowerShellExpandArchiveArgs(archivePath, targetDir) {
|
|
162
|
+
return [
|
|
163
|
+
'-NoProfile',
|
|
164
|
+
'-ExecutionPolicy',
|
|
165
|
+
'Bypass',
|
|
166
|
+
'-Command',
|
|
167
|
+
'& { param([string]$archive, [string]$destination) Expand-Archive -Force -LiteralPath $archive -DestinationPath $destination }',
|
|
168
|
+
archivePath,
|
|
169
|
+
targetDir,
|
|
170
|
+
];
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
async function extractZip(archivePath, targetDir, env = process.env) {
|
|
174
|
+
if (process.platform === 'win32') {
|
|
175
|
+
const shell = env.ComSpec || process.env.ComSpec || 'powershell.exe';
|
|
176
|
+
const isCmd = shell.toLowerCase().endsWith('cmd.exe');
|
|
177
|
+
const expandArgs = buildPowerShellExpandArchiveArgs(archivePath, targetDir);
|
|
178
|
+
if (isCmd) {
|
|
179
|
+
await runProcess('powershell.exe', expandArgs, { env });
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
await runProcess(shell, expandArgs, { env });
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
await runProcess('unzip', ['-q', archivePath, '-d', targetDir], { env });
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
async function extractTarGz(archivePath, targetDir, env = process.env) {
|
|
190
|
+
void env;
|
|
191
|
+
await tar.x({
|
|
192
|
+
file: archivePath,
|
|
193
|
+
cwd: targetDir,
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
async function findRuntimeExecutable(searchRoot, binaryName) {
|
|
198
|
+
const expectedNames = process.platform === 'win32'
|
|
199
|
+
? [`${binaryName}.exe`, binaryName]
|
|
200
|
+
: [binaryName];
|
|
201
|
+
const stack = [searchRoot];
|
|
202
|
+
|
|
203
|
+
while (stack.length > 0) {
|
|
204
|
+
const current = stack.pop();
|
|
205
|
+
const entries = await fs.readdir(current, { withFileTypes: true }).catch(() => []);
|
|
206
|
+
for (const entry of entries) {
|
|
207
|
+
const full = path.join(current, entry.name);
|
|
208
|
+
if (entry.isDirectory()) {
|
|
209
|
+
stack.push(full);
|
|
210
|
+
continue;
|
|
211
|
+
}
|
|
212
|
+
if (entry.isFile() && expectedNames.includes(entry.name)) {
|
|
213
|
+
return full;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
return null;
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
async function copyRuntimeWithSidecars(executablePath, currentDir) {
|
|
222
|
+
const executableDir = path.dirname(executablePath);
|
|
223
|
+
await fs.rm(currentDir, { recursive: true, force: true });
|
|
224
|
+
await fs.mkdir(currentDir, { recursive: true });
|
|
225
|
+
await fs.cp(executableDir, currentDir, { recursive: true, force: true });
|
|
226
|
+
return path.join(currentDir, path.basename(executablePath));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
async function validateManagedRuntimeExecutable(id, executablePath, env = process.env) {
|
|
230
|
+
if (id !== 'frankenphp') return true;
|
|
231
|
+
try {
|
|
232
|
+
await runProcess(executablePath, ['version'], {
|
|
233
|
+
env,
|
|
234
|
+
timeoutMs: 5000,
|
|
235
|
+
});
|
|
236
|
+
return true;
|
|
237
|
+
} catch {
|
|
238
|
+
return false;
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
async function installFrankenPhp(env = process.env) {
|
|
243
|
+
const releaseApiUrl = env.PIXCODE_FRANKENPHP_RELEASE_API
|
|
244
|
+
|| 'https://api.github.com/repos/php/frankenphp/releases/latest';
|
|
245
|
+
const release = env.PIXCODE_FRANKENPHP_URL
|
|
246
|
+
? null
|
|
247
|
+
: await fetchJson(releaseApiUrl, env, { accept: 'application/vnd.github+json' });
|
|
248
|
+
const asset = env.PIXCODE_FRANKENPHP_URL
|
|
249
|
+
? {
|
|
250
|
+
name: path.basename(new URL(env.PIXCODE_FRANKENPHP_URL).pathname),
|
|
251
|
+
browser_download_url: env.PIXCODE_FRANKENPHP_URL,
|
|
252
|
+
}
|
|
253
|
+
: selectFrankenPhpAsset(release);
|
|
254
|
+
|
|
255
|
+
if (!asset?.browser_download_url) {
|
|
256
|
+
throw new Error('No FrankenPHP binary is available for this operating system and CPU architecture.');
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
const baseDir = runtimeDir('frankenphp', env);
|
|
260
|
+
const stagingDir = path.join(baseDir, `.staging-${Date.now()}`);
|
|
261
|
+
const currentDir = path.join(baseDir, 'current');
|
|
262
|
+
await fs.mkdir(stagingDir, { recursive: true });
|
|
263
|
+
|
|
264
|
+
const archivePath = path.join(stagingDir, asset.name || 'frankenphp');
|
|
265
|
+
await downloadFile(asset.browser_download_url, archivePath, env);
|
|
266
|
+
|
|
267
|
+
let executablePath = archivePath;
|
|
268
|
+
const assetName = (asset.name || '').toLowerCase();
|
|
269
|
+
if (assetName.endsWith('.zip')) {
|
|
270
|
+
await extractZip(archivePath, stagingDir, env);
|
|
271
|
+
executablePath = await findRuntimeExecutable(stagingDir, 'frankenphp');
|
|
272
|
+
if (!executablePath) {
|
|
273
|
+
throw new Error('Downloaded FrankenPHP archive did not contain a frankenphp executable.');
|
|
274
|
+
}
|
|
275
|
+
} else if (assetName.endsWith('.tar.gz') || assetName.endsWith('.tgz')) {
|
|
276
|
+
await extractTarGz(archivePath, stagingDir, env);
|
|
277
|
+
executablePath = await findRuntimeExecutable(stagingDir, 'frankenphp');
|
|
278
|
+
if (!executablePath) {
|
|
279
|
+
throw new Error('Downloaded FrankenPHP archive did not contain a frankenphp executable.');
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
if (process.platform !== 'win32') {
|
|
284
|
+
await fs.chmod(executablePath, 0o755).catch(() => undefined);
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
const finalExecutable = await copyRuntimeWithSidecars(executablePath, currentDir);
|
|
288
|
+
if (process.platform !== 'win32') {
|
|
289
|
+
await fs.chmod(finalExecutable, 0o755).catch(() => undefined);
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
const manifest = {
|
|
293
|
+
id: 'frankenphp',
|
|
294
|
+
label: 'Pixcode PHP runtime',
|
|
295
|
+
provider: 'FrankenPHP',
|
|
296
|
+
version: release?.tag_name || 'custom',
|
|
297
|
+
executablePath: finalExecutable,
|
|
298
|
+
sourceUrl: asset.browser_download_url,
|
|
299
|
+
installedAt: new Date().toISOString(),
|
|
300
|
+
};
|
|
301
|
+
await fs.writeFile(manifestPath('frankenphp', env), JSON.stringify(manifest, null, 2));
|
|
302
|
+
await fs.rm(stagingDir, { recursive: true, force: true }).catch(() => undefined);
|
|
303
|
+
return manifest;
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
async function installNpmRuntime(env = process.env) {
|
|
307
|
+
const registryUrl = env.PIXCODE_NPM_RUNTIME_REGISTRY
|
|
308
|
+
|| 'https://registry.npmjs.org/npm/latest';
|
|
309
|
+
const metadata = await fetchJson(registryUrl, env, { accept: 'application/json' });
|
|
310
|
+
const tarballUrl = metadata?.dist?.tarball;
|
|
311
|
+
if (!tarballUrl) {
|
|
312
|
+
throw new Error('No npm runtime tarball is available from the npm registry.');
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
const baseDir = runtimeDir('npm', env);
|
|
316
|
+
const stagingDir = path.join(baseDir, `.staging-${Date.now()}`);
|
|
317
|
+
const currentDir = path.join(baseDir, 'current');
|
|
318
|
+
await fs.mkdir(stagingDir, { recursive: true });
|
|
319
|
+
|
|
320
|
+
const archivePath = path.join(stagingDir, 'npm-runtime.tgz');
|
|
321
|
+
await downloadFile(tarballUrl, archivePath, env);
|
|
322
|
+
await extractTarGz(archivePath, stagingDir, env);
|
|
323
|
+
|
|
324
|
+
const packageDir = path.join(stagingDir, 'package');
|
|
325
|
+
const executablePath = path.join(packageDir, 'bin', 'npm-cli.js');
|
|
326
|
+
if (!(await fileExists(executablePath))) {
|
|
327
|
+
throw new Error('Downloaded npm runtime did not contain bin/npm-cli.js.');
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
await fs.rm(currentDir, { recursive: true, force: true });
|
|
331
|
+
await fs.cp(packageDir, currentDir, { recursive: true, force: true });
|
|
332
|
+
|
|
333
|
+
const finalExecutable = path.join(currentDir, 'bin', 'npm-cli.js');
|
|
334
|
+
const manifest = {
|
|
335
|
+
id: 'npm',
|
|
336
|
+
label: 'Pixcode Node package runner',
|
|
337
|
+
provider: 'npm',
|
|
338
|
+
version: metadata?.version || 'latest',
|
|
339
|
+
executablePath: finalExecutable,
|
|
340
|
+
runner: 'node',
|
|
341
|
+
sourceUrl: tarballUrl,
|
|
342
|
+
installedAt: new Date().toISOString(),
|
|
343
|
+
};
|
|
344
|
+
await fs.writeFile(manifestPath('npm', env), JSON.stringify(manifest, null, 2));
|
|
345
|
+
await fs.rm(stagingDir, { recursive: true, force: true }).catch(() => undefined);
|
|
346
|
+
return manifest;
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
export async function getManagedRuntimeStatus(id, options = {}) {
|
|
350
|
+
const env = options.env || process.env;
|
|
351
|
+
const preferManaged = Boolean(options.preferManaged);
|
|
352
|
+
if (id !== 'frankenphp' && id !== 'npm') {
|
|
353
|
+
return {
|
|
354
|
+
id,
|
|
355
|
+
status: 'unsupported',
|
|
356
|
+
installable: false,
|
|
357
|
+
reason: 'Pixcode does not have a managed runtime for this stack yet.',
|
|
358
|
+
};
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
if (id === 'npm') {
|
|
362
|
+
if (!preferManaged) {
|
|
363
|
+
const spawnEnv = buildCliSpawnEnv(env);
|
|
364
|
+
const npmExecutable = resolveNpmCommand(spawnEnv);
|
|
365
|
+
if (npmExecutable) {
|
|
366
|
+
return {
|
|
367
|
+
id,
|
|
368
|
+
label: 'npm',
|
|
369
|
+
status: 'system',
|
|
370
|
+
installable: true,
|
|
371
|
+
executablePath: npmExecutable,
|
|
372
|
+
runner: npmExecutable.endsWith('.js') ? 'node' : undefined,
|
|
373
|
+
};
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
const manifest = await readManifest(id, env);
|
|
378
|
+
if (manifest) {
|
|
379
|
+
return {
|
|
380
|
+
id,
|
|
381
|
+
label: manifest.label || 'Pixcode Node package runner',
|
|
382
|
+
status: 'installed',
|
|
383
|
+
installable: true,
|
|
384
|
+
executablePath: manifest.executablePath,
|
|
385
|
+
runner: manifest.runner || 'node',
|
|
386
|
+
version: manifest.version,
|
|
387
|
+
};
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
return {
|
|
391
|
+
id,
|
|
392
|
+
label: 'Pixcode Node package runner',
|
|
393
|
+
provider: 'npm',
|
|
394
|
+
status: 'missing',
|
|
395
|
+
installable: true,
|
|
396
|
+
reason: 'Pixcode will prepare a local Node package runner automatically before starting this project.',
|
|
397
|
+
};
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
if (!preferManaged) {
|
|
401
|
+
const spawnEnv = buildCliSpawnEnv(env);
|
|
402
|
+
const systemExecutable = findExecutableOnPath('frankenphp', spawnEnv);
|
|
403
|
+
if (systemExecutable) {
|
|
404
|
+
return {
|
|
405
|
+
id,
|
|
406
|
+
label: 'FrankenPHP',
|
|
407
|
+
status: 'system',
|
|
408
|
+
installable: true,
|
|
409
|
+
executablePath: systemExecutable,
|
|
410
|
+
};
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
|
|
414
|
+
const manifest = await readManifest(id, env);
|
|
415
|
+
if (manifest) {
|
|
416
|
+
const valid = await validateManagedRuntimeExecutable(id, manifest.executablePath, env);
|
|
417
|
+
if (!valid) {
|
|
418
|
+
return {
|
|
419
|
+
id,
|
|
420
|
+
label: 'Pixcode PHP runtime',
|
|
421
|
+
provider: 'FrankenPHP',
|
|
422
|
+
status: 'missing',
|
|
423
|
+
installable: true,
|
|
424
|
+
reason: 'The existing Pixcode PHP runtime is incomplete or cannot start. Pixcode will reinstall it automatically.',
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
return {
|
|
429
|
+
id,
|
|
430
|
+
label: manifest.label || 'Pixcode PHP runtime',
|
|
431
|
+
status: 'installed',
|
|
432
|
+
installable: true,
|
|
433
|
+
executablePath: manifest.executablePath,
|
|
434
|
+
version: manifest.version,
|
|
435
|
+
};
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
return {
|
|
439
|
+
id,
|
|
440
|
+
label: 'Pixcode PHP runtime',
|
|
441
|
+
provider: 'FrankenPHP',
|
|
442
|
+
status: 'missing',
|
|
443
|
+
installable: true,
|
|
444
|
+
reason: 'Pixcode will prepare a local PHP runtime automatically before starting this project.',
|
|
445
|
+
};
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
export async function ensureManagedRuntime(id, options = {}) {
|
|
449
|
+
const env = options.env || process.env;
|
|
450
|
+
const status = await getManagedRuntimeStatus(id, {
|
|
451
|
+
env,
|
|
452
|
+
preferManaged: options.preferManaged,
|
|
453
|
+
});
|
|
454
|
+
if (status.executablePath) return status;
|
|
455
|
+
if (!status.installable) {
|
|
456
|
+
throw new Error(status.reason || 'This runtime cannot be prepared automatically.');
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
const lockKey = `${runtimesHome(env)}:${id}`;
|
|
460
|
+
if (installLocks.has(lockKey)) return installLocks.get(lockKey);
|
|
461
|
+
|
|
462
|
+
const installPromise = (async () => {
|
|
463
|
+
if (id === 'frankenphp') {
|
|
464
|
+
const manifest = await installFrankenPhp(env);
|
|
465
|
+
return {
|
|
466
|
+
id,
|
|
467
|
+
label: manifest.label,
|
|
468
|
+
status: 'installed',
|
|
469
|
+
installable: true,
|
|
470
|
+
executablePath: manifest.executablePath,
|
|
471
|
+
version: manifest.version,
|
|
472
|
+
};
|
|
473
|
+
}
|
|
474
|
+
if (id === 'npm') {
|
|
475
|
+
const manifest = await installNpmRuntime(env);
|
|
476
|
+
return {
|
|
477
|
+
id,
|
|
478
|
+
label: manifest.label,
|
|
479
|
+
status: 'installed',
|
|
480
|
+
installable: true,
|
|
481
|
+
executablePath: manifest.executablePath,
|
|
482
|
+
runner: manifest.runner || 'node',
|
|
483
|
+
version: manifest.version,
|
|
484
|
+
};
|
|
485
|
+
}
|
|
486
|
+
throw new Error(`Unsupported managed runtime: ${id}`);
|
|
487
|
+
})().finally(() => {
|
|
488
|
+
installLocks.delete(lockKey);
|
|
489
|
+
});
|
|
490
|
+
|
|
491
|
+
installLocks.set(lockKey, installPromise);
|
|
492
|
+
return installPromise;
|
|
493
|
+
}
|