@ian2018cs/agenthub 0.1.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.
Files changed (136) hide show
  1. package/LICENSE +675 -0
  2. package/README.md +330 -0
  3. package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  4. package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  5. package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  6. package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  7. package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  8. package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  9. package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  10. package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  11. package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  12. package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  13. package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  14. package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  15. package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  16. package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  17. package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  18. package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  19. package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  20. package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  21. package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  22. package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  23. package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  24. package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  25. package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  26. package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  27. package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  28. package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  29. package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  30. package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  31. package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  32. package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  33. package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  34. package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  35. package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  36. package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  37. package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  38. package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  39. package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  40. package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  41. package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  42. package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  43. package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  44. package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  45. package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  46. package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  47. package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  48. package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  49. package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  50. package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  51. package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  52. package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  53. package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  54. package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  55. package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  56. package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  57. package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  58. package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  59. package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  60. package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  61. package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  62. package/dist/assets/index-B4ru3EJb.css +32 -0
  63. package/dist/assets/index-DDFuyrpY.js +154 -0
  64. package/dist/assets/vendor-codemirror-C_VWDoZS.js +39 -0
  65. package/dist/assets/vendor-icons-CJV4dnDL.js +326 -0
  66. package/dist/assets/vendor-katex-DK8hFnhL.js +261 -0
  67. package/dist/assets/vendor-markdown-VwNYkg_0.js +35 -0
  68. package/dist/assets/vendor-react-BeVl62c0.js +59 -0
  69. package/dist/assets/vendor-syntax-CdGaPJRS.js +16 -0
  70. package/dist/assets/vendor-utils-00TdZexr.js +1 -0
  71. package/dist/assets/vendor-xterm-CvdiG4-n.js +66 -0
  72. package/dist/clear-cache.html +85 -0
  73. package/dist/convert-icons.md +53 -0
  74. package/dist/favicon.png +0 -0
  75. package/dist/favicon.svg +9 -0
  76. package/dist/generate-icons.js +49 -0
  77. package/dist/icons/claude-ai-icon.svg +1 -0
  78. package/dist/icons/codex-white.svg +3 -0
  79. package/dist/icons/codex.svg +3 -0
  80. package/dist/icons/cursor-white.svg +12 -0
  81. package/dist/icons/cursor.svg +1 -0
  82. package/dist/icons/generate-icons.md +19 -0
  83. package/dist/icons/icon-128x128.png +0 -0
  84. package/dist/icons/icon-128x128.svg +12 -0
  85. package/dist/icons/icon-144x144.png +0 -0
  86. package/dist/icons/icon-144x144.svg +12 -0
  87. package/dist/icons/icon-152x152.png +0 -0
  88. package/dist/icons/icon-152x152.svg +12 -0
  89. package/dist/icons/icon-192x192.png +0 -0
  90. package/dist/icons/icon-192x192.svg +12 -0
  91. package/dist/icons/icon-384x384.png +0 -0
  92. package/dist/icons/icon-384x384.svg +12 -0
  93. package/dist/icons/icon-512x512.png +0 -0
  94. package/dist/icons/icon-512x512.svg +12 -0
  95. package/dist/icons/icon-72x72.png +0 -0
  96. package/dist/icons/icon-72x72.svg +12 -0
  97. package/dist/icons/icon-96x96.png +0 -0
  98. package/dist/icons/icon-96x96.svg +12 -0
  99. package/dist/icons/icon-template.svg +12 -0
  100. package/dist/index.html +57 -0
  101. package/dist/logo-128.png +0 -0
  102. package/dist/logo-256.png +0 -0
  103. package/dist/logo-32.png +0 -0
  104. package/dist/logo-512.png +0 -0
  105. package/dist/logo-64.png +0 -0
  106. package/dist/logo.svg +17 -0
  107. package/dist/manifest.json +61 -0
  108. package/dist/screenshots/cli-selection.png +0 -0
  109. package/dist/screenshots/desktop-main.png +0 -0
  110. package/dist/screenshots/mobile-chat.png +0 -0
  111. package/dist/screenshots/tools-modal.png +0 -0
  112. package/dist/sw.js +49 -0
  113. package/package.json +113 -0
  114. package/server/claude-sdk.js +791 -0
  115. package/server/cli.js +330 -0
  116. package/server/database/auth.db +0 -0
  117. package/server/database/db.js +523 -0
  118. package/server/database/init.sql +23 -0
  119. package/server/index.js +1678 -0
  120. package/server/load-env.js +27 -0
  121. package/server/middleware/auth.js +118 -0
  122. package/server/projects.js +899 -0
  123. package/server/routes/admin.js +89 -0
  124. package/server/routes/auth.js +144 -0
  125. package/server/routes/commands.js +570 -0
  126. package/server/routes/mcp-utils.js +37 -0
  127. package/server/routes/mcp.js +593 -0
  128. package/server/routes/projects.js +216 -0
  129. package/server/routes/skills.js +891 -0
  130. package/server/routes/usage.js +206 -0
  131. package/server/services/pricing.js +196 -0
  132. package/server/services/usage-scanner.js +283 -0
  133. package/server/services/user-directories.js +123 -0
  134. package/server/utils/commandParser.js +303 -0
  135. package/server/utils/mcp-detector.js +73 -0
  136. package/shared/modelConstants.js +23 -0
