@pixelbyte-software/pixcode 1.51.1 → 1.51.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CODE_OF_CONDUCT.md +41 -41
- package/CONTRIBUTING.md +155 -155
- package/LICENSE +718 -718
- package/README.de.md +169 -169
- package/README.ja.md +167 -167
- package/README.ko.md +167 -167
- package/README.md +419 -419
- package/README.ru.md +169 -169
- package/README.tr.md +298 -298
- package/README.zh-CN.md +167 -167
- package/SECURITY.md +46 -46
- package/dist/api-automation.html +110 -110
- package/dist/api-docs.html +548 -548
- package/dist/assets/{index-DARIZgoD.js → index-17CwxHSZ.js} +185 -185
- package/dist/assets/index-B9N-gfOQ.css +32 -0
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/docs.html +308 -308
- package/dist/favicon.svg +8 -8
- package/dist/features.html +133 -133
- package/dist/generate-icons.js +48 -48
- package/dist/humans.txt +15 -15
- package/dist/icons/codex-white.svg +3 -3
- package/dist/icons/codex.svg +3 -3
- package/dist/icons/cursor-white.svg +11 -11
- package/dist/icons/icon-128x128.svg +9 -9
- package/dist/icons/icon-144x144.svg +9 -9
- package/dist/icons/icon-152x152.svg +9 -9
- package/dist/icons/icon-192x192.svg +9 -9
- package/dist/icons/icon-384x384.svg +9 -9
- package/dist/icons/icon-512x512.svg +9 -9
- package/dist/icons/icon-72x72.svg +9 -9
- package/dist/icons/icon-96x96.svg +9 -9
- package/dist/icons/icon-template.svg +9 -9
- package/dist/icons/qwen-logo.svg +14 -14
- package/dist/index.html +59 -59
- package/dist/landing.html +268 -268
- package/dist/llms-full.txt +119 -119
- package/dist/llms.txt +53 -53
- package/dist/logo.svg +12 -12
- package/dist/manifest.json +60 -60
- package/dist/openapi.yaml +1696 -1696
- package/dist/orchestration.html +125 -125
- package/dist/robots.txt +4 -4
- package/dist/site.css +692 -692
- package/dist/sitemap.xml +51 -51
- package/dist/sw.js +132 -132
- package/dist-server/server/cli.js +96 -96
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +64 -64
- package/dist-server/server/index.js +125 -4
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
- package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
- package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
- package/dist-server/server/modules/orchestration/index.js +1 -0
- package/dist-server/server/modules/orchestration/index.js.map +1 -1
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +17 -17
- package/dist-server/server/routes/live-view.js +46 -46
- package/dist-server/server/services/hermes-gateway.js +310 -0
- package/dist-server/server/services/hermes-gateway.js.map +1 -1
- package/dist-server/server/services/public-api-manifest.js +59 -51
- package/dist-server/server/services/public-api-manifest.js.map +1 -1
- package/package.json +222 -222
- package/scripts/fix-node-pty.js +67 -67
- package/scripts/github/create-v1.38-issues.mjs +351 -351
- package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
- package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
- package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
- package/scripts/smoke/changes-panel-layout.mjs +48 -48
- package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
- package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
- package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
- package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
- package/scripts/smoke/chat-session-state.mjs +19 -19
- package/scripts/smoke/code-editor-theme.mjs +55 -55
- package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
- package/scripts/smoke/command-center-agent-writes.mjs +79 -79
- package/scripts/smoke/command-center-non-git.mjs +46 -46
- package/scripts/smoke/context-packet.mjs +43 -43
- package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
- package/scripts/smoke/daemon-entrypoint.mjs +20 -20
- package/scripts/smoke/default-landing-routing.mjs +33 -33
- package/scripts/smoke/desktop-native-notifications.mjs +30 -30
- package/scripts/smoke/desktop-tray-icon.mjs +33 -33
- package/scripts/smoke/discord-release-workflow.mjs +24 -24
- package/scripts/smoke/git-install-update.mjs +255 -255
- package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
- package/scripts/smoke/hermes-api-install.mjs +56 -56
- package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
- package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
- package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
- package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
- package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
- package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
- package/scripts/smoke/hermes-rest-live.mjs +42 -42
- package/scripts/smoke/hermes-roundtrip.mjs +167 -167
- package/scripts/smoke/hermes-settings-commands.mjs +349 -346
- package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
- package/scripts/smoke/live-view-diagnostics.mjs +53 -53
- package/scripts/smoke/live-view-environment.mjs +92 -92
- package/scripts/smoke/live-view-integration.mjs +450 -450
- package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
- package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
- package/scripts/smoke/model-registry.mjs +36 -36
- package/scripts/smoke/multi-project-ui.mjs +45 -45
- package/scripts/smoke/multi-worker-slots.mjs +42 -42
- package/scripts/smoke/notification-center.mjs +87 -87
- package/scripts/smoke/notification-inapp-preference.mjs +23 -23
- package/scripts/smoke/notification-taxonomy.mjs +58 -58
- package/scripts/smoke/orchestration-api.mjs +172 -172
- package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
- package/scripts/smoke/orchestration-live-run.mjs +176 -176
- package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
- package/scripts/smoke/orchestration-model-sync.mjs +30 -30
- package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
- package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
- package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
- package/scripts/smoke/permission-policy.mjs +50 -50
- package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -164
- package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
- package/scripts/smoke/provider-rest-api.mjs +124 -124
- package/scripts/smoke/provider-selection-status.mjs +52 -52
- package/scripts/smoke/run-state-refresh.mjs +52 -52
- package/scripts/smoke/runtime-manager.mjs +99 -99
- package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
- package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
- package/scripts/smoke/static-root-routing.mjs +21 -21
- package/scripts/smoke/strict-handoff-compact.mjs +60 -60
- package/scripts/smoke/taskmaster-config.mjs +24 -24
- package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
- package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
- package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
- package/scripts/smoke/telegram-control.mjs +242 -242
- package/scripts/smoke/tunnel-persistence.mjs +56 -56
- package/scripts/smoke/update-issue-progress.mjs +69 -69
- package/scripts/smoke/update-ux.mjs +55 -55
- package/scripts/smoke/v138-completion.mjs +132 -132
- package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
- package/scripts/smoke/v138-diagnostics.mjs +63 -63
- package/scripts/smoke/v138-issue-planner.mjs +33 -33
- package/scripts/smoke/v143-remote-control.mjs +76 -76
- package/scripts/smoke/v144-production-loop.mjs +47 -47
- package/scripts/smoke/v145-platformization.mjs +46 -46
- package/scripts/smoke/v146-control-room-ui.mjs +150 -150
- package/scripts/smoke/version-modal-autoshow.mjs +29 -29
- package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
- package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
- package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
- package/scripts/smoke/workflow-templates.mjs +43 -43
- package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
- package/scripts/update-git-install.mjs +293 -293
- package/server/claude-sdk.js +920 -920
- package/server/cli.js +1039 -1039
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +344 -344
- package/server/daemon/manager.js +563 -563
- package/server/daemon-manager.js +964 -964
- package/server/database/db.js +921 -921
- package/server/database/json-store.js +197 -197
- package/server/gemini-cli.js +550 -550
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +131 -3
- package/server/load-env.js +35 -35
- package/server/middleware/auth.js +175 -175
- package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
- package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
- package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
- package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
- package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
- package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
- package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
- package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
- package/server/modules/orchestration/a2a/agent-card.ts +55 -55
- package/server/modules/orchestration/a2a/routes.ts +590 -590
- package/server/modules/orchestration/a2a/task-store.ts +178 -178
- package/server/modules/orchestration/a2a/types.ts +126 -126
- package/server/modules/orchestration/a2a/validator.ts +113 -113
- package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
- package/server/modules/orchestration/index.ts +101 -100
- package/server/modules/orchestration/preview/port-watcher.ts +112 -112
- package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
- package/server/modules/orchestration/preview/types.ts +19 -19
- package/server/modules/orchestration/security/permission-policy.ts +401 -401
- package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
- package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
- package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
- package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
- package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
- package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
- package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
- package/server/modules/orchestration/workflows/context-packet.ts +186 -186
- package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
- package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
- package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
- package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
- package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
- package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
- package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
- package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
- package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
- package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
- package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
- package/server/modules/orchestration/workspace/path-safety.ts +55 -55
- package/server/modules/orchestration/workspace/types.ts +52 -52
- package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
- package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
- package/server/modules/providers/index.ts +2 -2
- package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
- package/server/modules/providers/list/claude/claude.provider.ts +15 -15
- package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
- package/server/modules/providers/list/codex/codex.provider.ts +15 -15
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
- package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
- package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
- package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
- package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
- package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
- package/server/modules/providers/provider.registry.ts +40 -40
- package/server/modules/providers/provider.routes.ts +944 -944
- package/server/modules/providers/services/mcp.service.ts +86 -86
- package/server/modules/providers/services/provider-auth.service.ts +26 -26
- package/server/modules/providers/services/sessions.service.ts +45 -45
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
- package/server/modules/providers/shared/provider-configs.ts +142 -142
- package/server/modules/providers/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +462 -462
- package/server/opencode-cli.js +491 -491
- package/server/opencode-response-handler.js +111 -111
- package/server/projects.js +3008 -3008
- package/server/qwen-code-cli.js +410 -410
- package/server/qwen-response-handler.js +73 -73
- package/server/routes/agent.js +1435 -1435
- package/server/routes/auth.js +159 -159
- package/server/routes/codex.js +20 -20
- package/server/routes/commands.js +570 -570
- package/server/routes/cursor.js +61 -61
- package/server/routes/diagnostics.js +41 -41
- package/server/routes/gemini.js +25 -25
- package/server/routes/git.js +1650 -1650
- package/server/routes/live-view.js +411 -411
- package/server/routes/mcp-utils.js +13 -13
- package/server/routes/messages.js +62 -62
- package/server/routes/network.js +125 -125
- package/server/routes/platformization.js +212 -212
- package/server/routes/plugins.js +320 -320
- package/server/routes/production-agent-loop.js +90 -90
- package/server/routes/projects.js +917 -917
- package/server/routes/public-api.js +34 -34
- package/server/routes/qwen.js +27 -27
- package/server/routes/remote.js +55 -55
- package/server/routes/settings.js +321 -321
- package/server/routes/telegram.js +140 -140
- package/server/routes/user.js +125 -125
- package/server/routes/webhooks.js +63 -63
- package/server/services/control-room.js +102 -102
- package/server/services/diagnostics.js +165 -165
- package/server/services/external-access.js +375 -375
- package/server/services/hermes-gateway.js +1562 -1247
- package/server/services/hermes-install-jobs.js +729 -729
- package/server/services/install-jobs.js +715 -715
- package/server/services/live-view.js +956 -956
- package/server/services/managed-runtimes.js +493 -493
- package/server/services/model-registry.js +144 -144
- package/server/services/notification-orchestrator.js +365 -365
- package/server/services/notification-taxonomy.js +204 -204
- package/server/services/platformization.js +815 -815
- package/server/services/production-agent-loop.js +248 -248
- package/server/services/provider-cli-versions.js +149 -149
- package/server/services/provider-credentials.js +189 -189
- package/server/services/provider-models.js +396 -396
- package/server/services/public-api-manifest.js +190 -182
- package/server/services/remote-connection.js +127 -127
- package/server/services/runtime-manager.js +323 -323
- package/server/services/startup-update.js +234 -234
- package/server/services/telegram/bot.js +331 -331
- package/server/services/telegram/control-center.js +979 -979
- package/server/services/telegram/telegram-http-client.js +151 -151
- package/server/services/telegram/translations.js +340 -340
- package/server/services/vapid-keys.js +36 -36
- package/server/services/webhooks.js +216 -216
- package/server/sessionManager.js +225 -225
- package/server/shared/interfaces.ts +54 -54
- package/server/shared/types.ts +172 -172
- package/server/shared/utils.ts +193 -193
- package/server/tsconfig.json +36 -36
- package/server/utils/colors.js +21 -21
- package/server/utils/commandParser.js +305 -305
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +185 -185
- package/server/utils/port-access.js +209 -209
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +79 -79
- package/shared/modelConstants.js +161 -161
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-DMz0zv6T.css +0 -32
|
@@ -1,175 +1,175 @@
|
|
|
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
|
-
const JWT_SECRET = process.env.JWT_SECRET || appConfigDb.getOrCreateJwtSecret();
|
|
8
|
-
const isPixcodeApiKey = (token) => typeof token === 'string' && (token.startsWith('px_') || token.startsWith('ck_'));
|
|
9
|
-
|
|
10
|
-
// Optional API key middleware
|
|
11
|
-
const validateApiKey = (req, res, next) => {
|
|
12
|
-
// Skip API key validation if not configured
|
|
13
|
-
if (!process.env.API_KEY) {
|
|
14
|
-
return next();
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
const apiKey = req.headers['x-api-key'];
|
|
18
|
-
if (apiKey !== process.env.API_KEY) {
|
|
19
|
-
return res.status(401).json({ error: 'Invalid API key' });
|
|
20
|
-
}
|
|
21
|
-
next();
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
// JWT authentication middleware
|
|
25
|
-
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);
|
|
58
|
-
|
|
59
|
-
if (explicitApiKey) {
|
|
60
|
-
try {
|
|
61
|
-
const user = apiKeysDb.validateApiKey(explicitApiKey);
|
|
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
|
-
}
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
// Otherwise fall back to JWT.
|
|
74
|
-
const jwtToken = bearerToken || queryToken;
|
|
75
|
-
if (!jwtToken) {
|
|
76
|
-
return res.status(401).json({ error: 'Access denied. No token provided.' });
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
try {
|
|
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
|
-
}
|
|
104
|
-
};
|
|
105
|
-
|
|
106
|
-
// Generate JWT token
|
|
107
|
-
const generateToken = (user) => {
|
|
108
|
-
return jwt.sign(
|
|
109
|
-
{
|
|
110
|
-
userId: user.id,
|
|
111
|
-
username: user.username
|
|
112
|
-
},
|
|
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 {
|
|
123
|
-
const user = userDb.getFirstUser();
|
|
124
|
-
if (user) {
|
|
125
|
-
return { id: user.id, userId: user.id, username: user.username };
|
|
126
|
-
}
|
|
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 {
|
|
146
|
-
const user = apiKeysDb.validateApiKey(token);
|
|
147
|
-
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 {
|
|
170
|
-
validateApiKey,
|
|
171
|
-
authenticateToken,
|
|
172
|
-
generateToken,
|
|
173
|
-
authenticateWebSocket,
|
|
174
|
-
JWT_SECRET
|
|
175
|
-
};
|
|
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
|
+
const JWT_SECRET = process.env.JWT_SECRET || appConfigDb.getOrCreateJwtSecret();
|
|
8
|
+
const isPixcodeApiKey = (token) => typeof token === 'string' && (token.startsWith('px_') || token.startsWith('ck_'));
|
|
9
|
+
|
|
10
|
+
// Optional API key middleware
|
|
11
|
+
const validateApiKey = (req, res, next) => {
|
|
12
|
+
// Skip API key validation if not configured
|
|
13
|
+
if (!process.env.API_KEY) {
|
|
14
|
+
return next();
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const apiKey = req.headers['x-api-key'];
|
|
18
|
+
if (apiKey !== process.env.API_KEY) {
|
|
19
|
+
return res.status(401).json({ error: 'Invalid API key' });
|
|
20
|
+
}
|
|
21
|
+
next();
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
// JWT authentication middleware
|
|
25
|
+
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);
|
|
58
|
+
|
|
59
|
+
if (explicitApiKey) {
|
|
60
|
+
try {
|
|
61
|
+
const user = apiKeysDb.validateApiKey(explicitApiKey);
|
|
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
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Otherwise fall back to JWT.
|
|
74
|
+
const jwtToken = bearerToken || queryToken;
|
|
75
|
+
if (!jwtToken) {
|
|
76
|
+
return res.status(401).json({ error: 'Access denied. No token provided.' });
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
try {
|
|
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
|
+
}
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
// Generate JWT token
|
|
107
|
+
const generateToken = (user) => {
|
|
108
|
+
return jwt.sign(
|
|
109
|
+
{
|
|
110
|
+
userId: user.id,
|
|
111
|
+
username: user.username
|
|
112
|
+
},
|
|
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 {
|
|
123
|
+
const user = userDb.getFirstUser();
|
|
124
|
+
if (user) {
|
|
125
|
+
return { id: user.id, userId: user.id, username: user.username };
|
|
126
|
+
}
|
|
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 {
|
|
146
|
+
const user = apiKeysDb.validateApiKey(token);
|
|
147
|
+
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 {
|
|
170
|
+
validateApiKey,
|
|
171
|
+
authenticateToken,
|
|
172
|
+
generateToken,
|
|
173
|
+
authenticateWebSocket,
|
|
174
|
+
JWT_SECRET
|
|
175
|
+
};
|
|
@@ -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 };
|