@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
package/server/load-env.js
CHANGED
|
@@ -1,35 +1,35 @@
|
|
|
1
|
-
// Load environment variables from .env before other imports execute.
|
|
2
|
-
import fs from 'fs';
|
|
3
|
-
import os from 'os';
|
|
4
|
-
import path from 'path';
|
|
5
|
-
|
|
6
|
-
import { findAppRoot, getModuleDir } from './utils/runtime-paths.js';
|
|
7
|
-
|
|
8
|
-
const __dirname = getModuleDir(import.meta.url);
|
|
9
|
-
// Resolve the repo/app root via the nearest /server folder so this file keeps finding the
|
|
10
|
-
// same top-level .env file from both /server/load-env.js and /dist-server/server/load-env.js.
|
|
11
|
-
const APP_ROOT = findAppRoot(__dirname);
|
|
12
|
-
|
|
13
|
-
try {
|
|
14
|
-
const envPath = path.join(APP_ROOT, '.env');
|
|
15
|
-
const envFile = fs.readFileSync(envPath, 'utf8');
|
|
16
|
-
envFile.split('\n').forEach(line => {
|
|
17
|
-
const trimmedLine = line.trim();
|
|
18
|
-
if (trimmedLine && !trimmedLine.startsWith('#')) {
|
|
19
|
-
const [key, ...valueParts] = trimmedLine.split('=');
|
|
20
|
-
if (key && valueParts.length > 0 && !process.env[key]) {
|
|
21
|
-
process.env[key] = valueParts.join('=').trim();
|
|
22
|
-
}
|
|
23
|
-
}
|
|
24
|
-
});
|
|
25
|
-
} catch (e) {
|
|
26
|
-
console.log('No .env file found or error reading it:', e.message);
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
// Keep the default database in a stable user-level location so rebuilding dist-server
|
|
30
|
-
// never changes where the backend stores auth.db when DATABASE_PATH is not set explicitly.
|
|
31
|
-
const DEFAULT_DATABASE_PATH = path.join(os.homedir(), '.pixcode', 'auth.db');
|
|
32
|
-
|
|
33
|
-
if (!process.env.DATABASE_PATH) {
|
|
34
|
-
process.env.DATABASE_PATH = DEFAULT_DATABASE_PATH;
|
|
35
|
-
}
|
|
1
|
+
// Load environment variables from .env before other imports execute.
|
|
2
|
+
import fs from 'fs';
|
|
3
|
+
import os from 'os';
|
|
4
|
+
import path from 'path';
|
|
5
|
+
|
|
6
|
+
import { findAppRoot, getModuleDir } from './utils/runtime-paths.js';
|
|
7
|
+
|
|
8
|
+
const __dirname = getModuleDir(import.meta.url);
|
|
9
|
+
// Resolve the repo/app root via the nearest /server folder so this file keeps finding the
|
|
10
|
+
// same top-level .env file from both /server/load-env.js and /dist-server/server/load-env.js.
|
|
11
|
+
const APP_ROOT = findAppRoot(__dirname);
|
|
12
|
+
|
|
13
|
+
try {
|
|
14
|
+
const envPath = path.join(APP_ROOT, '.env');
|
|
15
|
+
const envFile = fs.readFileSync(envPath, 'utf8');
|
|
16
|
+
envFile.split('\n').forEach(line => {
|
|
17
|
+
const trimmedLine = line.trim();
|
|
18
|
+
if (trimmedLine && !trimmedLine.startsWith('#')) {
|
|
19
|
+
const [key, ...valueParts] = trimmedLine.split('=');
|
|
20
|
+
if (key && valueParts.length > 0 && !process.env[key]) {
|
|
21
|
+
process.env[key] = valueParts.join('=').trim();
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
});
|
|
25
|
+
} catch (e) {
|
|
26
|
+
console.log('No .env file found or error reading it:', e.message);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// Keep the default database in a stable user-level location so rebuilding dist-server
|
|
30
|
+
// never changes where the backend stores auth.db when DATABASE_PATH is not set explicitly.
|
|
31
|
+
const DEFAULT_DATABASE_PATH = path.join(os.homedir(), '.pixcode', 'auth.db');
|
|
32
|
+
|
|
33
|
+
if (!process.env.DATABASE_PATH) {
|
|
34
|
+
process.env.DATABASE_PATH = DEFAULT_DATABASE_PATH;
|
|
35
|
+
}
|
|
@@ -1,175 +1,190 @@
|
|
|
1
|
-
import jwt from 'jsonwebtoken';
|
|
2
|
-
|
|
3
|
-
import { userDb, appConfigDb, apiKeysDb } from '../database/db.js';
|
|
4
|
-
import { IS_PLATFORM } from '../constants/config.js';
|
|
5
|
-
|
|
6
|
-
// Use env var if set, otherwise auto-generate a unique secret per installation
|
|
1
|
+
import jwt from 'jsonwebtoken';
|
|
2
|
+
|
|
3
|
+
import { userDb, appConfigDb, apiKeysDb } from '../database/db.js';
|
|
4
|
+
import { IS_PLATFORM } from '../constants/config.js';
|
|
5
|
+
|
|
6
|
+
// Use env var if set, otherwise auto-generate a unique secret per installation
|
|
7
7
|
const JWT_SECRET = process.env.JWT_SECRET || appConfigDb.getOrCreateJwtSecret();
|
|
8
8
|
const isPixcodeApiKey = (token) => typeof token === 'string' && (token.startsWith('px_') || token.startsWith('ck_'));
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
if
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
9
|
+
const ADMIN_ROLES = new Set(['owner', 'admin']);
|
|
10
|
+
|
|
11
|
+
// Optional API key middleware
|
|
12
|
+
const validateApiKey = (req, res, next) => {
|
|
13
|
+
// Skip API key validation if not configured
|
|
14
|
+
if (!process.env.API_KEY) {
|
|
15
|
+
return next();
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const apiKey = req.headers['x-api-key'];
|
|
19
|
+
if (apiKey !== process.env.API_KEY) {
|
|
20
|
+
return res.status(401).json({ error: 'Invalid API key' });
|
|
21
|
+
}
|
|
22
|
+
next();
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
// JWT authentication middleware
|
|
25
26
|
const authenticateToken = async (req, res, next) => {
|
|
26
|
-
// Platform mode: use single database user
|
|
27
|
-
if (IS_PLATFORM) {
|
|
28
|
-
try {
|
|
29
|
-
const user = userDb.getFirstUser();
|
|
30
|
-
if (!user) {
|
|
31
|
-
return res.status(500).json({ error: 'Platform mode: No user found in database' });
|
|
32
|
-
}
|
|
33
|
-
req.user = user;
|
|
34
|
-
return next();
|
|
35
|
-
} catch (error) {
|
|
36
|
-
console.error('Platform mode error:', error);
|
|
37
|
-
return res.status(500).json({ error: 'Platform mode: Failed to fetch user' });
|
|
38
|
-
}
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
// Pull credentials from any of the supported transports.
|
|
42
|
-
// - Authorization: Bearer <jwt-or-apikey>
|
|
43
|
-
// - X-API-Key: <apikey> (legacy, kept for /api/agent compatibility)
|
|
44
|
-
// - ?token=<jwt> (EventSource workaround — can't set headers)
|
|
45
|
-
// - ?apiKey=<apikey> (EventSource workaround)
|
|
46
|
-
// Auth-token mode is decided by the prefix: new keys generated by Pixcode
|
|
47
|
-
// start with `px_`; older `ck_` keys remain valid for existing installs.
|
|
48
|
-
const authHeader = req.headers['authorization'];
|
|
49
|
-
const bearerToken = authHeader && authHeader.startsWith('Bearer ') ? authHeader.slice(7).trim() : null;
|
|
50
|
-
const apiKeyHeader = req.headers['x-api-key'];
|
|
51
|
-
const queryToken = typeof req.query.token === 'string' ? req.query.token : null;
|
|
52
|
-
const queryApiKey = typeof req.query.apiKey === 'string' ? req.query.apiKey : null;
|
|
53
|
-
|
|
54
|
-
// Try API-key paths first when the credential is unambiguously an API key.
|
|
55
|
-
const explicitApiKey = apiKeyHeader || queryApiKey
|
|
56
|
-
|| (isPixcodeApiKey(bearerToken) ? bearerToken : null)
|
|
57
|
-
|| (isPixcodeApiKey(queryToken) ? queryToken : null);
|
|
27
|
+
// Platform mode: use single database user
|
|
28
|
+
if (IS_PLATFORM) {
|
|
29
|
+
try {
|
|
30
|
+
const user = userDb.getFirstUser();
|
|
31
|
+
if (!user) {
|
|
32
|
+
return res.status(500).json({ error: 'Platform mode: No user found in database' });
|
|
33
|
+
}
|
|
34
|
+
req.user = user;
|
|
35
|
+
return next();
|
|
36
|
+
} catch (error) {
|
|
37
|
+
console.error('Platform mode error:', error);
|
|
38
|
+
return res.status(500).json({ error: 'Platform mode: Failed to fetch user' });
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Pull credentials from any of the supported transports.
|
|
43
|
+
// - Authorization: Bearer <jwt-or-apikey>
|
|
44
|
+
// - X-API-Key: <apikey> (legacy, kept for /api/agent compatibility)
|
|
45
|
+
// - ?token=<jwt> (EventSource workaround — can't set headers)
|
|
46
|
+
// - ?apiKey=<apikey> (EventSource workaround)
|
|
47
|
+
// Auth-token mode is decided by the prefix: new keys generated by Pixcode
|
|
48
|
+
// start with `px_`; older `ck_` keys remain valid for existing installs.
|
|
49
|
+
const authHeader = req.headers['authorization'];
|
|
50
|
+
const bearerToken = authHeader && authHeader.startsWith('Bearer ') ? authHeader.slice(7).trim() : null;
|
|
51
|
+
const apiKeyHeader = req.headers['x-api-key'];
|
|
52
|
+
const queryToken = typeof req.query.token === 'string' ? req.query.token : null;
|
|
53
|
+
const queryApiKey = typeof req.query.apiKey === 'string' ? req.query.apiKey : null;
|
|
54
|
+
|
|
55
|
+
// Try API-key paths first when the credential is unambiguously an API key.
|
|
56
|
+
const explicitApiKey = apiKeyHeader || queryApiKey
|
|
57
|
+
|| (isPixcodeApiKey(bearerToken) ? bearerToken : null)
|
|
58
|
+
|| (isPixcodeApiKey(queryToken) ? queryToken : null);
|
|
59
|
+
|
|
60
|
+
if (explicitApiKey) {
|
|
61
|
+
try {
|
|
62
|
+
const user = apiKeysDb.validateApiKey(explicitApiKey);
|
|
63
|
+
if (!user) {
|
|
64
|
+
return res.status(401).json({ error: 'Invalid or inactive API key' });
|
|
65
|
+
}
|
|
66
|
+
req.user = user;
|
|
67
|
+
return next();
|
|
68
|
+
} catch (error) {
|
|
69
|
+
console.error('API key validation error:', error);
|
|
70
|
+
return res.status(500).json({ error: 'Authentication backend error' });
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Otherwise fall back to JWT.
|
|
75
|
+
const jwtToken = bearerToken || queryToken;
|
|
76
|
+
if (!jwtToken) {
|
|
77
|
+
return res.status(401).json({ error: 'Access denied. No token provided.' });
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
try {
|
|
81
|
+
const decoded = jwt.verify(jwtToken, JWT_SECRET);
|
|
82
|
+
|
|
83
|
+
// Verify user still exists and is active
|
|
84
|
+
const user = userDb.getUserById(decoded.userId);
|
|
85
|
+
if (!user) {
|
|
86
|
+
return res.status(401).json({ error: 'Invalid token. User not found.' });
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Auto-refresh: if token is past halfway through its lifetime, issue a new one
|
|
90
|
+
if (decoded.exp && decoded.iat) {
|
|
91
|
+
const now = Math.floor(Date.now() / 1000);
|
|
92
|
+
const halfLife = (decoded.exp - decoded.iat) / 2;
|
|
93
|
+
if (now > decoded.iat + halfLife) {
|
|
94
|
+
const newToken = generateToken(user);
|
|
95
|
+
res.setHeader('X-Refreshed-Token', newToken);
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
req.user = user;
|
|
100
|
+
next();
|
|
101
|
+
} catch (error) {
|
|
102
|
+
console.error('Token verification error:', error);
|
|
103
|
+
return res.status(403).json({ error: 'Invalid token' });
|
|
104
|
+
}
|
|
105
|
+
};
|
|
58
106
|
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
if (!user) {
|
|
63
|
-
return res.status(401).json({ error: 'Invalid or inactive API key' });
|
|
64
|
-
}
|
|
65
|
-
req.user = user;
|
|
66
|
-
return next();
|
|
67
|
-
} catch (error) {
|
|
68
|
-
console.error('API key validation error:', error);
|
|
69
|
-
return res.status(500).json({ error: 'Authentication backend error' });
|
|
70
|
-
}
|
|
107
|
+
const requireAdmin = (req, res, next) => {
|
|
108
|
+
if (!req.user) {
|
|
109
|
+
return res.status(401).json({ error: 'Access denied. No authenticated user.' });
|
|
71
110
|
}
|
|
72
111
|
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
if (!jwtToken) {
|
|
76
|
-
return res.status(401).json({ error: 'Access denied. No token provided.' });
|
|
112
|
+
if (!ADMIN_ROLES.has(req.user.role)) {
|
|
113
|
+
return res.status(403).json({ error: 'Admin access required.' });
|
|
77
114
|
}
|
|
78
115
|
|
|
79
|
-
|
|
80
|
-
const decoded = jwt.verify(jwtToken, JWT_SECRET);
|
|
81
|
-
|
|
82
|
-
// Verify user still exists and is active
|
|
83
|
-
const user = userDb.getUserById(decoded.userId);
|
|
84
|
-
if (!user) {
|
|
85
|
-
return res.status(401).json({ error: 'Invalid token. User not found.' });
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
// Auto-refresh: if token is past halfway through its lifetime, issue a new one
|
|
89
|
-
if (decoded.exp && decoded.iat) {
|
|
90
|
-
const now = Math.floor(Date.now() / 1000);
|
|
91
|
-
const halfLife = (decoded.exp - decoded.iat) / 2;
|
|
92
|
-
if (now > decoded.iat + halfLife) {
|
|
93
|
-
const newToken = generateToken(user);
|
|
94
|
-
res.setHeader('X-Refreshed-Token', newToken);
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
req.user = user;
|
|
99
|
-
next();
|
|
100
|
-
} catch (error) {
|
|
101
|
-
console.error('Token verification error:', error);
|
|
102
|
-
return res.status(403).json({ error: 'Invalid token' });
|
|
103
|
-
}
|
|
116
|
+
next();
|
|
104
117
|
};
|
|
105
|
-
|
|
106
|
-
// Generate JWT token
|
|
107
|
-
const generateToken = (user) => {
|
|
108
|
-
return jwt.sign(
|
|
109
|
-
{
|
|
118
|
+
|
|
119
|
+
// Generate JWT token
|
|
120
|
+
const generateToken = (user) => {
|
|
121
|
+
return jwt.sign(
|
|
122
|
+
{
|
|
110
123
|
userId: user.id,
|
|
111
|
-
username: user.username
|
|
124
|
+
username: user.username,
|
|
125
|
+
role: user.role || null,
|
|
112
126
|
},
|
|
113
|
-
JWT_SECRET,
|
|
114
|
-
{ expiresIn: '7d' }
|
|
115
|
-
);
|
|
116
|
-
};
|
|
117
|
-
|
|
118
|
-
// WebSocket authentication function
|
|
119
|
-
const authenticateWebSocket = (token) => {
|
|
120
|
-
// Platform mode: bypass token validation, return first user
|
|
121
|
-
if (IS_PLATFORM) {
|
|
122
|
-
try {
|
|
127
|
+
JWT_SECRET,
|
|
128
|
+
{ expiresIn: '7d' }
|
|
129
|
+
);
|
|
130
|
+
};
|
|
131
|
+
|
|
132
|
+
// WebSocket authentication function
|
|
133
|
+
const authenticateWebSocket = (token) => {
|
|
134
|
+
// Platform mode: bypass token validation, return first user
|
|
135
|
+
if (IS_PLATFORM) {
|
|
136
|
+
try {
|
|
123
137
|
const user = userDb.getFirstUser();
|
|
124
138
|
if (user) {
|
|
125
|
-
return { id: user.id, userId: user.id, username: user.username };
|
|
139
|
+
return { id: user.id, userId: user.id, username: user.username, role: user.role || null };
|
|
126
140
|
}
|
|
127
|
-
return null;
|
|
128
|
-
} catch (error) {
|
|
129
|
-
console.error('Platform mode WebSocket error:', error);
|
|
130
|
-
return null;
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
// Normal OSS validation — accept either an API key (`px_…` or legacy
|
|
135
|
-
// `ck_…`) or a JWT.
|
|
136
|
-
// Mirrors the REST `authenticateToken` middleware so any tool that has
|
|
137
|
-
// a Pixcode API key (CI scripts, the api-tester subagent, the user's own
|
|
138
|
-
// automation, ...) can also open a WebSocket without first exchanging
|
|
139
|
-
// the key for a JWT.
|
|
140
|
-
if (!token) {
|
|
141
|
-
return null;
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
if (isPixcodeApiKey(token)) {
|
|
145
|
-
try {
|
|
141
|
+
return null;
|
|
142
|
+
} catch (error) {
|
|
143
|
+
console.error('Platform mode WebSocket error:', error);
|
|
144
|
+
return null;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// Normal OSS validation — accept either an API key (`px_…` or legacy
|
|
149
|
+
// `ck_…`) or a JWT.
|
|
150
|
+
// Mirrors the REST `authenticateToken` middleware so any tool that has
|
|
151
|
+
// a Pixcode API key (CI scripts, the api-tester subagent, the user's own
|
|
152
|
+
// automation, ...) can also open a WebSocket without first exchanging
|
|
153
|
+
// the key for a JWT.
|
|
154
|
+
if (!token) {
|
|
155
|
+
return null;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
if (isPixcodeApiKey(token)) {
|
|
159
|
+
try {
|
|
146
160
|
const user = apiKeysDb.validateApiKey(token);
|
|
147
161
|
if (!user) return null;
|
|
148
|
-
return { userId: user.id, username: user.username };
|
|
149
|
-
} catch (error) {
|
|
150
|
-
console.error('WebSocket API key validation error:', error);
|
|
151
|
-
return null;
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
try {
|
|
156
|
-
const decoded = jwt.verify(token, JWT_SECRET);
|
|
157
|
-
// Verify user actually exists in database (matches REST authenticateToken behavior)
|
|
158
|
-
const user = userDb.getUserById(decoded.userId);
|
|
159
|
-
if (!user) {
|
|
160
|
-
return null;
|
|
161
|
-
}
|
|
162
|
-
return { userId: user.id, username: user.username };
|
|
163
|
-
} catch (error) {
|
|
164
|
-
console.error('WebSocket token verification error:', error);
|
|
165
|
-
return null;
|
|
166
|
-
}
|
|
167
|
-
};
|
|
168
|
-
|
|
169
|
-
export {
|
|
162
|
+
return { id: user.id, userId: user.id, username: user.username, role: user.role || null };
|
|
163
|
+
} catch (error) {
|
|
164
|
+
console.error('WebSocket API key validation error:', error);
|
|
165
|
+
return null;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
try {
|
|
170
|
+
const decoded = jwt.verify(token, JWT_SECRET);
|
|
171
|
+
// Verify user actually exists in database (matches REST authenticateToken behavior)
|
|
172
|
+
const user = userDb.getUserById(decoded.userId);
|
|
173
|
+
if (!user) {
|
|
174
|
+
return null;
|
|
175
|
+
}
|
|
176
|
+
return { id: user.id, userId: user.id, username: user.username, role: user.role || null };
|
|
177
|
+
} catch (error) {
|
|
178
|
+
console.error('WebSocket token verification error:', error);
|
|
179
|
+
return null;
|
|
180
|
+
}
|
|
181
|
+
};
|
|
182
|
+
|
|
183
|
+
export {
|
|
170
184
|
validateApiKey,
|
|
171
185
|
authenticateToken,
|
|
186
|
+
requireAdmin,
|
|
172
187
|
generateToken,
|
|
173
|
-
authenticateWebSocket,
|
|
174
|
-
JWT_SECRET
|
|
175
|
-
};
|
|
188
|
+
authenticateWebSocket,
|
|
189
|
+
JWT_SECRET
|
|
190
|
+
};
|
|
@@ -1,108 +1,108 @@
|
|
|
1
|
-
// server/modules/orchestration/a2a/adapter-registry.ts
|
|
2
|
-
// In-process registry mapping adapter ids to AbstractA2AAdapter
|
|
3
|
-
// instances. Resolution supports three id forms:
|
|
4
|
-
// - "claude-code" explicit
|
|
5
|
-
// - "skill:<skillId>" first REGISTERED adapter advertising that skill
|
|
6
|
-
// (Map iteration is insertion-ordered per ES spec).
|
|
7
|
-
// - "auto" first registered adapter (deterministic fallback
|
|
8
|
-
// until smarter routing arrives in a later plan)
|
|
9
|
-
|
|
10
|
-
import type { AbstractA2AAdapter } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
|
|
11
|
-
import type { AgentCard } from '@/modules/orchestration/a2a/types.js';
|
|
12
|
-
|
|
13
|
-
interface ResolveAdapterOptions {
|
|
14
|
-
preferredAdapterId?: string;
|
|
15
|
-
preferredProvider?: string;
|
|
16
|
-
preferredSkillId?: string;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
class AdapterRegistry {
|
|
20
|
-
// Map iteration order is insertion-ordered (ES spec); auto and skill: resolution depend on this.
|
|
21
|
-
private readonly byId = new Map<string, AbstractA2AAdapter>();
|
|
22
|
-
|
|
23
|
-
register(adapter: AbstractA2AAdapter): void {
|
|
24
|
-
if (this.byId.has(adapter.id)) {
|
|
25
|
-
throw new Error(`A2A adapter already registered: ${adapter.id}`);
|
|
26
|
-
}
|
|
27
|
-
this.byId.set(adapter.id, adapter);
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
get(id: string): AbstractA2AAdapter | undefined {
|
|
31
|
-
return this.byId.get(id);
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
resolve(idOrSelector: string, options: ResolveAdapterOptions = {}): AbstractA2AAdapter | undefined {
|
|
35
|
-
const normalizedSelector = idOrSelector.trim();
|
|
36
|
-
if (!normalizedSelector) {
|
|
37
|
-
return undefined;
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
if (normalizedSelector === 'auto') {
|
|
41
|
-
return this.pickPreferred(this.list(), options);
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
if (normalizedSelector.startsWith('skill:')) {
|
|
45
|
-
const skill = normalizedSelector.slice('skill:'.length);
|
|
46
|
-
const matches = this.list().filter((adapter) =>
|
|
47
|
-
adapter.agentCard.skills.some((s) => s.id === skill),
|
|
48
|
-
);
|
|
49
|
-
if (matches.length === 0) {
|
|
50
|
-
return undefined;
|
|
51
|
-
}
|
|
52
|
-
return this.pickPreferred(matches, {
|
|
53
|
-
...options,
|
|
54
|
-
preferredSkillId: options.preferredSkillId ?? skill,
|
|
55
|
-
});
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
return this.byId.get(normalizedSelector);
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
list(): AbstractA2AAdapter[] {
|
|
62
|
-
return [...this.byId.values()];
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
agentCards(): AgentCard[] {
|
|
66
|
-
return this.list().map((a) => a.agentCard);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
private pickPreferred(
|
|
70
|
-
adapters: AbstractA2AAdapter[],
|
|
71
|
-
options: ResolveAdapterOptions,
|
|
72
|
-
): AbstractA2AAdapter | undefined {
|
|
73
|
-
const {
|
|
74
|
-
preferredAdapterId,
|
|
75
|
-
preferredProvider,
|
|
76
|
-
preferredSkillId,
|
|
77
|
-
} = options;
|
|
78
|
-
|
|
79
|
-
if (preferredAdapterId) {
|
|
80
|
-
const byAdapterId = adapters.find((adapter) => adapter.id === preferredAdapterId);
|
|
81
|
-
if (byAdapterId) {
|
|
82
|
-
return byAdapterId;
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
if (preferredProvider) {
|
|
87
|
-
const normalizedProvider = preferredProvider.trim().toLowerCase();
|
|
88
|
-
const byProvider = adapters.find((adapter) => adapter.id === normalizedProvider);
|
|
89
|
-
if (byProvider) {
|
|
90
|
-
return byProvider;
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
if (preferredSkillId) {
|
|
95
|
-
const bySkill = adapters.find((adapter) =>
|
|
96
|
-
adapter.agentCard.skills.some((skill) => skill.id === preferredSkillId),
|
|
97
|
-
);
|
|
98
|
-
if (bySkill) {
|
|
99
|
-
return bySkill;
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
return adapters[0];
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
export const adapterRegistry = new AdapterRegistry();
|
|
108
|
-
export type { AdapterRegistry, ResolveAdapterOptions };
|
|
1
|
+
// server/modules/orchestration/a2a/adapter-registry.ts
|
|
2
|
+
// In-process registry mapping adapter ids to AbstractA2AAdapter
|
|
3
|
+
// instances. Resolution supports three id forms:
|
|
4
|
+
// - "claude-code" explicit
|
|
5
|
+
// - "skill:<skillId>" first REGISTERED adapter advertising that skill
|
|
6
|
+
// (Map iteration is insertion-ordered per ES spec).
|
|
7
|
+
// - "auto" first registered adapter (deterministic fallback
|
|
8
|
+
// until smarter routing arrives in a later plan)
|
|
9
|
+
|
|
10
|
+
import type { AbstractA2AAdapter } from '@/modules/orchestration/a2a/adapters/abstract-a2a.adapter.js';
|
|
11
|
+
import type { AgentCard } from '@/modules/orchestration/a2a/types.js';
|
|
12
|
+
|
|
13
|
+
interface ResolveAdapterOptions {
|
|
14
|
+
preferredAdapterId?: string;
|
|
15
|
+
preferredProvider?: string;
|
|
16
|
+
preferredSkillId?: string;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
class AdapterRegistry {
|
|
20
|
+
// Map iteration order is insertion-ordered (ES spec); auto and skill: resolution depend on this.
|
|
21
|
+
private readonly byId = new Map<string, AbstractA2AAdapter>();
|
|
22
|
+
|
|
23
|
+
register(adapter: AbstractA2AAdapter): void {
|
|
24
|
+
if (this.byId.has(adapter.id)) {
|
|
25
|
+
throw new Error(`A2A adapter already registered: ${adapter.id}`);
|
|
26
|
+
}
|
|
27
|
+
this.byId.set(adapter.id, adapter);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
get(id: string): AbstractA2AAdapter | undefined {
|
|
31
|
+
return this.byId.get(id);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
resolve(idOrSelector: string, options: ResolveAdapterOptions = {}): AbstractA2AAdapter | undefined {
|
|
35
|
+
const normalizedSelector = idOrSelector.trim();
|
|
36
|
+
if (!normalizedSelector) {
|
|
37
|
+
return undefined;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if (normalizedSelector === 'auto') {
|
|
41
|
+
return this.pickPreferred(this.list(), options);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
if (normalizedSelector.startsWith('skill:')) {
|
|
45
|
+
const skill = normalizedSelector.slice('skill:'.length);
|
|
46
|
+
const matches = this.list().filter((adapter) =>
|
|
47
|
+
adapter.agentCard.skills.some((s) => s.id === skill),
|
|
48
|
+
);
|
|
49
|
+
if (matches.length === 0) {
|
|
50
|
+
return undefined;
|
|
51
|
+
}
|
|
52
|
+
return this.pickPreferred(matches, {
|
|
53
|
+
...options,
|
|
54
|
+
preferredSkillId: options.preferredSkillId ?? skill,
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return this.byId.get(normalizedSelector);
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
list(): AbstractA2AAdapter[] {
|
|
62
|
+
return [...this.byId.values()];
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
agentCards(): AgentCard[] {
|
|
66
|
+
return this.list().map((a) => a.agentCard);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
private pickPreferred(
|
|
70
|
+
adapters: AbstractA2AAdapter[],
|
|
71
|
+
options: ResolveAdapterOptions,
|
|
72
|
+
): AbstractA2AAdapter | undefined {
|
|
73
|
+
const {
|
|
74
|
+
preferredAdapterId,
|
|
75
|
+
preferredProvider,
|
|
76
|
+
preferredSkillId,
|
|
77
|
+
} = options;
|
|
78
|
+
|
|
79
|
+
if (preferredAdapterId) {
|
|
80
|
+
const byAdapterId = adapters.find((adapter) => adapter.id === preferredAdapterId);
|
|
81
|
+
if (byAdapterId) {
|
|
82
|
+
return byAdapterId;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
if (preferredProvider) {
|
|
87
|
+
const normalizedProvider = preferredProvider.trim().toLowerCase();
|
|
88
|
+
const byProvider = adapters.find((adapter) => adapter.id === normalizedProvider);
|
|
89
|
+
if (byProvider) {
|
|
90
|
+
return byProvider;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (preferredSkillId) {
|
|
95
|
+
const bySkill = adapters.find((adapter) =>
|
|
96
|
+
adapter.agentCard.skills.some((skill) => skill.id === preferredSkillId),
|
|
97
|
+
);
|
|
98
|
+
if (bySkill) {
|
|
99
|
+
return bySkill;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
return adapters[0];
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
export const adapterRegistry = new AdapterRegistry();
|
|
108
|
+
export type { AdapterRegistry, ResolveAdapterOptions };
|