@pixelbyte-software/pixcode 1.30.1 → 1.31.0
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/LICENSE +718 -718
- package/README.de.md +248 -248
- package/README.ja.md +240 -240
- package/README.ko.md +240 -240
- package/README.md +295 -285
- package/README.ru.md +248 -248
- package/README.tr.md +250 -250
- package/README.zh-CN.md +240 -240
- package/dist/api-docs.html +879 -879
- package/dist/assets/index-BRRJ47XQ.css +32 -0
- package/dist/assets/index-EQohwyiC.js +837 -0
- package/dist/clear-cache.html +85 -85
- package/dist/convert-icons.md +52 -52
- package/dist/favicon.png +0 -0
- package/dist/favicon.svg +7 -8
- package/dist/generate-icons.js +48 -48
- 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.png +0 -0
- package/dist/icons/icon-128x128.svg +9 -12
- package/dist/icons/icon-144x144.png +0 -0
- package/dist/icons/icon-144x144.svg +9 -12
- package/dist/icons/icon-152x152.png +0 -0
- package/dist/icons/icon-152x152.svg +9 -12
- package/dist/icons/icon-192x192.png +0 -0
- package/dist/icons/icon-192x192.svg +9 -12
- package/dist/icons/icon-384x384.png +0 -0
- package/dist/icons/icon-384x384.svg +9 -12
- package/dist/icons/icon-512x512.png +0 -0
- package/dist/icons/icon-512x512.svg +9 -12
- package/dist/icons/icon-72x72.png +0 -0
- package/dist/icons/icon-72x72.svg +9 -12
- package/dist/icons/icon-96x96.png +0 -0
- package/dist/icons/icon-96x96.svg +9 -12
- package/dist/icons/icon-template.svg +9 -12
- package/dist/icons/qwen-ai-icon.png +0 -0
- package/dist/index.html +59 -49
- package/dist/logo.png +0 -0
- package/dist/logo.svg +11 -16
- package/dist/manifest.json +60 -60
- package/dist/sw.js +124 -124
- package/dist-server/server/cli.js +100 -97
- package/dist-server/server/cli.js.map +1 -1
- package/dist-server/server/daemon/manager.js +33 -33
- package/dist-server/server/daemon-manager.js +62 -62
- package/dist-server/server/database/db.js +114 -22
- package/dist-server/server/database/db.js.map +1 -1
- package/dist-server/server/database/schema.js +122 -89
- package/dist-server/server/database/schema.js.map +1 -1
- package/dist-server/server/gemini-cli.js +6 -1
- package/dist-server/server/gemini-cli.js.map +1 -1
- package/dist-server/server/index.js +234 -65
- package/dist-server/server/index.js.map +1 -1
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +29 -2
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +22 -2
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +2 -2
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +14 -2
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -1
- package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js +132 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js +87 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js +201 -0
- package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/qwen/qwen.provider.js +19 -0
- package/dist-server/server/modules/providers/list/qwen/qwen.provider.js.map +1 -0
- package/dist-server/server/modules/providers/provider.registry.js +2 -0
- package/dist-server/server/modules/providers/provider.registry.js.map +1 -1
- package/dist-server/server/modules/providers/provider.routes.js +310 -1
- package/dist-server/server/modules/providers/provider.routes.js.map +1 -1
- package/dist-server/server/projects.js +197 -6
- package/dist-server/server/projects.js.map +1 -1
- package/dist-server/server/qwen-code-cli.js +350 -0
- package/dist-server/server/qwen-code-cli.js.map +1 -0
- package/dist-server/server/qwen-response-handler.js +70 -0
- package/dist-server/server/qwen-response-handler.js.map +1 -0
- package/dist-server/server/routes/commands.js +25 -25
- package/dist-server/server/routes/git.js +17 -17
- package/dist-server/server/routes/network.js +116 -0
- package/dist-server/server/routes/network.js.map +1 -0
- package/dist-server/server/routes/projects.js +43 -0
- package/dist-server/server/routes/projects.js.map +1 -1
- package/dist-server/server/routes/qwen.js +23 -0
- package/dist-server/server/routes/qwen.js.map +1 -0
- package/dist-server/server/routes/taskmaster.js +419 -419
- package/dist-server/server/routes/telegram.js +119 -0
- package/dist-server/server/routes/telegram.js.map +1 -0
- package/dist-server/server/services/external-access.js +228 -0
- package/dist-server/server/services/external-access.js.map +1 -0
- package/dist-server/server/services/install-jobs.js +394 -0
- package/dist-server/server/services/install-jobs.js.map +1 -0
- package/dist-server/server/services/notification-orchestrator.js +19 -5
- package/dist-server/server/services/notification-orchestrator.js.map +1 -1
- package/dist-server/server/services/provider-credentials.js +154 -0
- package/dist-server/server/services/provider-credentials.js.map +1 -0
- package/dist-server/server/services/provider-models.js +218 -0
- package/dist-server/server/services/provider-models.js.map +1 -0
- package/dist-server/server/services/telegram/bot.js +259 -0
- package/dist-server/server/services/telegram/bot.js.map +1 -0
- package/dist-server/server/services/telegram/translations.js +160 -0
- package/dist-server/server/services/telegram/translations.js.map +1 -0
- package/dist-server/server/utils/port-access.js +196 -0
- package/dist-server/server/utils/port-access.js.map +1 -0
- package/dist-server/shared/modelConstants.js +18 -0
- package/dist-server/shared/modelConstants.js.map +1 -1
- package/package.json +177 -168
- package/scripts/fix-node-pty.js +67 -67
- package/server/claude-sdk.js +834 -834
- package/server/cli.js +940 -937
- package/server/constants/config.js +4 -4
- package/server/cursor-cli.js +342 -342
- package/server/daemon/manager.js +564 -564
- package/server/daemon-manager.js +920 -920
- package/server/database/db.js +696 -593
- package/server/database/schema.js +138 -102
- package/server/gemini-cli.js +475 -469
- package/server/gemini-response-handler.js +79 -79
- package/server/index.js +2730 -2557
- package/server/load-env.js +34 -34
- package/server/middleware/auth.js +132 -132
- package/server/modules/providers/list/claude/claude-auth.provider.ts +145 -123
- 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 +115 -100
- 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 +143 -143
- 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 +163 -151
- 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/qwen/qwen-auth.provider.ts +145 -0
- package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -0
- package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +218 -0
- package/server/modules/providers/list/qwen/qwen.provider.ts +21 -0
- package/server/modules/providers/provider.registry.ts +38 -36
- package/server/modules/providers/provider.routes.ts +583 -217
- package/server/modules/providers/services/mcp.service.ts +94 -94
- 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/tests/mcp.test.ts +293 -293
- package/server/openai-codex.js +426 -426
- package/server/projects.js +2993 -2792
- package/server/qwen-code-cli.js +392 -0
- package/server/qwen-response-handler.js +73 -0
- package/server/routes/agent.js +1245 -1245
- package/server/routes/auth.js +134 -134
- package/server/routes/codex.js +19 -19
- package/server/routes/commands.js +554 -554
- package/server/routes/cursor.js +52 -52
- package/server/routes/gemini.js +24 -24
- package/server/routes/git.js +1488 -1488
- package/server/routes/mcp-utils.js +31 -31
- package/server/routes/messages.js +61 -61
- package/server/routes/network.js +128 -0
- package/server/routes/plugins.js +307 -307
- package/server/routes/projects.js +675 -627
- package/server/routes/qwen.js +27 -0
- package/server/routes/settings.js +286 -286
- package/server/routes/taskmaster.js +1471 -1471
- package/server/routes/telegram.js +125 -0
- package/server/routes/user.js +123 -123
- package/server/services/external-access.js +240 -0
- package/server/services/install-jobs.js +410 -0
- package/server/services/notification-orchestrator.js +242 -227
- package/server/services/provider-credentials.js +151 -0
- package/server/services/provider-models.js +225 -0
- package/server/services/telegram/bot.js +280 -0
- package/server/services/telegram/translations.js +170 -0
- package/server/services/vapid-keys.js +35 -35
- 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 +303 -303
- package/server/utils/frontmatter.js +18 -18
- package/server/utils/gitConfig.js +34 -34
- package/server/utils/mcp-detector.js +147 -147
- package/server/utils/plugin-loader.js +457 -457
- package/server/utils/plugin-process-manager.js +184 -184
- package/server/utils/port-access.js +209 -0
- package/server/utils/runtime-paths.js +37 -37
- package/server/utils/taskmaster-websocket.js +128 -128
- package/server/utils/url-detection.js +71 -71
- package/server/vite-daemon.js +78 -78
- package/shared/modelConstants.js +117 -97
- package/shared/networkHosts.js +22 -22
- package/dist/assets/index-C2c9QNwK.css +0 -32
- package/dist/assets/index-DyXDZED-.js +0 -1277
- package/dist-server/server/routes/cli-auth.js +0 -25
- package/dist-server/server/routes/cli-auth.js.map +0 -1
- package/server/routes/cli-auth.js +0 -27
|
@@ -1,217 +1,583 @@
|
|
|
1
|
-
import express, { type Request, type Response } from 'express';
|
|
2
|
-
|
|
3
|
-
import { providerAuthService } from '@/modules/providers/services/provider-auth.service.js';
|
|
4
|
-
import { providerMcpService } from '@/modules/providers/services/mcp.service.js';
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
};
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
const
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
)
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
)
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
1
|
+
import express, { type Request, type Response } from 'express';
|
|
2
|
+
|
|
3
|
+
import { providerAuthService } from '@/modules/providers/services/provider-auth.service.js';
|
|
4
|
+
import { providerMcpService } from '@/modules/providers/services/mcp.service.js';
|
|
5
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
6
|
+
// @ts-ignore — plain-JS service, typed via inference
|
|
7
|
+
import {
|
|
8
|
+
applyProviderCredentialsToEnv,
|
|
9
|
+
listProviderCredentialSummaries,
|
|
10
|
+
setProviderCredentials,
|
|
11
|
+
PROVIDER_ENV_VARS,
|
|
12
|
+
} from '@/services/provider-credentials.js';
|
|
13
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
14
|
+
// @ts-ignore — plain-JS service
|
|
15
|
+
import { getProviderModels, clearProviderModelCache } from '@/services/provider-models.js';
|
|
16
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
17
|
+
// @ts-ignore — plain-JS service
|
|
18
|
+
import {
|
|
19
|
+
createInstallJob,
|
|
20
|
+
getInstallJob,
|
|
21
|
+
cancelInstallJob,
|
|
22
|
+
snapshotDonePayload,
|
|
23
|
+
} from '@/services/install-jobs.js';
|
|
24
|
+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
25
|
+
// @ts-ignore — plain-JS shared module
|
|
26
|
+
import {
|
|
27
|
+
CLAUDE_MODELS,
|
|
28
|
+
CODEX_MODELS,
|
|
29
|
+
GEMINI_MODELS,
|
|
30
|
+
QWEN_MODELS,
|
|
31
|
+
CURSOR_MODELS,
|
|
32
|
+
} from '../../../shared/modelConstants.js';
|
|
33
|
+
|
|
34
|
+
const STATIC_MODELS_BY_PROVIDER: Record<LLMProvider, Array<{ value: string; label: string }>> = {
|
|
35
|
+
claude: CLAUDE_MODELS.OPTIONS,
|
|
36
|
+
codex: CODEX_MODELS.OPTIONS,
|
|
37
|
+
cursor: CURSOR_MODELS.OPTIONS,
|
|
38
|
+
gemini: GEMINI_MODELS.OPTIONS,
|
|
39
|
+
qwen: QWEN_MODELS.OPTIONS,
|
|
40
|
+
};
|
|
41
|
+
import type { LLMProvider, McpScope, McpTransport, UpsertProviderMcpServerInput } from '@/shared/types.js';
|
|
42
|
+
import { AppError, asyncHandler, createApiSuccessResponse } from '@/shared/utils.js';
|
|
43
|
+
import http from 'node:http';
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* npm-global install command per provider. Used by POST
|
|
47
|
+
* /api/providers/:p/install to run the install directly from Pixcode so
|
|
48
|
+
* users don't have to drop into a shell just to get a CLI on the host.
|
|
49
|
+
* Cursor uses its own install script, not npm.
|
|
50
|
+
*/
|
|
51
|
+
/**
|
|
52
|
+
* npm package name per provider. The in-app installer drops these into
|
|
53
|
+
* ~/.pixcode/cli-bin/ as LOCAL deps (no -g, no sudo). A sibling string
|
|
54
|
+
* for display ("npm install -g …") is surfaced in the UI so users who
|
|
55
|
+
* prefer to install manually still see a recognizable command.
|
|
56
|
+
*/
|
|
57
|
+
const PROVIDER_INSTALL_PACKAGES: Record<LLMProvider, string | null> = {
|
|
58
|
+
claude: '@anthropic-ai/claude-code',
|
|
59
|
+
codex: '@openai/codex',
|
|
60
|
+
gemini: '@google/gemini-cli',
|
|
61
|
+
qwen: '@qwen-code/qwen-code',
|
|
62
|
+
// Cursor ships via a bash script hosted at cursor.com; safer to ask
|
|
63
|
+
// users to run it themselves than to pipe-to-bash from our server.
|
|
64
|
+
cursor: null,
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
const PROVIDER_INSTALL_COMMANDS: Record<LLMProvider, string | null> = {
|
|
68
|
+
claude: 'npm install -g @anthropic-ai/claude-code',
|
|
69
|
+
codex: 'npm install -g @openai/codex',
|
|
70
|
+
gemini: 'npm install -g @google/gemini-cli',
|
|
71
|
+
qwen: 'npm install -g @qwen-code/qwen-code',
|
|
72
|
+
cursor: null,
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
const router = express.Router();
|
|
76
|
+
|
|
77
|
+
const readPathParam = (value: unknown, name: string): string => {
|
|
78
|
+
if (typeof value === 'string') {
|
|
79
|
+
return value;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (Array.isArray(value) && typeof value[0] === 'string') {
|
|
83
|
+
return value[0];
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
throw new AppError(`${name} path parameter is invalid.`, {
|
|
87
|
+
code: 'INVALID_PATH_PARAMETER',
|
|
88
|
+
statusCode: 400,
|
|
89
|
+
});
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
const normalizeProviderParam = (value: unknown): string =>
|
|
93
|
+
readPathParam(value, 'provider').trim().toLowerCase();
|
|
94
|
+
|
|
95
|
+
const readOptionalQueryString = (value: unknown): string | undefined => {
|
|
96
|
+
if (typeof value !== 'string') {
|
|
97
|
+
return undefined;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
const normalized = value.trim();
|
|
101
|
+
return normalized.length > 0 ? normalized : undefined;
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
const parseMcpScope = (value: unknown): McpScope | undefined => {
|
|
105
|
+
if (value === undefined) {
|
|
106
|
+
return undefined;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
const normalized = readOptionalQueryString(value);
|
|
110
|
+
if (!normalized) {
|
|
111
|
+
return undefined;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
if (normalized === 'user' || normalized === 'local' || normalized === 'project') {
|
|
115
|
+
return normalized;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
throw new AppError(`Unsupported MCP scope "${normalized}".`, {
|
|
119
|
+
code: 'INVALID_MCP_SCOPE',
|
|
120
|
+
statusCode: 400,
|
|
121
|
+
});
|
|
122
|
+
};
|
|
123
|
+
|
|
124
|
+
const parseMcpTransport = (value: unknown): McpTransport => {
|
|
125
|
+
const normalized = readOptionalQueryString(value);
|
|
126
|
+
if (!normalized) {
|
|
127
|
+
throw new AppError('transport is required.', {
|
|
128
|
+
code: 'MCP_TRANSPORT_REQUIRED',
|
|
129
|
+
statusCode: 400,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (normalized === 'stdio' || normalized === 'http' || normalized === 'sse') {
|
|
134
|
+
return normalized;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
throw new AppError(`Unsupported MCP transport "${normalized}".`, {
|
|
138
|
+
code: 'INVALID_MCP_TRANSPORT',
|
|
139
|
+
statusCode: 400,
|
|
140
|
+
});
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
const parseMcpUpsertPayload = (payload: unknown): UpsertProviderMcpServerInput => {
|
|
144
|
+
if (!payload || typeof payload !== 'object') {
|
|
145
|
+
throw new AppError('Request body must be an object.', {
|
|
146
|
+
code: 'INVALID_REQUEST_BODY',
|
|
147
|
+
statusCode: 400,
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const body = payload as Record<string, unknown>;
|
|
152
|
+
const name = readOptionalQueryString(body.name);
|
|
153
|
+
if (!name) {
|
|
154
|
+
throw new AppError('name is required.', {
|
|
155
|
+
code: 'MCP_NAME_REQUIRED',
|
|
156
|
+
statusCode: 400,
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
const transport = parseMcpTransport(body.transport);
|
|
161
|
+
const scope = parseMcpScope(body.scope);
|
|
162
|
+
const workspacePath = readOptionalQueryString(body.workspacePath);
|
|
163
|
+
|
|
164
|
+
return {
|
|
165
|
+
name,
|
|
166
|
+
transport,
|
|
167
|
+
scope,
|
|
168
|
+
workspacePath,
|
|
169
|
+
command: readOptionalQueryString(body.command),
|
|
170
|
+
args: Array.isArray(body.args) ? body.args.filter((entry): entry is string => typeof entry === 'string') : undefined,
|
|
171
|
+
env: typeof body.env === 'object' && body.env !== null
|
|
172
|
+
? Object.fromEntries(
|
|
173
|
+
Object.entries(body.env as Record<string, unknown>).filter(
|
|
174
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
175
|
+
),
|
|
176
|
+
)
|
|
177
|
+
: undefined,
|
|
178
|
+
cwd: readOptionalQueryString(body.cwd),
|
|
179
|
+
url: readOptionalQueryString(body.url),
|
|
180
|
+
headers: typeof body.headers === 'object' && body.headers !== null
|
|
181
|
+
? Object.fromEntries(
|
|
182
|
+
Object.entries(body.headers as Record<string, unknown>).filter(
|
|
183
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
184
|
+
),
|
|
185
|
+
)
|
|
186
|
+
: undefined,
|
|
187
|
+
envVars: Array.isArray(body.envVars)
|
|
188
|
+
? body.envVars.filter((entry): entry is string => typeof entry === 'string')
|
|
189
|
+
: undefined,
|
|
190
|
+
bearerTokenEnvVar: readOptionalQueryString(body.bearerTokenEnvVar),
|
|
191
|
+
envHttpHeaders: typeof body.envHttpHeaders === 'object' && body.envHttpHeaders !== null
|
|
192
|
+
? Object.fromEntries(
|
|
193
|
+
Object.entries(body.envHttpHeaders as Record<string, unknown>).filter(
|
|
194
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
195
|
+
),
|
|
196
|
+
)
|
|
197
|
+
: undefined,
|
|
198
|
+
};
|
|
199
|
+
};
|
|
200
|
+
|
|
201
|
+
const parseProvider = (value: unknown): LLMProvider => {
|
|
202
|
+
const normalized = normalizeProviderParam(value);
|
|
203
|
+
if (
|
|
204
|
+
normalized === 'claude' ||
|
|
205
|
+
normalized === 'codex' ||
|
|
206
|
+
normalized === 'cursor' ||
|
|
207
|
+
normalized === 'gemini' ||
|
|
208
|
+
normalized === 'qwen'
|
|
209
|
+
) {
|
|
210
|
+
return normalized;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
throw new AppError(`Unsupported provider "${normalized}".`, {
|
|
214
|
+
code: 'UNSUPPORTED_PROVIDER',
|
|
215
|
+
statusCode: 400,
|
|
216
|
+
});
|
|
217
|
+
};
|
|
218
|
+
|
|
219
|
+
router.get(
|
|
220
|
+
'/:provider/auth/status',
|
|
221
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
222
|
+
const provider = parseProvider(req.params.provider);
|
|
223
|
+
const status = await providerAuthService.getProviderAuthStatus(provider);
|
|
224
|
+
res.json(createApiSuccessResponse(status));
|
|
225
|
+
}),
|
|
226
|
+
);
|
|
227
|
+
|
|
228
|
+
router.get(
|
|
229
|
+
'/:provider/mcp/servers',
|
|
230
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
231
|
+
const provider = parseProvider(req.params.provider);
|
|
232
|
+
const workspacePath = readOptionalQueryString(req.query.workspacePath);
|
|
233
|
+
const scope = parseMcpScope(req.query.scope);
|
|
234
|
+
|
|
235
|
+
if (scope) {
|
|
236
|
+
const servers = await providerMcpService.listProviderMcpServersForScope(provider, scope, { workspacePath });
|
|
237
|
+
res.json(createApiSuccessResponse({ provider, scope, servers }));
|
|
238
|
+
return;
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
const groupedServers = await providerMcpService.listProviderMcpServers(provider, { workspacePath });
|
|
242
|
+
res.json(createApiSuccessResponse({ provider, scopes: groupedServers }));
|
|
243
|
+
}),
|
|
244
|
+
);
|
|
245
|
+
|
|
246
|
+
router.post(
|
|
247
|
+
'/:provider/mcp/servers',
|
|
248
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
249
|
+
const provider = parseProvider(req.params.provider);
|
|
250
|
+
const payload = parseMcpUpsertPayload(req.body);
|
|
251
|
+
const server = await providerMcpService.upsertProviderMcpServer(provider, payload);
|
|
252
|
+
res.status(201).json(createApiSuccessResponse({ server }));
|
|
253
|
+
}),
|
|
254
|
+
);
|
|
255
|
+
|
|
256
|
+
router.delete(
|
|
257
|
+
'/:provider/mcp/servers/:name',
|
|
258
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
259
|
+
const provider = parseProvider(req.params.provider);
|
|
260
|
+
const scope = parseMcpScope(req.query.scope);
|
|
261
|
+
const workspacePath = readOptionalQueryString(req.query.workspacePath);
|
|
262
|
+
const result = await providerMcpService.removeProviderMcpServer(provider, {
|
|
263
|
+
name: readPathParam(req.params.name, 'name'),
|
|
264
|
+
scope,
|
|
265
|
+
workspacePath,
|
|
266
|
+
});
|
|
267
|
+
res.json(createApiSuccessResponse(result));
|
|
268
|
+
}),
|
|
269
|
+
);
|
|
270
|
+
|
|
271
|
+
/**
|
|
272
|
+
* GET /api/providers/credentials
|
|
273
|
+
* Summary for every provider (hasKey + baseUrl + updatedAt). Used by the
|
|
274
|
+
* Settings UI to pre-fill the "API Key" tab.
|
|
275
|
+
*/
|
|
276
|
+
router.get(
|
|
277
|
+
'/credentials',
|
|
278
|
+
asyncHandler(async (_req: Request, res: Response) => {
|
|
279
|
+
const summaries = await listProviderCredentialSummaries();
|
|
280
|
+
res.json(createApiSuccessResponse(summaries));
|
|
281
|
+
}),
|
|
282
|
+
);
|
|
283
|
+
|
|
284
|
+
/**
|
|
285
|
+
* POST /api/providers/:provider/auth/api-key
|
|
286
|
+
* Body: { apiKey: string, baseUrl?: string }. Stores the credentials in
|
|
287
|
+
* ~/.pixcode/provider-credentials.json and applies them to process.env
|
|
288
|
+
* so the next CLI spawn/SDK call picks them up. Empty apiKey clears.
|
|
289
|
+
*/
|
|
290
|
+
router.post(
|
|
291
|
+
'/:provider/auth/api-key',
|
|
292
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
293
|
+
const provider = parseProvider(req.params.provider);
|
|
294
|
+
if (!(provider in PROVIDER_ENV_VARS)) {
|
|
295
|
+
throw new AppError(`Provider "${provider}" does not accept API-key auth.`, {
|
|
296
|
+
code: 'PROVIDER_NO_API_KEY',
|
|
297
|
+
statusCode: 400,
|
|
298
|
+
});
|
|
299
|
+
}
|
|
300
|
+
const body = (req.body ?? {}) as Record<string, unknown>;
|
|
301
|
+
const apiKey = typeof body.apiKey === 'string' ? body.apiKey : '';
|
|
302
|
+
const baseUrl = typeof body.baseUrl === 'string' ? body.baseUrl : '';
|
|
303
|
+
|
|
304
|
+
await setProviderCredentials(provider, { apiKey, baseUrl });
|
|
305
|
+
await applyProviderCredentialsToEnv(provider);
|
|
306
|
+
|
|
307
|
+
res.json(createApiSuccessResponse({ provider, stored: Boolean(apiKey.trim()) }));
|
|
308
|
+
}),
|
|
309
|
+
);
|
|
310
|
+
|
|
311
|
+
/**
|
|
312
|
+
* POST /api/providers/:provider/oauth-paste
|
|
313
|
+
* Body: { callbackUrl: string }.
|
|
314
|
+
*
|
|
315
|
+
* When the CLI starts an OAuth flow it spins up a local HTTP server on
|
|
316
|
+
* 127.0.0.1:<PORT> and expects the OAuth provider to redirect the user's
|
|
317
|
+
* browser to `http://127.0.0.1:<PORT>/callback?code=...`. On remote VPS
|
|
318
|
+
* setups that redirect hits the user's laptop localhost (which has nothing
|
|
319
|
+
* listening), not the server running the CLI. This endpoint is the escape
|
|
320
|
+
* hatch: the user copies the dead callback URL from their browser and
|
|
321
|
+
* posts it here; we parse out the port + code and forward the original
|
|
322
|
+
* GET to the VPS-side 127.0.0.1:PORT so the CLI's local handler completes
|
|
323
|
+
* the token exchange.
|
|
324
|
+
*/
|
|
325
|
+
router.post(
|
|
326
|
+
'/:provider/oauth-paste',
|
|
327
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
328
|
+
parseProvider(req.params.provider); // validate id but we don't use it further
|
|
329
|
+
const body = (req.body ?? {}) as Record<string, unknown>;
|
|
330
|
+
const raw = typeof body.callbackUrl === 'string' ? body.callbackUrl.trim() : '';
|
|
331
|
+
if (!raw) {
|
|
332
|
+
throw new AppError('callbackUrl is required.', {
|
|
333
|
+
code: 'OAUTH_PASTE_URL_REQUIRED',
|
|
334
|
+
statusCode: 400,
|
|
335
|
+
});
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
let parsed: URL;
|
|
339
|
+
try {
|
|
340
|
+
parsed = new URL(raw);
|
|
341
|
+
} catch {
|
|
342
|
+
throw new AppError('callbackUrl must be a valid URL.', {
|
|
343
|
+
code: 'OAUTH_PASTE_URL_INVALID',
|
|
344
|
+
statusCode: 400,
|
|
345
|
+
});
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
// Accept localhost / 127.0.0.1 callbacks — reject anything else so we
|
|
349
|
+
// never proxy arbitrary outbound requests on behalf of a user.
|
|
350
|
+
const host = parsed.hostname;
|
|
351
|
+
if (host !== '127.0.0.1' && host !== 'localhost' && host !== '::1') {
|
|
352
|
+
throw new AppError('Only local CLI callback URLs are accepted.', {
|
|
353
|
+
code: 'OAUTH_PASTE_URL_NOT_LOCAL',
|
|
354
|
+
statusCode: 400,
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
const port = Number(parsed.port);
|
|
359
|
+
if (!port || port < 1 || port > 65535) {
|
|
360
|
+
throw new AppError('Callback URL must include the CLI callback port.', {
|
|
361
|
+
code: 'OAUTH_PASTE_PORT_INVALID',
|
|
362
|
+
statusCode: 400,
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
const pathAndQuery = parsed.pathname + parsed.search;
|
|
367
|
+
await new Promise<void>((resolve, reject) => {
|
|
368
|
+
const forwardReq = http.request(
|
|
369
|
+
{
|
|
370
|
+
host: '127.0.0.1',
|
|
371
|
+
port,
|
|
372
|
+
method: 'GET',
|
|
373
|
+
path: pathAndQuery,
|
|
374
|
+
timeout: 10000,
|
|
375
|
+
},
|
|
376
|
+
(forwardRes) => {
|
|
377
|
+
forwardRes.resume(); // drain
|
|
378
|
+
forwardRes.on('end', () => resolve());
|
|
379
|
+
},
|
|
380
|
+
);
|
|
381
|
+
forwardReq.on('timeout', () => {
|
|
382
|
+
forwardReq.destroy(new Error('CLI callback server did not respond within 10s'));
|
|
383
|
+
});
|
|
384
|
+
forwardReq.on('error', (err) => reject(err));
|
|
385
|
+
forwardReq.end();
|
|
386
|
+
});
|
|
387
|
+
|
|
388
|
+
res.json(createApiSuccessResponse({ forwarded: true, port }));
|
|
389
|
+
}),
|
|
390
|
+
);
|
|
391
|
+
|
|
392
|
+
/**
|
|
393
|
+
* GET /api/providers/:provider/models?refresh=1
|
|
394
|
+
* Merged model catalog: hardcoded defaults + live API discovery when an
|
|
395
|
+
* API key is configured. Ships a stable baseline so dropdowns never sit
|
|
396
|
+
* empty, then overlays whatever the upstream API reports so users get
|
|
397
|
+
* new models without a Pixcode release. 6-hour cache; pass `refresh=1`
|
|
398
|
+
* to force an upstream hit.
|
|
399
|
+
*/
|
|
400
|
+
router.get(
|
|
401
|
+
'/:provider/models',
|
|
402
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
403
|
+
const provider = parseProvider(req.params.provider);
|
|
404
|
+
const forceRefresh = String(req.query.refresh || '').toLowerCase() === '1';
|
|
405
|
+
const result = await getProviderModels(provider, {
|
|
406
|
+
forceRefresh,
|
|
407
|
+
staticList: STATIC_MODELS_BY_PROVIDER[provider] ?? [],
|
|
408
|
+
});
|
|
409
|
+
res.json(createApiSuccessResponse(result));
|
|
410
|
+
}),
|
|
411
|
+
);
|
|
412
|
+
|
|
413
|
+
router.delete(
|
|
414
|
+
'/:provider/models/cache',
|
|
415
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
416
|
+
const provider = parseProvider(req.params.provider);
|
|
417
|
+
await clearProviderModelCache(provider);
|
|
418
|
+
res.json(createApiSuccessResponse({ cleared: true, provider }));
|
|
419
|
+
}),
|
|
420
|
+
);
|
|
421
|
+
|
|
422
|
+
/**
|
|
423
|
+
* POST /api/providers/:provider/install
|
|
424
|
+
* Kicks off the install in the background and immediately returns
|
|
425
|
+
* `{ jobId }`. The actual log stream is fetched separately via
|
|
426
|
+
* GET /install/:jobId/stream (EventSource). This split solves the
|
|
427
|
+
* "Client disconnected before install finished" class of errors,
|
|
428
|
+
* where a single long-lived POST SSE would get torn down by dev
|
|
429
|
+
* proxies, service-worker reloads, or Vite HMR and short-circuit
|
|
430
|
+
* an in-flight install. The child now outlives the request.
|
|
431
|
+
*/
|
|
432
|
+
router.post(
|
|
433
|
+
'/:provider/install',
|
|
434
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
435
|
+
const parsed = parseProvider(req.params.provider);
|
|
436
|
+
const packageName = PROVIDER_INSTALL_PACKAGES[parsed];
|
|
437
|
+
const installCmd = PROVIDER_INSTALL_COMMANDS[parsed];
|
|
438
|
+
if (!packageName || !installCmd) {
|
|
439
|
+
throw new AppError(
|
|
440
|
+
`${parsed} cannot be installed automatically — please follow the documented install steps.`,
|
|
441
|
+
{ code: 'PROVIDER_NOT_AUTO_INSTALLABLE', statusCode: 400 },
|
|
442
|
+
);
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
const job = createInstallJob({ provider: parsed, installCmd, packageName });
|
|
446
|
+
res.json(createApiSuccessResponse({
|
|
447
|
+
jobId: job.id,
|
|
448
|
+
provider: parsed,
|
|
449
|
+
installCmd,
|
|
450
|
+
startedAt: job.startedAt,
|
|
451
|
+
}));
|
|
452
|
+
}),
|
|
453
|
+
);
|
|
454
|
+
|
|
455
|
+
/**
|
|
456
|
+
* GET /api/providers/:provider/install/:jobId/stream
|
|
457
|
+
* SSE endpoint (EventSource-friendly). Replays every buffered log line
|
|
458
|
+
* to the new subscriber, then forwards live stdout/stderr until the
|
|
459
|
+
* child exits. Clients can reconnect freely — reconnects replay from
|
|
460
|
+
* the start, so you never miss output, even if the browser dropped
|
|
461
|
+
* the previous connection while npm was mid-download.
|
|
462
|
+
*
|
|
463
|
+
* EventSource can't set custom headers, so this endpoint also accepts
|
|
464
|
+
* ?token=... as a fallback auth channel (same pattern the search
|
|
465
|
+
* endpoint uses).
|
|
466
|
+
*/
|
|
467
|
+
router.get(
|
|
468
|
+
'/:provider/install/:jobId/stream',
|
|
469
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
470
|
+
const parsed = parseProvider(req.params.provider);
|
|
471
|
+
const jobId = readPathParam(req.params.jobId, 'jobId');
|
|
472
|
+
const job = getInstallJob(jobId);
|
|
473
|
+
if (!job || job.provider !== parsed) {
|
|
474
|
+
throw new AppError('Install job not found or already expired.', {
|
|
475
|
+
code: 'INSTALL_JOB_NOT_FOUND',
|
|
476
|
+
statusCode: 404,
|
|
477
|
+
});
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
res.setHeader('Content-Type', 'text/event-stream');
|
|
481
|
+
res.setHeader('Cache-Control', 'no-cache, no-transform');
|
|
482
|
+
res.setHeader('Connection', 'keep-alive');
|
|
483
|
+
res.setHeader('X-Accel-Buffering', 'no');
|
|
484
|
+
if (typeof res.flushHeaders === 'function') res.flushHeaders();
|
|
485
|
+
try {
|
|
486
|
+
(res.socket as NodeJS.Socket & { setNoDelay?: (on: boolean) => void })?.setNoDelay?.(true);
|
|
487
|
+
} catch { /* noop */ }
|
|
488
|
+
|
|
489
|
+
let closed = false;
|
|
490
|
+
const write = (event: string, payload: unknown) => {
|
|
491
|
+
if (closed) return;
|
|
492
|
+
try {
|
|
493
|
+
res.write(`event: ${event}\n`);
|
|
494
|
+
res.write(`data: ${JSON.stringify(payload)}\n\n`);
|
|
495
|
+
} catch { /* socket gone */ }
|
|
496
|
+
};
|
|
497
|
+
|
|
498
|
+
// Immediate primer + heartbeat, same as before — keeps intermediary
|
|
499
|
+
// proxies from treating the connection as idle.
|
|
500
|
+
try { res.write(': start\n\n'); } catch { /* noop */ }
|
|
501
|
+
const heartbeat = setInterval(() => {
|
|
502
|
+
if (closed) return;
|
|
503
|
+
try { res.write(': ping\n\n'); } catch { /* noop */ }
|
|
504
|
+
}, 5000);
|
|
505
|
+
|
|
506
|
+
// Replay the buffered transcript first so late subscribers see
|
|
507
|
+
// every line npm has already produced.
|
|
508
|
+
for (const entry of job.logs) {
|
|
509
|
+
write('log', { stream: entry.stream, chunk: entry.chunk });
|
|
510
|
+
}
|
|
511
|
+
|
|
512
|
+
const onLog = (entry: { stream: string; chunk: string }) => {
|
|
513
|
+
write('log', { stream: entry.stream, chunk: entry.chunk });
|
|
514
|
+
};
|
|
515
|
+
const onDone = (payload: Record<string, unknown>) => {
|
|
516
|
+
write('done', payload);
|
|
517
|
+
cleanup();
|
|
518
|
+
try { res.end(); } catch { /* noop */ }
|
|
519
|
+
};
|
|
520
|
+
|
|
521
|
+
const cleanup = () => {
|
|
522
|
+
if (closed) return;
|
|
523
|
+
closed = true;
|
|
524
|
+
clearInterval(heartbeat);
|
|
525
|
+
job.emitter.off('log', onLog);
|
|
526
|
+
job.emitter.off('done', onDone);
|
|
527
|
+
};
|
|
528
|
+
|
|
529
|
+
if (job.status !== 'running') {
|
|
530
|
+
// Job already finished — replay the terminal done frame and exit.
|
|
531
|
+
write('done', snapshotDonePayload(job));
|
|
532
|
+
cleanup();
|
|
533
|
+
try { res.end(); } catch { /* noop */ }
|
|
534
|
+
return;
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
job.emitter.on('log', onLog);
|
|
538
|
+
job.emitter.once('done', onDone);
|
|
539
|
+
|
|
540
|
+
req.on('close', () => {
|
|
541
|
+
// Client walked away. DO NOT cancel the install — detaching is fine.
|
|
542
|
+
cleanup();
|
|
543
|
+
});
|
|
544
|
+
}),
|
|
545
|
+
);
|
|
546
|
+
|
|
547
|
+
router.delete(
|
|
548
|
+
'/:provider/install/:jobId',
|
|
549
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
550
|
+
const parsed = parseProvider(req.params.provider);
|
|
551
|
+
const jobId = readPathParam(req.params.jobId, 'jobId');
|
|
552
|
+
const job = getInstallJob(jobId);
|
|
553
|
+
if (!job || job.provider !== parsed) {
|
|
554
|
+
throw new AppError('Install job not found.', {
|
|
555
|
+
code: 'INSTALL_JOB_NOT_FOUND',
|
|
556
|
+
statusCode: 404,
|
|
557
|
+
});
|
|
558
|
+
}
|
|
559
|
+
const cancelled = cancelInstallJob(jobId);
|
|
560
|
+
res.json(createApiSuccessResponse({ cancelled }));
|
|
561
|
+
}),
|
|
562
|
+
);
|
|
563
|
+
|
|
564
|
+
router.post(
|
|
565
|
+
'/mcp/servers/global',
|
|
566
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
567
|
+
const payload = parseMcpUpsertPayload(req.body);
|
|
568
|
+
if (payload.scope === 'local') {
|
|
569
|
+
throw new AppError('Global MCP add supports only "user" or "project" scopes.', {
|
|
570
|
+
code: 'INVALID_GLOBAL_MCP_SCOPE',
|
|
571
|
+
statusCode: 400,
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
const results = await providerMcpService.addMcpServerToAllProviders({
|
|
576
|
+
...payload,
|
|
577
|
+
scope: payload.scope === 'user' ? 'user' : 'project',
|
|
578
|
+
});
|
|
579
|
+
res.status(201).json(createApiSuccessResponse({ results }));
|
|
580
|
+
}),
|
|
581
|
+
);
|
|
582
|
+
|
|
583
|
+
export default router;
|