@pixelbyte-software/pixcode 1.30.2 → 1.31.1

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.
Files changed (209) hide show
  1. package/LICENSE +718 -718
  2. package/README.de.md +248 -248
  3. package/README.ja.md +240 -240
  4. package/README.ko.md +240 -240
  5. package/README.md +295 -285
  6. package/README.ru.md +248 -248
  7. package/README.tr.md +250 -250
  8. package/README.zh-CN.md +240 -240
  9. package/dist/api-docs.html +879 -879
  10. package/dist/assets/index-BtOeB3cE.js +837 -0
  11. package/dist/assets/index-CDpePeIN.css +32 -0
  12. package/dist/assets/vendor-codemirror-CzYAOTxS.js +41 -0
  13. package/dist/clear-cache.html +85 -85
  14. package/dist/convert-icons.md +52 -52
  15. package/dist/favicon.png +0 -0
  16. package/dist/favicon.svg +7 -8
  17. package/dist/generate-icons.js +48 -48
  18. package/dist/icons/codex-white.svg +3 -3
  19. package/dist/icons/codex.svg +3 -3
  20. package/dist/icons/cursor-white.svg +11 -11
  21. package/dist/icons/icon-128x128.png +0 -0
  22. package/dist/icons/icon-128x128.svg +9 -12
  23. package/dist/icons/icon-144x144.png +0 -0
  24. package/dist/icons/icon-144x144.svg +9 -12
  25. package/dist/icons/icon-152x152.png +0 -0
  26. package/dist/icons/icon-152x152.svg +9 -12
  27. package/dist/icons/icon-192x192.png +0 -0
  28. package/dist/icons/icon-192x192.svg +9 -12
  29. package/dist/icons/icon-384x384.png +0 -0
  30. package/dist/icons/icon-384x384.svg +9 -12
  31. package/dist/icons/icon-512x512.png +0 -0
  32. package/dist/icons/icon-512x512.svg +9 -12
  33. package/dist/icons/icon-72x72.png +0 -0
  34. package/dist/icons/icon-72x72.svg +9 -12
  35. package/dist/icons/icon-96x96.png +0 -0
  36. package/dist/icons/icon-96x96.svg +9 -12
  37. package/dist/icons/icon-template.svg +9 -12
  38. package/dist/icons/qwen-ai-icon.png +0 -0
  39. package/dist/index.html +60 -50
  40. package/dist/logo.png +0 -0
  41. package/dist/logo.svg +11 -16
  42. package/dist/manifest.json +60 -60
  43. package/dist/sw.js +124 -124
  44. package/dist-server/server/claude-sdk.js +28 -5
  45. package/dist-server/server/claude-sdk.js.map +1 -1
  46. package/dist-server/server/cli.js +100 -97
  47. package/dist-server/server/cli.js.map +1 -1
  48. package/dist-server/server/daemon/manager.js +33 -33
  49. package/dist-server/server/daemon-manager.js +62 -62
  50. package/dist-server/server/database/db.js +114 -22
  51. package/dist-server/server/database/db.js.map +1 -1
  52. package/dist-server/server/database/schema.js +122 -89
  53. package/dist-server/server/database/schema.js.map +1 -1
  54. package/dist-server/server/gemini-cli.js +6 -1
  55. package/dist-server/server/gemini-cli.js.map +1 -1
  56. package/dist-server/server/index.js +346 -61
  57. package/dist-server/server/index.js.map +1 -1
  58. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +29 -2
  59. package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -1
  60. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +22 -2
  61. package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -1
  62. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +2 -2
  63. package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -1
  64. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +14 -2
  65. package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -1
  66. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js +132 -0
  67. package/dist-server/server/modules/providers/list/qwen/qwen-auth.provider.js.map +1 -0
  68. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js +87 -0
  69. package/dist-server/server/modules/providers/list/qwen/qwen-mcp.provider.js.map +1 -0
  70. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js +201 -0
  71. package/dist-server/server/modules/providers/list/qwen/qwen-sessions.provider.js.map +1 -0
  72. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js +19 -0
  73. package/dist-server/server/modules/providers/list/qwen/qwen.provider.js.map +1 -0
  74. package/dist-server/server/modules/providers/provider.registry.js +2 -0
  75. package/dist-server/server/modules/providers/provider.registry.js.map +1 -1
  76. package/dist-server/server/modules/providers/provider.routes.js +478 -1
  77. package/dist-server/server/modules/providers/provider.routes.js.map +1 -1
  78. package/dist-server/server/modules/providers/shared/provider-configs.js +105 -0
  79. package/dist-server/server/modules/providers/shared/provider-configs.js.map +1 -0
  80. package/dist-server/server/projects.js +197 -6
  81. package/dist-server/server/projects.js.map +1 -1
  82. package/dist-server/server/qwen-code-cli.js +350 -0
  83. package/dist-server/server/qwen-code-cli.js.map +1 -0
  84. package/dist-server/server/qwen-response-handler.js +70 -0
  85. package/dist-server/server/qwen-response-handler.js.map +1 -0
  86. package/dist-server/server/routes/commands.js +25 -25
  87. package/dist-server/server/routes/git.js +17 -17
  88. package/dist-server/server/routes/network.js +116 -0
  89. package/dist-server/server/routes/network.js.map +1 -0
  90. package/dist-server/server/routes/projects.js +166 -1
  91. package/dist-server/server/routes/projects.js.map +1 -1
  92. package/dist-server/server/routes/qwen.js +23 -0
  93. package/dist-server/server/routes/qwen.js.map +1 -0
  94. package/dist-server/server/routes/taskmaster.js +419 -419
  95. package/dist-server/server/routes/telegram.js +119 -0
  96. package/dist-server/server/routes/telegram.js.map +1 -0
  97. package/dist-server/server/services/external-access.js +228 -0
  98. package/dist-server/server/services/external-access.js.map +1 -0
  99. package/dist-server/server/services/install-jobs.js +552 -0
  100. package/dist-server/server/services/install-jobs.js.map +1 -0
  101. package/dist-server/server/services/notification-orchestrator.js +19 -5
  102. package/dist-server/server/services/notification-orchestrator.js.map +1 -1
  103. package/dist-server/server/services/provider-credentials.js +154 -0
  104. package/dist-server/server/services/provider-credentials.js.map +1 -0
  105. package/dist-server/server/services/provider-models.js +218 -0
  106. package/dist-server/server/services/provider-models.js.map +1 -0
  107. package/dist-server/server/services/telegram/bot.js +259 -0
  108. package/dist-server/server/services/telegram/bot.js.map +1 -0
  109. package/dist-server/server/services/telegram/translations.js +160 -0
  110. package/dist-server/server/services/telegram/translations.js.map +1 -0
  111. package/dist-server/server/utils/port-access.js +196 -0
  112. package/dist-server/server/utils/port-access.js.map +1 -0
  113. package/dist-server/shared/modelConstants.js +18 -0
  114. package/dist-server/shared/modelConstants.js.map +1 -1
  115. package/package.json +177 -168
  116. package/scripts/fix-node-pty.js +67 -67
  117. package/server/claude-sdk.js +857 -834
  118. package/server/cli.js +940 -937
  119. package/server/constants/config.js +4 -4
  120. package/server/cursor-cli.js +342 -342
  121. package/server/daemon/manager.js +564 -564
  122. package/server/daemon-manager.js +920 -920
  123. package/server/database/db.js +696 -593
  124. package/server/database/schema.js +138 -102
  125. package/server/gemini-cli.js +475 -469
  126. package/server/gemini-response-handler.js +79 -79
  127. package/server/index.js +2854 -2556
  128. package/server/load-env.js +34 -34
  129. package/server/middleware/auth.js +132 -132
  130. package/server/modules/providers/list/claude/claude-auth.provider.ts +145 -123
  131. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  132. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  133. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  134. package/server/modules/providers/list/codex/codex-auth.provider.ts +115 -100
  135. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  136. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  137. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  138. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -143
  139. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  140. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  141. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  142. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +163 -151
  143. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  144. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  145. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  146. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +145 -0
  147. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -0
  148. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +218 -0
  149. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -0
  150. package/server/modules/providers/provider.registry.ts +38 -36
  151. package/server/modules/providers/provider.routes.ts +781 -217
  152. package/server/modules/providers/services/mcp.service.ts +94 -94
  153. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  154. package/server/modules/providers/services/sessions.service.ts +45 -45
  155. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  156. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  157. package/server/modules/providers/shared/provider-configs.ts +118 -0
  158. package/server/modules/providers/tests/mcp.test.ts +293 -293
  159. package/server/openai-codex.js +426 -426
  160. package/server/projects.js +2993 -2792
  161. package/server/qwen-code-cli.js +392 -0
  162. package/server/qwen-response-handler.js +73 -0
  163. package/server/routes/agent.js +1245 -1245
  164. package/server/routes/auth.js +134 -134
  165. package/server/routes/codex.js +19 -19
  166. package/server/routes/commands.js +554 -554
  167. package/server/routes/cursor.js +52 -52
  168. package/server/routes/gemini.js +24 -24
  169. package/server/routes/git.js +1488 -1488
  170. package/server/routes/mcp-utils.js +31 -31
  171. package/server/routes/messages.js +61 -61
  172. package/server/routes/network.js +128 -0
  173. package/server/routes/plugins.js +307 -307
  174. package/server/routes/projects.js +795 -627
  175. package/server/routes/qwen.js +27 -0
  176. package/server/routes/settings.js +286 -286
  177. package/server/routes/taskmaster.js +1471 -1471
  178. package/server/routes/telegram.js +125 -0
  179. package/server/routes/user.js +123 -123
  180. package/server/services/external-access.js +240 -0
  181. package/server/services/install-jobs.js +569 -0
  182. package/server/services/notification-orchestrator.js +242 -227
  183. package/server/services/provider-credentials.js +151 -0
  184. package/server/services/provider-models.js +225 -0
  185. package/server/services/telegram/bot.js +280 -0
  186. package/server/services/telegram/translations.js +170 -0
  187. package/server/services/vapid-keys.js +35 -35
  188. package/server/sessionManager.js +225 -225
  189. package/server/shared/interfaces.ts +54 -54
  190. package/server/shared/types.ts +172 -172
  191. package/server/shared/utils.ts +193 -193
  192. package/server/tsconfig.json +36 -36
  193. package/server/utils/colors.js +21 -21
  194. package/server/utils/commandParser.js +303 -303
  195. package/server/utils/frontmatter.js +18 -18
  196. package/server/utils/gitConfig.js +34 -34
  197. package/server/utils/mcp-detector.js +147 -147
  198. package/server/utils/plugin-loader.js +457 -457
  199. package/server/utils/plugin-process-manager.js +184 -184
  200. package/server/utils/port-access.js +209 -0
  201. package/server/utils/runtime-paths.js +37 -37
  202. package/server/utils/taskmaster-websocket.js +128 -128
  203. package/server/utils/url-detection.js +71 -71
  204. package/server/vite-daemon.js +78 -78
  205. package/shared/modelConstants.js +117 -97
  206. package/shared/networkHosts.js +22 -22
  207. package/dist/assets/index-C2c9QNwK.css +0 -32
  208. package/dist/assets/index-DyXDZED-.js +0 -1277
  209. package/dist/assets/vendor-codemirror-NA4v81it.js +0 -41