@@ -0,0 +1,593 @@
1
+ import express from 'express';
2
+ import { promises as fs } from 'fs';
3
+ import path from 'path';
4
+ import { fileURLToPath } from 'url';
5
+ import { dirname } from 'path';
6
+ import { getUserPaths } from '../services/user-directories.js';
7
+
8
+ const router = express.Router();
9
+ const __filename = fileURLToPath(import.meta.url);
10
+ const __dirname = dirname(__filename);
11
+
12
+ // Claude CLI command routes
13
+
14
+ // GET /api/mcp/cli/list - List MCP servers using Claude CLI
15
+ router.get('/cli/list', async (req, res) => {
16
+ try {
17
+ console.log('📋 Listing MCP servers using Claude CLI');
18
+
19
+ const userUuid = req.user?.uuid;
20
+ if (!userUuid) {
21
+ return res.status(401).json({ error: 'User authentication required' });
22
+ }
23
+
24
+ const userPaths = getUserPaths(userUuid);
25
+ const { spawn } = await import('child_process');
26
+
27
+ const cliProcess = spawn('claude', ['mcp', 'list'], {
28
+ stdio: ['pipe', 'pipe', 'pipe'],
29
+ env: { ...process.env, CLAUDE_CONFIG_DIR: userPaths.claudeDir }
30
+ });
31
+
32
+ let stdout = '';
33
+ let stderr = '';
34
+
35
+ cliProcess.stdout.on('data', (data) => {
36
+ stdout += data.toString();
37
+ });
38
+
39
+ cliProcess.stderr.on('data', (data) => {
40
+ stderr += data.toString();
41
+ });
42
+
43
+ cliProcess.on('close', (code) => {
44
+ if (code === 0) {
45
+ res.json({ success: true, output: stdout, servers: parseClaudeListOutput(stdout) });
46
+ } else {
47
+ console.error('Claude CLI error:', stderr);
48
+ res.status(500).json({ error: 'Claude CLI command failed', details: stderr });
49
+ }
50
+ });
51
+
52
+ cliProcess.on('error', (error) => {
53
+ console.error('Error running Claude CLI:', error);
54
+ res.status(500).json({ error: 'Failed to run Claude CLI', details: error.message });
55
+ });
56
+ } catch (error) {
57
+ console.error('Error listing MCP servers via CLI:', error);
58
+ res.status(500).json({ error: 'Failed to list MCP servers', details: error.message });
59
+ }
60
+ });
61
+
62
+ // POST /api/mcp/cli/add - Add MCP server using Claude CLI
63
+ router.post('/cli/add', async (req, res) => {
64
+ try {
65
+ const { name, type = 'stdio', command, args = [], url, headers = {}, env = {}, scope = 'user', projectPath } = req.body;
66
+
67
+ const userUuid = req.user?.uuid;
68
+ if (!userUuid) {
69
+ return res.status(401).json({ error: 'User authentication required' });
70
+ }
71
+
72
+ console.log(`➕ Adding MCP server using Claude CLI (${scope} scope):`, name);
73
+
74
+ const userPaths = getUserPaths(userUuid);
75
+ const { spawn } = await import('child_process');
76
+
77
+ let cliArgs = ['mcp', 'add'];
78
+
79
+ // Add scope flag
80
+ cliArgs.push('--scope', scope);
81
+
82
+ if (type === 'http') {
83
+ cliArgs.push('--transport', 'http', name, url);
84
+ // Add headers if provided
85
+ Object.entries(headers).forEach(([key, value]) => {
86
+ cliArgs.push('--header', `${key}: ${value}`);
87
+ });
88
+ } else if (type === 'sse') {
89
+ cliArgs.push('--transport', 'sse', name, url);
90
+ // Add headers if provided
91
+ Object.entries(headers).forEach(([key, value]) => {
92
+ cliArgs.push('--header', `${key}: ${value}`);
93
+ });
94
+ } else {
95
+ // stdio (default): claude mcp add --scope user <name> <command> [args...]
96
+ cliArgs.push(name);
97
+ // Add environment variables
98
+ Object.entries(env).forEach(([key, value]) => {
99
+ cliArgs.push('-e', `${key}=${value}`);
100
+ });
101
+ cliArgs.push(command);
102
+ if (args && args.length > 0) {
103
+ cliArgs.push(...args);
104
+ }
105
+ }
106
+
107
+ console.log('🔧 Running Claude CLI command:', 'claude', cliArgs.join(' '));
108
+
109
+ // For local scope, we need to run the command in the project directory
110
+ const spawnOptions = {
111
+ stdio: ['pipe', 'pipe', 'pipe'],
112
+ env: { ...process.env, CLAUDE_CONFIG_DIR: userPaths.claudeDir }
113
+ };
114
+
115
+ if (scope === 'local' && projectPath) {
116
+ spawnOptions.cwd = projectPath;
117
+ console.log('📁 Running in project directory:', projectPath);
118
+ }
119
+
120
+ const cliProcess = spawn('claude', cliArgs, spawnOptions);
121
+
122
+ let stdout = '';
123
+ let stderr = '';
124
+
125
+ cliProcess.stdout.on('data', (data) => {
126
+ stdout += data.toString();
127
+ });
128
+
129
+ cliProcess.stderr.on('data', (data) => {
130
+ stderr += data.toString();
131
+ });
132
+
133
+ cliProcess.on('close', (code) => {
134
+ if (code === 0) {
135
+ res.json({ success: true, output: stdout, message: `MCP server "${name}" added successfully` });
136
+ } else {
137
+ console.error('Claude CLI error:', stderr);
138
+ res.status(400).json({ error: 'Claude CLI command failed', details: stderr });
139
+ }
140
+ });
141
+
142
+ cliProcess.on('error', (error) => {
143
+ console.error('Error running Claude CLI:', error);
144
+ res.status(500).json({ error: 'Failed to run Claude CLI', details: error.message });
145
+ });
146
+ } catch (error) {
147
+ console.error('Error adding MCP server via CLI:', error);
148
+ res.status(500).json({ error: 'Failed to add MCP server', details: error.message });
149
+ }
150
+ });
151
+
152
+ // POST /api/mcp/cli/add-json - Add MCP server using JSON format
153
+ router.post('/cli/add-json', async (req, res) => {
154
+ try {
155
+ const { name, jsonConfig, scope = 'user', projectPath } = req.body;
156
+
157
+ const userUuid = req.user?.uuid;
158
+ if (!userUuid) {
159
+ return res.status(401).json({ error: 'User authentication required' });
160
+ }
161
+
162
+ console.log('➕ Adding MCP server using JSON format:', name);
163
+
164
+ // Validate and parse JSON config
165
+ let parsedConfig;
166
+ try {
167
+ parsedConfig = typeof jsonConfig === 'string' ? JSON.parse(jsonConfig) : jsonConfig;
168
+ } catch (parseError) {
169
+ return res.status(400).json({
170
+ error: 'Invalid JSON configuration',
171
+ details: parseError.message
172
+ });
173
+ }
174
+
175
+ // Validate required fields
176
+ if (!parsedConfig.type) {
177
+ return res.status(400).json({
178
+ error: 'Invalid configuration',
179
+ details: 'Missing required field: type'
180
+ });
181
+ }
182
+
183
+ if (parsedConfig.type === 'stdio' && !parsedConfig.command) {
184
+ return res.status(400).json({
185
+ error: 'Invalid configuration',
186
+ details: 'stdio type requires a command field'
187
+ });
188
+ }
189
+
190
+ if ((parsedConfig.type === 'http' || parsedConfig.type === 'sse') && !parsedConfig.url) {
191
+ return res.status(400).json({
192
+ error: 'Invalid configuration',
193
+ details: `${parsedConfig.type} type requires a url field`
194
+ });
195
+ }
196
+
197
+ const userPaths = getUserPaths(userUuid);
198
+ const { spawn } = await import('child_process');
199
+
200
+ // Build the command: claude mcp add-json --scope <scope> <name> '<json>'
201
+ const cliArgs = ['mcp', 'add-json', '--scope', scope, name];
202
+
203
+ // Add the JSON config as a properly formatted string
204
+ const jsonString = JSON.stringify(parsedConfig);
205
+ cliArgs.push(jsonString);
206
+
207
+ console.log('🔧 Running Claude CLI command:', 'claude', cliArgs[0], cliArgs[1], cliArgs[2], cliArgs[3], cliArgs[4], jsonString);
208
+
209
+ // For local scope, we need to run the command in the project directory
210
+ const spawnOptions = {
211
+ stdio: ['pipe', 'pipe', 'pipe'],
212
+ env: { ...process.env, CLAUDE_CONFIG_DIR: userPaths.claudeDir }
213
+ };
214
+
215
+ if (scope === 'local' && projectPath) {
216
+ spawnOptions.cwd = projectPath;
217
+ console.log('📁 Running in project directory:', projectPath);
218
+ }
219
+
220
+ const cliProcess = spawn('claude', cliArgs, spawnOptions);
221
+
222
+ let stdout = '';
223
+ let stderr = '';
224
+
225
+ cliProcess.stdout.on('data', (data) => {
226
+ stdout += data.toString();
227
+ });
228
+
229
+ cliProcess.stderr.on('data', (data) => {
230
+ stderr += data.toString();
231
+ });
232
+
233
+ cliProcess.on('close', (code) => {
234
+ if (code === 0) {
235
+ res.json({ success: true, output: stdout, message: `MCP server "${name}" added successfully via JSON` });
236
+ } else {
237
+ console.error('Claude CLI error:', stderr);
238
+ res.status(400).json({ error: 'Claude CLI command failed', details: stderr });
239
+ }
240
+ });
241
+
242
+ cliProcess.on('error', (error) => {
243
+ console.error('Error running Claude CLI:', error);
244
+ res.status(500).json({ error: 'Failed to run Claude CLI', details: error.message });
245
+ });
246
+ } catch (error) {
247
+ console.error('Error adding MCP server via JSON:', error);
248
+ res.status(500).json({ error: 'Failed to add MCP server', details: error.message });
249
+ }
250
+ });
251
+
252
+ // DELETE /api/mcp/cli/remove/:name - Remove MCP server using Claude CLI
253
+ router.delete('/cli/remove/:name', async (req, res) => {
254
+ try {
255
+ const { name } = req.params;
256
+ const { scope } = req.query; // Get scope from query params
257
+
258
+ const userUuid = req.user?.uuid;
259
+ if (!userUuid) {
260
+ return res.status(401).json({ error: 'User authentication required' });
261
+ }
262
+
263
+ // Handle the ID format (remove scope prefix if present)
264
+ let actualName = name;
265
+ let actualScope = scope;
266
+
267
+ // If the name includes a scope prefix like "local:test", extract it
268
+ if (name.includes(':')) {
269
+ const [prefix, serverName] = name.split(':');
270
+ actualName = serverName;
271
+ actualScope = actualScope || prefix; // Use prefix as scope if not provided in query
272
+ }
273
+
274
+ console.log('🗑️ Removing MCP server using Claude CLI:', actualName, 'scope:', actualScope);
275
+
276
+ const userPaths = getUserPaths(userUuid);
277
+ const { spawn } = await import('child_process');
278
+
279
+ // Build command args based on scope
280
+ let cliArgs = ['mcp', 'remove'];
281
+
282
+ // Add scope flag if it's local scope
283
+ if (actualScope === 'local') {
284
+ cliArgs.push('--scope', 'local');
285
+ } else if (actualScope === 'user' || !actualScope) {
286
+ // User scope is default, but we can be explicit
287
+ cliArgs.push('--scope', 'user');
288
+ }
289
+
290
+ cliArgs.push(actualName);
291
+
292
+ console.log('🔧 Running Claude CLI command:', 'claude', cliArgs.join(' '));
293
+
294
+ const cliProcess = spawn('claude', cliArgs, {
295
+ stdio: ['pipe', 'pipe', 'pipe'],
296
+ env: { ...process.env, CLAUDE_CONFIG_DIR: userPaths.claudeDir }
297
+ });
298
+
299
+ let stdout = '';
300
+ let stderr = '';
301
+
302
+ cliProcess.stdout.on('data', (data) => {
303
+ stdout += data.toString();
304
+ });
305
+
306
+ cliProcess.stderr.on('data', (data) => {
307
+ stderr += data.toString();
308
+ });
309
+
310
+ cliProcess.on('close', (code) => {
311
+ if (code === 0) {
312
+ res.json({ success: true, output: stdout, message: `MCP server "${name}" removed successfully` });
313
+ } else {
314
+ console.error('Claude CLI error:', stderr);
315
+ res.status(400).json({ error: 'Claude CLI command failed', details: stderr });
316
+ }
317
+ });
318
+
319
+ cliProcess.on('error', (error) => {
320
+ console.error('Error running Claude CLI:', error);
321
+ res.status(500).json({ error: 'Failed to run Claude CLI', details: error.message });
322
+ });
323
+ } catch (error) {
324
+ console.error('Error removing MCP server via CLI:', error);
325
+ res.status(500).json({ error: 'Failed to remove MCP server', details: error.message });
326
+ }
327
+ });
328
+
329
+ // GET /api/mcp/cli/get/:name - Get MCP server details using Claude CLI
330
+ router.get('/cli/get/:name', async (req, res) => {
331
+ try {
332
+ const { name } = req.params;
333
+
334
+ const userUuid = req.user?.uuid;
335
+ if (!userUuid) {
336
+ return res.status(401).json({ error: 'User authentication required' });
337
+ }
338
+
339
+ console.log('📄 Getting MCP server details using Claude CLI:', name);
340
+
341
+ const userPaths = getUserPaths(userUuid);
342
+ const { spawn } = await import('child_process');
343
+
344
+ const cliProcess = spawn('claude', ['mcp', 'get', name], {
345
+ stdio: ['pipe', 'pipe', 'pipe'],
346
+ env: { ...process.env, CLAUDE_CONFIG_DIR: userPaths.claudeDir }
347
+ });
348
+
349
+ let stdout = '';
350
+ let stderr = '';
351
+
352
+ cliProcess.stdout.on('data', (data) => {
353
+ stdout += data.toString();
354
+ });
355
+
356
+ cliProcess.stderr.on('data', (data) => {
357
+ stderr += data.toString();
358
+ });
359
+
360
+ cliProcess.on('close', (code) => {
361
+ if (code === 0) {
362
+ res.json({ success: true, output: stdout, server: parseClaudeGetOutput(stdout) });
363
+ } else {
364
+ console.error('Claude CLI error:', stderr);
365
+ res.status(404).json({ error: 'Claude CLI command failed', details: stderr });
366
+ }
367
+ });
368
+
369
+ cliProcess.on('error', (error) => {
370
+ console.error('Error running Claude CLI:', error);
371
+ res.status(500).json({ error: 'Failed to run Claude CLI', details: error.message });
372
+ });
373
+ } catch (error) {
374
+ console.error('Error getting MCP server details via CLI:', error);
375
+ res.status(500).json({ error: 'Failed to get MCP server details', details: error.message });
376
+ }
377
+ });
378
+
379
+ // GET /api/mcp/config/read - Read MCP servers directly from Claude config files
380
+ router.get('/config/read', async (req, res) => {
381
+ try {
382
+ console.log('📖 Reading MCP servers from Claude config files');
383
+
384
+ const userUuid = req.user?.uuid;
385
+ if (!userUuid) {
386
+ return res.status(401).json({
387
+ success: false,
388
+ error: 'User authentication required',
389
+ servers: []
390
+ });
391
+ }
392
+
393
+ const userPaths = getUserPaths(userUuid);
394
+ const configPaths = [
395
+ path.join(userPaths.claudeDir, '.claude.json'),
396
+ path.join(userPaths.claudeDir, 'settings.json')
397
+ ];
398
+
399
+ let configData = null;
400
+ let configPath = null;
401
+
402
+ // Try to read from either config file
403
+ for (const filepath of configPaths) {
404
+ try {
405
+ const fileContent = await fs.readFile(filepath, 'utf8');
406
+ configData = JSON.parse(fileContent);
407
+ configPath = filepath;
408
+ console.log(`✅ Found Claude config at: ${filepath}`);
409
+ break;
410
+ } catch (error) {
411
+ // File doesn't exist or is not valid JSON, try next
412
+ console.log(`ℹ️ Config not found or invalid at: ${filepath}`);
413
+ }
414
+ }
415
+
416
+ if (!configData) {
417
+ return res.json({
418
+ success: false,
419
+ message: 'No Claude configuration file found',
420
+ servers: []
421
+ });
422
+ }
423
+
424
+ // Extract MCP servers from the config
425
+ const servers = [];
426
+
427
+ // Check for user-scoped MCP servers (at root level)
428
+ if (configData.mcpServers && typeof configData.mcpServers === 'object' && Object.keys(configData.mcpServers).length > 0) {
429
+ console.log('🔍 Found user-scoped MCP servers:', Object.keys(configData.mcpServers));
430
+ for (const [name, config] of Object.entries(configData.mcpServers)) {
431
+ const server = {
432
+ id: name,
433
+ name: name,
434
+ type: 'stdio', // Default type
435
+ scope: 'user', // User scope - available across all projects
436
+ config: {},
437
+ raw: config // Include raw config for full details
438
+ };
439
+
440
+ // Determine transport type and extract config
441
+ if (config.command) {
442
+ server.type = 'stdio';
443
+ server.config.command = config.command;
444
+ server.config.args = config.args || [];
445
+ server.config.env = config.env || {};
446
+ } else if (config.url) {
447
+ server.type = config.transport || 'http';
448
+ server.config.url = config.url;
449
+ server.config.headers = config.headers || {};
450
+ }
451
+
452
+ servers.push(server);
453
+ }
454
+ }
455
+
456
+ // Check for local-scoped MCP servers (project-specific)
457
+ const currentProjectPath = process.cwd();
458
+
459
+ // Check under 'projects' key
460
+ if (configData.projects && configData.projects[currentProjectPath]) {
461
+ const projectConfig = configData.projects[currentProjectPath];
462
+ if (projectConfig.mcpServers && typeof projectConfig.mcpServers === 'object' && Object.keys(projectConfig.mcpServers).length > 0) {
463
+ console.log(`🔍 Found local-scoped MCP servers for ${currentProjectPath}:`, Object.keys(projectConfig.mcpServers));
464
+ for (const [name, config] of Object.entries(projectConfig.mcpServers)) {
465
+ const server = {
466
+ id: `local:${name}`, // Prefix with scope for uniqueness
467
+ name: name, // Keep original name
468
+ type: 'stdio', // Default type
469
+ scope: 'local', // Local scope - only for this project
470
+ projectPath: currentProjectPath,
471
+ config: {},
472
+ raw: config // Include raw config for full details
473
+ };
474
+
475
+ // Determine transport type and extract config
476
+ if (config.command) {
477
+ server.type = 'stdio';
478
+ server.config.command = config.command;
479
+ server.config.args = config.args || [];
480
+ server.config.env = config.env || {};
481
+ } else if (config.url) {
482
+ server.type = config.transport || 'http';
483
+ server.config.url = config.url;
484
+ server.config.headers = config.headers || {};
485
+ }
486
+
487
+ servers.push(server);
488
+ }
489
+ }
490
+ }
491
+
492
+ console.log(`📋 Found ${servers.length} MCP servers in config`);
493
+
494
+ res.json({
495
+ success: true,
496
+ configPath: configPath,
497
+ servers: servers
498
+ });
499
+ } catch (error) {
500
+ console.error('Error reading Claude config:', error);
501
+ res.status(500).json({
502
+ error: 'Failed to read Claude configuration',
503
+ details: error.message
504
+ });
505
+ }
506
+ });
507
+
508
+ // Helper functions to parse Claude CLI output
509
+ function parseClaudeListOutput(output) {
510
+ const servers = [];
511
+ const lines = output.split('\n').filter(line => line.trim());
512
+
513
+ for (const line of lines) {
514
+ // Skip the header line
515
+ if (line.includes('Checking MCP server health')) continue;
516
+
517
+ // Parse lines like "test: test test - ✗ Failed to connect"
518
+ // or "server-name: command or description - ✓ Connected"
519
+ if (line.includes(':')) {
520
+ const colonIndex = line.indexOf(':');
521
+ const name = line.substring(0, colonIndex).trim();
522
+
523
+ // Skip empty names
524
+ if (!name) continue;
525
+
526
+ // Extract the rest after the name
527
+ const rest = line.substring(colonIndex + 1).trim();
528
+
529
+ // Try to extract description and status
530
+ let description = rest;
531
+ let status = 'unknown';
532
+ let type = 'stdio'; // default type
533
+
534
+ // Check for status indicators
535
+ if (rest.includes('✓') || rest.includes('✗')) {
536
+ const statusMatch = rest.match(/(.*?)\s*-\s*([✓✗].*)$/);
537
+ if (statusMatch) {
538
+ description = statusMatch[1].trim();
539
+ status = statusMatch[2].includes('✓') ? 'connected' : 'failed';
540
+ }
541
+ }
542
+
543
+ // Try to determine type from description
544
+ if (description.startsWith('http://') || description.startsWith('https://')) {
545
+ type = 'http';
546
+ }
547
+
548
+ servers.push({
549
+ name,
550
+ type,
551
+ status: status || 'active',
552
+ description
553
+ });
554
+ }
555
+ }
556
+
557
+ console.log('🔍 Parsed Claude CLI servers:', servers);
558
+ return servers;
559
+ }
560
+
561
+ function parseClaudeGetOutput(output) {
562
+ // Parse the output from 'claude mcp get <name>' command
563
+ // This is a simple parser - might need adjustment based on actual output format
564
+ try {
565
+ // Try to extract JSON if present
566
+ const jsonMatch = output.match(/\{[\s\S]*\}/);
567
+ if (jsonMatch) {
568
+ return JSON.parse(jsonMatch[0]);
569
+ }
570
+
571
+ // Otherwise, parse as text
572
+ const server = { raw_output: output };
573
+ const lines = output.split('\n');
574
+
575
+ for (const line of lines) {
576
+ if (line.includes('Name:')) {
577
+ server.name = line.split(':')[1]?.trim();
578
+ } else if (line.includes('Type:')) {
579
+ server.type = line.split(':')[1]?.trim();
580
+ } else if (line.includes('Command:')) {
581
+ server.command = line.split(':')[1]?.trim();
582
+ } else if (line.includes('URL:')) {
583
+ server.url = line.split(':')[1]?.trim();
584
+ }
585
+ }
586
+
587
+ return server;
588
+ } catch (error) {
589
+ return { raw_output: output, parse_error: error.message };
590
+ }
591
+ }
592
+
593
+ export default router;