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