@@ -1,151 +1,151 @@
1
- import path from 'node:path';
2
-
3
- import type { IProviderMcp } from '@/shared/interfaces.js';
4
- import type { LLMProvider, McpScope, McpTransport, ProviderMcpServer, UpsertProviderMcpServerInput } from '@/shared/types.js';
5
- import { AppError } from '@/shared/utils.js';
6
-
7
- const resolveWorkspacePath = (workspacePath?: string): string =>
8
- path.resolve(workspacePath ?? process.cwd());
9
-
10
- const normalizeServerName = (name: string): string => {
11
- const normalized = name.trim();
12
- if (!normalized) {
13
- throw new AppError('MCP server name is required.', {
14
- code: 'MCP_SERVER_NAME_REQUIRED',
15
- statusCode: 400,
16
- });
17
- }
18
-
19
- return normalized;
20
- };
21
-
22
- /**
23
- * Shared MCP provider for provider-specific config readers/writers.
24
- */
25
- export abstract class McpProvider implements IProviderMcp {
26
- protected readonly provider: LLMProvider;
27
- protected readonly supportedScopes: McpScope[];
28
- protected readonly supportedTransports: McpTransport[];
29
-
30
- protected constructor(
31
- provider: LLMProvider,
32
- supportedScopes: McpScope[],
33
- supportedTransports: McpTransport[],
34
- ) {
35
- this.provider = provider;
36
- this.supportedScopes = supportedScopes;
37
- this.supportedTransports = supportedTransports;
38
- }
39
-
40
- async listServers(options?: { workspacePath?: string }): Promise<Record<McpScope, ProviderMcpServer[]>> {
41
- const grouped: Record<McpScope, ProviderMcpServer[]> = {
42
- user: [],
43
- local: [],
44
- project: [],
45
- };
46
-
47
- for (const scope of this.supportedScopes) {
48
- grouped[scope] = await this.listServersForScope(scope, options);
49
- }
50
-
51
- return grouped;
52
- }
53
-
54
- async listServersForScope(
55
- scope: McpScope,
56
- options?: { workspacePath?: string },
57
- ): Promise<ProviderMcpServer[]> {
58
- if (!this.supportedScopes.includes(scope)) {
59
- return [];
60
- }
61
-
62
- const workspacePath = resolveWorkspacePath(options?.workspacePath);
63
- const scopedServers = await this.readScopedServers(scope, workspacePath);
64
- return Object.entries(scopedServers)
65
- .map(([name, rawConfig]) => this.normalizeServerConfig(scope, name, rawConfig))
66
- .filter((entry): entry is ProviderMcpServer => entry !== null);
67
- }
68
-
69
- async upsertServer(input: UpsertProviderMcpServerInput): Promise<ProviderMcpServer> {
70
- const scope = input.scope ?? 'project';
71
- this.assertScopeAndTransport(scope, input.transport);
72
-
73
- const workspacePath = resolveWorkspacePath(input.workspacePath);
74
- const normalizedName = normalizeServerName(input.name);
75
- const scopedServers = await this.readScopedServers(scope, workspacePath);
76
- scopedServers[normalizedName] = this.buildServerConfig(input);
77
- await this.writeScopedServers(scope, workspacePath, scopedServers);
78
-
79
- return {
80
- provider: this.provider,
81
- name: normalizedName,
82
- scope,
83
- transport: input.transport,
84
- command: input.command,
85
- args: input.args,
86
- env: input.env,
87
- cwd: input.cwd,
88
- url: input.url,
89
- headers: input.headers,
90
- envVars: input.envVars,
91
- bearerTokenEnvVar: input.bearerTokenEnvVar,
92
- envHttpHeaders: input.envHttpHeaders,
93
- };
94
- }
95
-
96
- async removeServer(
97
- input: { name: string; scope?: McpScope; workspacePath?: string },
98
- ): Promise<{ removed: boolean; provider: LLMProvider; name: string; scope: McpScope }> {
99
- const scope = input.scope ?? 'project';
100
- this.assertScope(scope);
101
-
102
- const workspacePath = resolveWorkspacePath(input.workspacePath);
103
- const normalizedName = normalizeServerName(input.name);
104
- const scopedServers = await this.readScopedServers(scope, workspacePath);
105
- const removed = Object.prototype.hasOwnProperty.call(scopedServers, normalizedName);
106
- if (removed) {
107
- delete scopedServers[normalizedName];
108
- await this.writeScopedServers(scope, workspacePath, scopedServers);
109
- }
110
-
111
- return { removed, provider: this.provider, name: normalizedName, scope };
112
- }
113
-
114
- protected abstract readScopedServers(
115
- scope: McpScope,
116
- workspacePath: string,
117
- ): Promise<Record<string, unknown>>;
118
-
119
- protected abstract writeScopedServers(
120
- scope: McpScope,
121
- workspacePath: string,
122
- servers: Record<string, unknown>,
123
- ): Promise<void>;
124
-
125
- protected abstract buildServerConfig(input: UpsertProviderMcpServerInput): Record<string, unknown>;
126
-
127
- protected abstract normalizeServerConfig(
128
- scope: McpScope,
129
- name: string,
130
- rawConfig: unknown,
131
- ): ProviderMcpServer | null;
132
-
133
- protected assertScope(scope: McpScope): void {
134
- if (!this.supportedScopes.includes(scope)) {
135
- throw new AppError(`Provider "${this.provider}" does not support "${scope}" MCP scope.`, {
136
- code: 'MCP_SCOPE_NOT_SUPPORTED',
137
- statusCode: 400,
138
- });
139
- }
140
- }
141
-
142
- protected assertScopeAndTransport(scope: McpScope, transport: McpTransport): void {
143
- this.assertScope(scope);
144
- if (!this.supportedTransports.includes(transport)) {
145
- throw new AppError(`Provider "${this.provider}" does not support "${transport}" MCP transport.`, {
146
- code: 'MCP_TRANSPORT_NOT_SUPPORTED',
147
- statusCode: 400,
148
- });
149
- }
150
- }
151
- }
1
+ import path from 'node:path';
2
+
3
+ import type { IProviderMcp } from '@/shared/interfaces.js';
4
+ import type { LLMProvider, McpScope, McpTransport, ProviderMcpServer, UpsertProviderMcpServerInput } from '@/shared/types.js';
5
+ import { AppError } from '@/shared/utils.js';
6
+
7
+ const resolveWorkspacePath = (workspacePath?: string): string =>
8
+ path.resolve(workspacePath ?? process.cwd());
9
+
10
+ const normalizeServerName = (name: string): string => {
11
+ const normalized = name.trim();
12
+ if (!normalized) {
13
+ throw new AppError('MCP server name is required.', {
14
+ code: 'MCP_SERVER_NAME_REQUIRED',
15
+ statusCode: 400,
16
+ });
17
+ }
18
+
19
+ return normalized;
20
+ };
21
+
22
+ /**
23
+ * Shared MCP provider for provider-specific config readers/writers.
24
+ */
25
+ export abstract class McpProvider implements IProviderMcp {
26
+ protected readonly provider: LLMProvider;
27
+ protected readonly supportedScopes: McpScope[];
28
+ protected readonly supportedTransports: McpTransport[];
29
+
30
+ protected constructor(
31
+ provider: LLMProvider,
32
+ supportedScopes: McpScope[],
33
+ supportedTransports: McpTransport[],
34
+ ) {
35
+ this.provider = provider;
36
+ this.supportedScopes = supportedScopes;
37
+ this.supportedTransports = supportedTransports;
38
+ }
39
+
40
+ async listServers(options?: { workspacePath?: string }): Promise<Record<McpScope, ProviderMcpServer[]>> {
41
+ const grouped: Record<McpScope, ProviderMcpServer[]> = {
42
+ user: [],
43
+ local: [],
44
+ project: [],
45
+ };
46
+
47
+ for (const scope of this.supportedScopes) {
48
+ grouped[scope] = await this.listServersForScope(scope, options);
49
+ }
50
+
51
+ return grouped;
52
+ }
53
+
54
+ async listServersForScope(
55
+ scope: McpScope,
56
+ options?: { workspacePath?: string },
57
+ ): Promise<ProviderMcpServer[]> {
58
+ if (!this.supportedScopes.includes(scope)) {
59
+ return [];
60
+ }
61
+
62
+ const workspacePath = resolveWorkspacePath(options?.workspacePath);
63
+ const scopedServers = await this.readScopedServers(scope, workspacePath);
64
+ return Object.entries(scopedServers)
65
+ .map(([name, rawConfig]) => this.normalizeServerConfig(scope, name, rawConfig))
66
+ .filter((entry): entry is ProviderMcpServer => entry !== null);
67
+ }
68
+
69
+ async upsertServer(input: UpsertProviderMcpServerInput): Promise<ProviderMcpServer> {
70
+ const scope = input.scope ?? 'project';
71
+ this.assertScopeAndTransport(scope, input.transport);
72
+
73
+ const workspacePath = resolveWorkspacePath(input.workspacePath);
74
+ const normalizedName = normalizeServerName(input.name);
75
+ const scopedServers = await this.readScopedServers(scope, workspacePath);
76
+ scopedServers[normalizedName] = this.buildServerConfig(input);
77
+ await this.writeScopedServers(scope, workspacePath, scopedServers);
78
+
79
+ return {
80
+ provider: this.provider,
81
+ name: normalizedName,
82
+ scope,
83
+ transport: input.transport,
84
+ command: input.command,
85
+ args: input.args,
86
+ env: input.env,
87
+ cwd: input.cwd,
88
+ url: input.url,
89
+ headers: input.headers,
90
+ envVars: input.envVars,
91
+ bearerTokenEnvVar: input.bearerTokenEnvVar,
92
+ envHttpHeaders: input.envHttpHeaders,
93
+ };
94
+ }
95
+
96
+ async removeServer(
97
+ input: { name: string; scope?: McpScope; workspacePath?: string },
98
+ ): Promise<{ removed: boolean; provider: LLMProvider; name: string; scope: McpScope }> {
99
+ const scope = input.scope ?? 'project';
100
+ this.assertScope(scope);
101
+
102
+ const workspacePath = resolveWorkspacePath(input.workspacePath);
103
+ const normalizedName = normalizeServerName(input.name);
104
+ const scopedServers = await this.readScopedServers(scope, workspacePath);
105
+ const removed = Object.prototype.hasOwnProperty.call(scopedServers, normalizedName);
106
+ if (removed) {
107
+ delete scopedServers[normalizedName];
108
+ await this.writeScopedServers(scope, workspacePath, scopedServers);
109
+ }
110
+
111
+ return { removed, provider: this.provider, name: normalizedName, scope };
112
+ }
113
+
114
+ protected abstract readScopedServers(
115
+ scope: McpScope,
116
+ workspacePath: string,
117
+ ): Promise<Record<string, unknown>>;
118
+
119
+ protected abstract writeScopedServers(
120
+ scope: McpScope,
121
+ workspacePath: string,
122
+ servers: Record<string, unknown>,
123
+ ): Promise<void>;
124
+
125
+ protected abstract buildServerConfig(input: UpsertProviderMcpServerInput): Record<string, unknown>;
126
+
127
+ protected abstract normalizeServerConfig(
128
+ scope: McpScope,
129
+ name: string,
130
+ rawConfig: unknown,
131
+ ): ProviderMcpServer | null;
132
+
133
+ protected assertScope(scope: McpScope): void {
134
+ if (!this.supportedScopes.includes(scope)) {
135
+ throw new AppError(`Provider "${this.provider}" does not support "${scope}" MCP scope.`, {
136
+ code: 'MCP_SCOPE_NOT_SUPPORTED',
137
+ statusCode: 400,
138
+ });
139
+ }
140
+ }
141
+
142
+ protected assertScopeAndTransport(scope: McpScope, transport: McpTransport): void {
143
+ this.assertScope(scope);
144
+ if (!this.supportedTransports.includes(transport)) {
145
+ throw new AppError(`Provider "${this.provider}" does not support "${transport}" MCP transport.`, {
146
+ code: 'MCP_TRANSPORT_NOT_SUPPORTED',
147
+ statusCode: 400,
148
+ });
149
+ }
150
+ }
151
+ }
@@ -0,0 +1,118 @@
1
+ /**
2
+ * Registry of user-editable config files per provider CLI.
3
+ *
4
+ * This is the single source of truth for the Settings → Agents →
5
+ * Configuration tab. Adding a new provider? Append a row here and the
6
+ * UI + API pick it up — no component changes required.
7
+ *
8
+ * Rules:
9
+ * - `relativePath` is relative to the user's home directory.
10
+ * We never accept absolute paths from the client; the server
11
+ * resolves these explicitly so path traversal is impossible.
12
+ * - `format` drives the CodeMirror language extension on the client.
13
+ * - `readonly: true` hides the Save button and the server rejects
14
+ * writes. Use it for files the CLI owns (e.g. OAuth tokens).
15
+ * - `description` is shown as a subtle caption under the editor.
16
+ */
17
+
18
+ export type ProviderConfigFormat = 'json' | 'toml' | 'env' | 'text';
19
+
20
+ export type ProviderConfigFile = {
21
+ id: string;
22
+ label: string;
23
+ relativePath: string;
24
+ format: ProviderConfigFormat;
25
+ readonly?: boolean;
26
+ description?: string;
27
+ };
28
+
29
+ export const PROVIDER_CONFIG_FILES: Record<string, ProviderConfigFile[]> = {
30
+ claude: [
31
+ {
32
+ id: 'settings',
33
+ label: 'settings.json',
34
+ relativePath: '.claude/settings.json',
35
+ format: 'json',
36
+ description: 'Main Claude Code settings — default model, system prompt, tool policy.',
37
+ },
38
+ {
39
+ id: 'env',
40
+ label: '.env',
41
+ relativePath: '.claude/.env',
42
+ format: 'env',
43
+ description: 'Environment variables loaded when Claude runs (e.g. ANTHROPIC_API_KEY).',
44
+ },
45
+ ],
46
+ codex: [
47
+ {
48
+ id: 'config',
49
+ label: 'config.toml',
50
+ relativePath: '.codex/config.toml',
51
+ format: 'toml',
52
+ description: 'Main Codex CLI config — models, MCP servers, approval policy, sandbox mode.',
53
+ },
54
+ {
55
+ id: 'env',
56
+ label: '.env',
57
+ relativePath: '.codex/.env',
58
+ format: 'env',
59
+ description: 'Environment variables (OPENAI_API_KEY, OPENAI_BASE_URL, …).',
60
+ },
61
+ {
62
+ id: 'auth',
63
+ label: 'auth.json',
64
+ relativePath: '.codex/auth.json',
65
+ format: 'json',
66
+ readonly: true,
67
+ description: 'OAuth tokens managed by `codex login`. Read-only; editing here would corrupt the session.',
68
+ },
69
+ ],
70
+ cursor: [
71
+ {
72
+ id: 'env',
73
+ label: '.env',
74
+ relativePath: '.cursor/.env',
75
+ format: 'env',
76
+ description: 'Cursor CLI environment variables.',
77
+ },
78
+ ],
79
+ gemini: [
80
+ {
81
+ id: 'settings',
82
+ label: 'settings.json',
83
+ relativePath: '.gemini/settings.json',
84
+ format: 'json',
85
+ description: 'Main Gemini CLI settings — selected model, MCP servers, tool approval mode.',
86
+ },
87
+ {
88
+ id: 'env',
89
+ label: '.env',
90
+ relativePath: '.gemini/.env',
91
+ format: 'env',
92
+ description: 'Environment variables (GOOGLE_API_KEY, GEMINI_API_KEY, …).',
93
+ },
94
+ ],
95
+ qwen: [
96
+ {
97
+ id: 'settings',
98
+ label: 'settings.json',
99
+ relativePath: '.qwen/settings.json',
100
+ format: 'json',
101
+ description: 'Main Qwen Code settings — selected model, MCP servers, approval mode.',
102
+ },
103
+ {
104
+ id: 'env',
105
+ label: '.env',
106
+ relativePath: '.qwen/.env',
107
+ format: 'env',
108
+ description: 'Environment variables (DASHSCOPE_API_KEY, OPENAI_API_KEY for OpenAI-compatible routes, …).',
109
+ },
110
+ ],
111
+ };
112
+
113
+ export const SUPPORTED_CONFIG_PROVIDERS = Object.keys(PROVIDER_CONFIG_FILES);
114
+
115
+ // Hard cap — no config file we care about is remotely this big, but we
116
+ // want to refuse reads and writes that would swell memory. Editing a 1 MB
117
+ // settings.json is already a smell.
118
+ export const MAX_CONFIG_FILE_SIZE_BYTES = 1_048_576; // 1 MB