Skip to main content
Glama
DaxianLee

Cocos Creator MCP Server Plugin

by DaxianLee
mcp-server.js62.1 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); Object.defineProperty(exports, "__esModule", { value: true }); exports.MCPServer = void 0; const http = __importStar(require("http")); const url = __importStar(require("url")); const scene_tools_1 = require("./tools/scene-tools"); const node_tools_1 = require("./tools/node-tools"); const component_tools_1 = require("./tools/component-tools"); const prefab_tools_1 = require("./tools/prefab-tools"); const project_tools_1 = require("./tools/project-tools"); const debug_tools_1 = require("./tools/debug-tools"); const preferences_tools_1 = require("./tools/preferences-tools"); const server_tools_1 = require("./tools/server-tools"); const broadcast_tools_1 = require("./tools/broadcast-tools"); const scene_advanced_tools_1 = require("./tools/scene-advanced-tools"); const scene_view_tools_1 = require("./tools/scene-view-tools"); const reference_image_tools_1 = require("./tools/reference-image-tools"); const asset_advanced_tools_1 = require("./tools/asset-advanced-tools"); const validation_tools_1 = require("./tools/validation-tools"); class MCPServer { constructor(settings) { this.httpServer = null; this.clients = new Map(); this.tools = {}; this.toolsList = []; this.enabledTools = []; // 存储启用的工具列表 this.settings = settings; this.initializeTools(); } initializeTools() { try { console.log('[MCPServer] Initializing tools...'); this.tools.scene = new scene_tools_1.SceneTools(); this.tools.node = new node_tools_1.NodeTools(); this.tools.component = new component_tools_1.ComponentTools(); this.tools.prefab = new prefab_tools_1.PrefabTools(); this.tools.project = new project_tools_1.ProjectTools(); this.tools.debug = new debug_tools_1.DebugTools(); this.tools.preferences = new preferences_tools_1.PreferencesTools(); this.tools.server = new server_tools_1.ServerTools(); this.tools.broadcast = new broadcast_tools_1.BroadcastTools(); this.tools.sceneAdvanced = new scene_advanced_tools_1.SceneAdvancedTools(); this.tools.sceneView = new scene_view_tools_1.SceneViewTools(); this.tools.referenceImage = new reference_image_tools_1.ReferenceImageTools(); this.tools.assetAdvanced = new asset_advanced_tools_1.AssetAdvancedTools(); this.tools.validation = new validation_tools_1.ValidationTools(); console.log('[MCPServer] Tools initialized successfully'); } catch (error) { console.error('[MCPServer] Error initializing tools:', error); throw error; } } async start() { if (this.httpServer) { console.log('[MCPServer] Server is already running'); return; } try { console.log(`[MCPServer] Starting HTTP server on port ${this.settings.port}...`); this.httpServer = http.createServer(this.handleHttpRequest.bind(this)); await new Promise((resolve, reject) => { this.httpServer.listen(this.settings.port, '127.0.0.1', () => { console.log(`[MCPServer] ✅ HTTP server started successfully on http://127.0.0.1:${this.settings.port}`); console.log(`[MCPServer] Health check: http://127.0.0.1:${this.settings.port}/health`); console.log(`[MCPServer] MCP endpoint: http://127.0.0.1:${this.settings.port}/mcp`); resolve(); }); this.httpServer.on('error', (err) => { console.error('[MCPServer] ❌ Failed to start server:', err); if (err.code === 'EADDRINUSE') { console.error(`[MCPServer] Port ${this.settings.port} is already in use. Please change the port in settings.`); } reject(err); }); }); this.setupTools(); console.log('[MCPServer] 🚀 MCP Server is ready for connections'); } catch (error) { console.error('[MCPServer] ❌ Failed to start server:', error); throw error; } } setupTools() { this.toolsList = []; // 如果没有启用工具配置,返回所有工具 if (!this.enabledTools || this.enabledTools.length === 0) { for (const [category, toolSet] of Object.entries(this.tools)) { const tools = toolSet.getTools(); for (const tool of tools) { this.toolsList.push({ name: `${category}_${tool.name}`, description: tool.description, inputSchema: tool.inputSchema }); } } } else { // 根据启用的工具配置过滤 const enabledToolNames = new Set(this.enabledTools.map(tool => `${tool.category}_${tool.name}`)); for (const [category, toolSet] of Object.entries(this.tools)) { const tools = toolSet.getTools(); for (const tool of tools) { const toolName = `${category}_${tool.name}`; if (enabledToolNames.has(toolName)) { this.toolsList.push({ name: toolName, description: tool.description, inputSchema: tool.inputSchema }); } } } } console.log(`[MCPServer] Setup tools: ${this.toolsList.length} tools available`); } getFilteredTools(enabledTools) { if (!enabledTools || enabledTools.length === 0) { return this.toolsList; // 如果没有过滤配置,返回所有工具 } const enabledToolNames = new Set(enabledTools.map(tool => `${tool.category}_${tool.name}`)); return this.toolsList.filter(tool => enabledToolNames.has(tool.name)); } async executeToolCall(toolName, args) { const parts = toolName.split('_'); const category = parts[0]; const toolMethodName = parts.slice(1).join('_'); if (this.tools[category]) { return await this.tools[category].execute(toolMethodName, args); } throw new Error(`Tool ${toolName} not found`); } getClients() { return Array.from(this.clients.values()); } getAvailableTools() { return this.toolsList; } updateEnabledTools(enabledTools) { console.log(`[MCPServer] Updating enabled tools: ${enabledTools.length} tools`); this.enabledTools = enabledTools; this.setupTools(); // 重新设置工具列表 } getSettings() { return this.settings; } async handleHttpRequest(req, res) { const parsedUrl = url.parse(req.url || '', true); const pathname = parsedUrl.pathname; // Set CORS headers res.setHeader('Access-Control-Allow-Origin', '*'); res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS'); res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization'); res.setHeader('Content-Type', 'application/json'); if (req.method === 'OPTIONS') { res.writeHead(200); res.end(); return; } try { if (pathname === '/mcp' && req.method === 'POST') { await this.handleMCPRequest(req, res); } else if (pathname === '/health' && req.method === 'GET') { res.writeHead(200); res.end(JSON.stringify({ status: 'ok', tools: this.toolsList.length })); } else if ((pathname === null || pathname === void 0 ? void 0 : pathname.startsWith('/api/')) && req.method === 'POST') { await this.handleSimpleAPIRequest(req, res, pathname); } else if (pathname === '/api/tools' && req.method === 'GET') { res.writeHead(200); res.end(JSON.stringify({ tools: this.getSimplifiedToolsList() })); } else { res.writeHead(404); res.end(JSON.stringify({ error: 'Not found' })); } } catch (error) { console.error('HTTP request error:', error); res.writeHead(500); res.end(JSON.stringify({ error: 'Internal server error' })); } } async handleMCPRequest(req, res) { let body = ''; req.on('data', (chunk) => { body += chunk.toString(); }); req.on('end', async () => { try { // Enhanced JSON parsing with better error handling let message; try { message = JSON.parse(body); } catch (parseError) { // Try to fix common JSON issues const fixedBody = this.fixCommonJsonIssues(body); try { message = JSON.parse(fixedBody); console.log('[MCPServer] Fixed JSON parsing issue'); } catch (secondError) { throw new Error(`JSON parsing failed: ${parseError.message}. Original body: ${body.substring(0, 500)}...`); } } const response = await this.handleMessage(message); res.writeHead(200); res.end(JSON.stringify(response)); } catch (error) { console.error('Error handling MCP request:', error); res.writeHead(400); res.end(JSON.stringify({ jsonrpc: '2.0', id: null, error: { code: -32700, message: `Parse error: ${error.message}` } })); } }); } async handleMessage(message) { const { id, method, params } = message; try { let result; switch (method) { case 'tools/list': result = { tools: this.getAvailableTools() }; break; case 'tools/call': const { name, arguments: args } = params; const toolResult = await this.executeToolCall(name, args); result = { content: [{ type: 'text', text: JSON.stringify(toolResult) }] }; break; case 'initialize': // MCP initialization result = { protocolVersion: '2024-11-05', capabilities: { tools: {} }, serverInfo: { name: 'cocos-mcp-server', version: '1.0.0' } }; break; default: throw new Error(`Unknown method: ${method}`); } return { jsonrpc: '2.0', id, result }; } catch (error) { return { jsonrpc: '2.0', id, error: { code: -32603, message: error.message } }; } } fixCommonJsonIssues(jsonStr) { let fixed = jsonStr; // Fix common escape character issues fixed = fixed // Fix unescaped quotes in strings .replace(/([^\\])"([^"]*[^\\])"([^,}\]:])/g, '$1\\"$2\\"$3') // Fix unescaped backslashes .replace(/([^\\])\\([^"\\\/bfnrt])/g, '$1\\\\$2') // Fix trailing commas .replace(/,(\s*[}\]])/g, '$1') // Fix single quotes (should be double quotes) .replace(/'/g, '"') // Fix common control characters .replace(/\n/g, '\\n') .replace(/\r/g, '\\r') .replace(/\t/g, '\\t'); return fixed; } stop() { if (this.httpServer) { this.httpServer.close(); this.httpServer = null; console.log('[MCPServer] HTTP server stopped'); } this.clients.clear(); } getStatus() { return { running: !!this.httpServer, port: this.settings.port, clients: 0 // HTTP is stateless, no persistent clients }; } async handleSimpleAPIRequest(req, res, pathname) { let body = ''; req.on('data', (chunk) => { body += chunk.toString(); }); req.on('end', async () => { try { // Extract tool name from path like /api/node/set_position const pathParts = pathname.split('/').filter(p => p); if (pathParts.length < 3) { res.writeHead(400); res.end(JSON.stringify({ error: 'Invalid API path. Use /api/{category}/{tool_name}' })); return; } const category = pathParts[1]; const toolName = pathParts[2]; const fullToolName = `${category}_${toolName}`; // Parse parameters with enhanced error handling let params; try { params = body ? JSON.parse(body) : {}; } catch (parseError) { // Try to fix JSON issues const fixedBody = this.fixCommonJsonIssues(body); try { params = JSON.parse(fixedBody); console.log('[MCPServer] Fixed API JSON parsing issue'); } catch (secondError) { res.writeHead(400); res.end(JSON.stringify({ error: 'Invalid JSON in request body', details: parseError.message, receivedBody: body.substring(0, 200) })); return; } } // Execute tool const result = await this.executeToolCall(fullToolName, params); res.writeHead(200); res.end(JSON.stringify({ success: true, tool: fullToolName, result: result })); } catch (error) { console.error('Simple API error:', error); res.writeHead(500); res.end(JSON.stringify({ success: false, error: error.message, tool: pathname })); } }); } getSimplifiedToolsList() { return this.toolsList.map(tool => { const parts = tool.name.split('_'); const category = parts[0]; const toolName = parts.slice(1).join('_'); return { name: tool.name, category: category, toolName: toolName, description: tool.description, apiPath: `/api/${category}/${toolName}`, curlExample: this.generateCurlExample(category, toolName, tool.inputSchema) }; }); } generateCurlExample(category, toolName, schema) { // Generate sample parameters based on schema const sampleParams = this.generateSampleParams(schema); const jsonString = JSON.stringify(sampleParams, null, 2); return `curl -X POST http://127.0.0.1:8585/api/${category}/${toolName} \\ -H "Content-Type: application/json" \\ -d '${jsonString}'`; } generateSampleParams(schema) { if (!schema || !schema.properties) return {}; const sample = {}; for (const [key, prop] of Object.entries(schema.properties)) { const propSchema = prop; switch (propSchema.type) { case 'string': sample[key] = propSchema.default || 'example_string'; break; case 'number': sample[key] = propSchema.default || 42; break; case 'boolean': sample[key] = propSchema.default || true; break; case 'object': sample[key] = propSchema.default || { x: 0, y: 0, z: 0 }; break; default: sample[key] = 'example_value'; } } return sample; } updateSettings(settings) { this.settings = settings; if (this.httpServer) { this.stop(); this.start(); } } } exports.MCPServer = MCPServer; // HTTP transport doesn't need persistent connections // MCP over HTTP uses request-response pattern //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mcp-server.js","sourceRoot":"","sources":["../source/mcp-server.ts"],"names":[],"mappings":"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","sourcesContent":["import * as http from 'http';\nimport * as url from 'url';\nimport { v4 as uuidv4 } from 'uuid';\nimport { MCPServerSettings, ServerStatus, MCPClient, ToolDefinition } from './types';\nimport { SceneTools } from './tools/scene-tools';\nimport { NodeTools } from './tools/node-tools';\nimport { ComponentTools } from './tools/component-tools';\nimport { PrefabTools } from './tools/prefab-tools';\nimport { ProjectTools } from './tools/project-tools';\nimport { DebugTools } from './tools/debug-tools';\nimport { PreferencesTools } from './tools/preferences-tools';\nimport { ServerTools } from './tools/server-tools';\nimport { BroadcastTools } from './tools/broadcast-tools';\nimport { SceneAdvancedTools } from './tools/scene-advanced-tools';\nimport { SceneViewTools } from './tools/scene-view-tools';\nimport { ReferenceImageTools } from './tools/reference-image-tools';\nimport { AssetAdvancedTools } from './tools/asset-advanced-tools';\nimport { ValidationTools } from './tools/validation-tools';\n\nexport class MCPServer {\n    private settings: MCPServerSettings;\n    private httpServer: http.Server | null = null;\n    private clients: Map<string, MCPClient> = new Map();\n    private tools: Record<string, any> = {};\n    private toolsList: ToolDefinition[] = [];\n    private enabledTools: any[] = []; // 存储启用的工具列表\n\n    constructor(settings: MCPServerSettings) {\n        this.settings = settings;\n        this.initializeTools();\n    }\n\n    private initializeTools(): void {\n        try {\n            console.log('[MCPServer] Initializing tools...');\n            this.tools.scene = new SceneTools();\n            this.tools.node = new NodeTools();\n            this.tools.component = new ComponentTools();\n            this.tools.prefab = new PrefabTools();\n            this.tools.project = new ProjectTools();\n            this.tools.debug = new DebugTools();\n            this.tools.preferences = new PreferencesTools();\n            this.tools.server = new ServerTools();\n            this.tools.broadcast = new BroadcastTools();\n            this.tools.sceneAdvanced = new SceneAdvancedTools();\n            this.tools.sceneView = new SceneViewTools();\n            this.tools.referenceImage = new ReferenceImageTools();\n            this.tools.assetAdvanced = new AssetAdvancedTools();\n            this.tools.validation = new ValidationTools();\n            console.log('[MCPServer] Tools initialized successfully');\n        } catch (error) {\n            console.error('[MCPServer] Error initializing tools:', error);\n            throw error;\n        }\n    }\n\n    public async start(): Promise<void> {\n        if (this.httpServer) {\n            console.log('[MCPServer] Server is already running');\n            return;\n        }\n\n        try {\n            console.log(`[MCPServer] Starting HTTP server on port ${this.settings.port}...`);\n            this.httpServer = http.createServer(this.handleHttpRequest.bind(this));\n\n            await new Promise<void>((resolve, reject) => {\n                this.httpServer!.listen(this.settings.port, '127.0.0.1', () => {\n                    console.log(`[MCPServer] ✅ HTTP server started successfully on http://127.0.0.1:${this.settings.port}`);\n                    console.log(`[MCPServer] Health check: http://127.0.0.1:${this.settings.port}/health`);\n                    console.log(`[MCPServer] MCP endpoint: http://127.0.0.1:${this.settings.port}/mcp`);\n                    resolve();\n                });\n                this.httpServer!.on('error', (err: any) => {\n                    console.error('[MCPServer] ❌ Failed to start server:', err);\n                    if (err.code === 'EADDRINUSE') {\n                        console.error(`[MCPServer] Port ${this.settings.port} is already in use. Please change the port in settings.`);\n                    }\n                    reject(err);\n                });\n            });\n\n            this.setupTools();\n            console.log('[MCPServer] 🚀 MCP Server is ready for connections');\n        } catch (error) {\n            console.error('[MCPServer] ❌ Failed to start server:', error);\n            throw error;\n        }\n    }\n\n    private setupTools(): void {\n        this.toolsList = [];\n        \n        // 如果没有启用工具配置，返回所有工具\n        if (!this.enabledTools || this.enabledTools.length === 0) {\n            for (const [category, toolSet] of Object.entries(this.tools)) {\n                const tools = toolSet.getTools();\n                for (const tool of tools) {\n                    this.toolsList.push({\n                        name: `${category}_${tool.name}`,\n                        description: tool.description,\n                        inputSchema: tool.inputSchema\n                    });\n                }\n            }\n        } else {\n            // 根据启用的工具配置过滤\n            const enabledToolNames = new Set(this.enabledTools.map(tool => `${tool.category}_${tool.name}`));\n            \n            for (const [category, toolSet] of Object.entries(this.tools)) {\n                const tools = toolSet.getTools();\n                for (const tool of tools) {\n                    const toolName = `${category}_${tool.name}`;\n                    if (enabledToolNames.has(toolName)) {\n                        this.toolsList.push({\n                            name: toolName,\n                            description: tool.description,\n                            inputSchema: tool.inputSchema\n                        });\n                    }\n                }\n            }\n        }\n        \n        console.log(`[MCPServer] Setup tools: ${this.toolsList.length} tools available`);\n    }\n\n    public getFilteredTools(enabledTools: any[]): ToolDefinition[] {\n        if (!enabledTools || enabledTools.length === 0) {\n            return this.toolsList; // 如果没有过滤配置，返回所有工具\n        }\n\n        const enabledToolNames = new Set(enabledTools.map(tool => `${tool.category}_${tool.name}`));\n        return this.toolsList.filter(tool => enabledToolNames.has(tool.name));\n    }\n\n    public async executeToolCall(toolName: string, args: any): Promise<any> {\n        const parts = toolName.split('_');\n        const category = parts[0];\n        const toolMethodName = parts.slice(1).join('_');\n        \n        if (this.tools[category]) {\n            return await this.tools[category].execute(toolMethodName, args);\n        }\n        \n        throw new Error(`Tool ${toolName} not found`);\n    }\n\n    public getClients(): MCPClient[] {\n        return Array.from(this.clients.values());\n    }\n    public getAvailableTools(): ToolDefinition[] {\n        return this.toolsList;\n    }\n\n    public updateEnabledTools(enabledTools: any[]): void {\n        console.log(`[MCPServer] Updating enabled tools: ${enabledTools.length} tools`);\n        this.enabledTools = enabledTools;\n        this.setupTools(); // 重新设置工具列表\n    }\n\n    public getSettings(): MCPServerSettings {\n        return this.settings;\n    }\n\n    private async handleHttpRequest(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\n        const parsedUrl = url.parse(req.url || '', true);\n        const pathname = parsedUrl.pathname;\n        \n        // Set CORS headers\n        res.setHeader('Access-Control-Allow-Origin', '*');\n        res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS');\n        res.setHeader('Access-Control-Allow-Headers', 'Content-Type, Authorization');\n        res.setHeader('Content-Type', 'application/json');\n        \n        if (req.method === 'OPTIONS') {\n            res.writeHead(200);\n            res.end();\n            return;\n        }\n        \n        try {\n            if (pathname === '/mcp' && req.method === 'POST') {\n                await this.handleMCPRequest(req, res);\n            } else if (pathname === '/health' && req.method === 'GET') {\n                res.writeHead(200);\n                res.end(JSON.stringify({ status: 'ok', tools: this.toolsList.length }));\n            } else if (pathname?.startsWith('/api/') && req.method === 'POST') {\n                await this.handleSimpleAPIRequest(req, res, pathname);\n            } else if (pathname === '/api/tools' && req.method === 'GET') {\n                res.writeHead(200);\n                res.end(JSON.stringify({ tools: this.getSimplifiedToolsList() }));\n            } else {\n                res.writeHead(404);\n                res.end(JSON.stringify({ error: 'Not found' }));\n            }\n        } catch (error) {\n            console.error('HTTP request error:', error);\n            res.writeHead(500);\n            res.end(JSON.stringify({ error: 'Internal server error' }));\n        }\n    }\n    \n    private async handleMCPRequest(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\n        let body = '';\n        \n        req.on('data', (chunk) => {\n            body += chunk.toString();\n        });\n        \n        req.on('end', async () => {\n            try {\n                // Enhanced JSON parsing with better error handling\n                let message;\n                try {\n                    message = JSON.parse(body);\n                } catch (parseError: any) {\n                    // Try to fix common JSON issues\n                    const fixedBody = this.fixCommonJsonIssues(body);\n                    try {\n                        message = JSON.parse(fixedBody);\n                        console.log('[MCPServer] Fixed JSON parsing issue');\n                    } catch (secondError) {\n                        throw new Error(`JSON parsing failed: ${parseError.message}. Original body: ${body.substring(0, 500)}...`);\n                    }\n                }\n                \n                const response = await this.handleMessage(message);\n                res.writeHead(200);\n                res.end(JSON.stringify(response));\n            } catch (error: any) {\n                console.error('Error handling MCP request:', error);\n                res.writeHead(400);\n                res.end(JSON.stringify({\n                    jsonrpc: '2.0',\n                    id: null,\n                    error: {\n                        code: -32700,\n                        message: `Parse error: ${error.message}`\n                    }\n                }));\n            }\n        });\n    }\n\n    private async handleMessage(message: any): Promise<any> {\n        const { id, method, params } = message;\n\n        try {\n            let result: any;\n\n            switch (method) {\n                case 'tools/list':\n                    result = { tools: this.getAvailableTools() };\n                    break;\n                case 'tools/call':\n                    const { name, arguments: args } = params;\n                    const toolResult = await this.executeToolCall(name, args);\n                    result = { content: [{ type: 'text', text: JSON.stringify(toolResult) }] };\n                    break;\n                case 'initialize':\n                    // MCP initialization\n                    result = {\n                        protocolVersion: '2024-11-05',\n                        capabilities: {\n                            tools: {}\n                        },\n                        serverInfo: {\n                            name: 'cocos-mcp-server',\n                            version: '1.0.0'\n                        }\n                    };\n                    break;\n                default:\n                    throw new Error(`Unknown method: ${method}`);\n            }\n\n            return {\n                jsonrpc: '2.0',\n                id,\n                result\n            };\n        } catch (error: any) {\n            return {\n                jsonrpc: '2.0',\n                id,\n                error: {\n                    code: -32603,\n                    message: error.message\n                }\n            };\n        }\n    }\n\n    private fixCommonJsonIssues(jsonStr: string): string {\n        let fixed = jsonStr;\n        \n        // Fix common escape character issues\n        fixed = fixed\n            // Fix unescaped quotes in strings\n            .replace(/([^\\\\])\"([^\"]*[^\\\\])\"([^,}\\]:])/g, '$1\\\\\"$2\\\\\"$3')\n            // Fix unescaped backslashes\n            .replace(/([^\\\\])\\\\([^\"\\\\\\/bfnrt])/g, '$1\\\\\\\\$2')\n            // Fix trailing commas\n            .replace(/,(\\s*[}\\]])/g, '$1')\n            // Fix single quotes (should be double quotes)\n            .replace(/'/g, '\"')\n            // Fix common control characters\n            .replace(/\\n/g, '\\\\n')\n            .replace(/\\r/g, '\\\\r')\n            .replace(/\\t/g, '\\\\t');\n        \n        return fixed;\n    }\n\n    public stop(): void {\n        if (this.httpServer) {\n            this.httpServer.close();\n            this.httpServer = null;\n            console.log('[MCPServer] HTTP server stopped');\n        }\n\n        this.clients.clear();\n    }\n\n    public getStatus(): ServerStatus {\n        return {\n            running: !!this.httpServer,\n            port: this.settings.port,\n            clients: 0 // HTTP is stateless, no persistent clients\n        };\n    }\n\n    private async handleSimpleAPIRequest(req: http.IncomingMessage, res: http.ServerResponse, pathname: string): Promise<void> {\n        let body = '';\n        \n        req.on('data', (chunk) => {\n            body += chunk.toString();\n        });\n        \n        req.on('end', async () => {\n            try {\n                // Extract tool name from path like /api/node/set_position\n                const pathParts = pathname.split('/').filter(p => p);\n                if (pathParts.length < 3) {\n                    res.writeHead(400);\n                    res.end(JSON.stringify({ error: 'Invalid API path. Use /api/{category}/{tool_name}' }));\n                    return;\n                }\n                \n                const category = pathParts[1];\n                const toolName = pathParts[2];\n                const fullToolName = `${category}_${toolName}`;\n                \n                // Parse parameters with enhanced error handling\n                let params;\n                try {\n                    params = body ? JSON.parse(body) : {};\n                } catch (parseError: any) {\n                    // Try to fix JSON issues\n                    const fixedBody = this.fixCommonJsonIssues(body);\n                    try {\n                        params = JSON.parse(fixedBody);\n                        console.log('[MCPServer] Fixed API JSON parsing issue');\n                    } catch (secondError: any) {\n                        res.writeHead(400);\n                        res.end(JSON.stringify({\n                            error: 'Invalid JSON in request body',\n                            details: parseError.message,\n                            receivedBody: body.substring(0, 200)\n                        }));\n                        return;\n                    }\n                }\n                \n                // Execute tool\n                const result = await this.executeToolCall(fullToolName, params);\n                \n                res.writeHead(200);\n                res.end(JSON.stringify({\n                    success: true,\n                    tool: fullToolName,\n                    result: result\n                }));\n                \n            } catch (error: any) {\n                console.error('Simple API error:', error);\n                res.writeHead(500);\n                res.end(JSON.stringify({\n                    success: false,\n                    error: error.message,\n                    tool: pathname\n                }));\n            }\n        });\n    }\n\n    private getSimplifiedToolsList(): any[] {\n        return this.toolsList.map(tool => {\n            const parts = tool.name.split('_');\n            const category = parts[0];\n            const toolName = parts.slice(1).join('_');\n            \n            return {\n                name: tool.name,\n                category: category,\n                toolName: toolName,\n                description: tool.description,\n                apiPath: `/api/${category}/${toolName}`,\n                curlExample: this.generateCurlExample(category, toolName, tool.inputSchema)\n            };\n        });\n    }\n\n    private generateCurlExample(category: string, toolName: string, schema: any): string {\n        // Generate sample parameters based on schema\n        const sampleParams = this.generateSampleParams(schema);\n        const jsonString = JSON.stringify(sampleParams, null, 2);\n        \n        return `curl -X POST http://127.0.0.1:8585/api/${category}/${toolName} \\\\\n  -H \"Content-Type: application/json\" \\\\\n  -d '${jsonString}'`;\n    }\n\n    private generateSampleParams(schema: any): any {\n        if (!schema || !schema.properties) return {};\n        \n        const sample: any = {};\n        for (const [key, prop] of Object.entries(schema.properties as any)) {\n            const propSchema = prop as any;\n            switch (propSchema.type) {\n                case 'string':\n                    sample[key] = propSchema.default || 'example_string';\n                    break;\n                case 'number':\n                    sample[key] = propSchema.default || 42;\n                    break;\n                case 'boolean':\n                    sample[key] = propSchema.default || true;\n                    break;\n                case 'object':\n                    sample[key] = propSchema.default || { x: 0, y: 0, z: 0 };\n                    break;\n                default:\n                    sample[key] = 'example_value';\n            }\n        }\n        return sample;\n    }\n\n    public updateSettings(settings: MCPServerSettings) {\n        this.settings = settings;\n        if (this.httpServer) {\n            this.stop();\n            this.start();\n        }\n    }\n}\n\n// HTTP transport doesn't need persistent connections\n// MCP over HTTP uses request-response pattern"]}

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/DaxianLee/cocos-mcp-server'

If you have feedback or need assistance with the MCP directory API, please join our Discord server