Skip to main content
Glama

MySQL MCP Server

index.js3.65 kB
#!/usr/bin/env node import { Server } from "@modelcontextprotocol/sdk/server/index.js"; import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js"; import { CallToolRequestSchema, ListResourcesRequestSchema, ListToolsRequestSchema, ReadResourceRequestSchema, } from "@modelcontextprotocol/sdk/types.js"; import mysql from "mysql2/promise"; const server = new Server( { name: "example-servers/mysql", version: "0.1.0", }, { capabilities: { resources: {}, tools: {}, }, } ); const args = process.argv.slice(2); if (args.length === 0) { console.error("Please provide a database URL as a command-line argument"); process.exit(1); } const databaseUrl = args[0]; const resourceBaseUrl = new URL(databaseUrl); resourceBaseUrl.protocol = "mysql:"; const pool = mysql.createPool(databaseUrl); const SCHEMA_PATH = "schema"; server.setRequestHandler(ListResourcesRequestSchema, async () => { const connection = await pool.getConnection(); console.log(connection); try { const [rows] = await connection.query( "SELECT TABLE_NAME FROM information_schema.tables WHERE TABLE_SCHEMA = DATABASE()" ); return { resources: rows.map((row) => ({ uri: new URL(`${row.TABLE_NAME}/${SCHEMA_PATH}`, resourceBaseUrl).href, mimeType: "application/json", name: `"${row.TABLE_NAME}" database schema`, })), }; } finally { connection.release(); } }); server.setRequestHandler(ReadResourceRequestSchema, async (request) => { const resourceUrl = new URL(request.params.uri); const pathComponents = resourceUrl.pathname.split("/"); const schema = pathComponents.pop(); const tableName = pathComponents.pop(); if (schema !== SCHEMA_PATH) { throw new Error("Invalid resource URI"); } const connection = await pool.getConnection(); try { const [rows] = await connection.query( "SELECT COLUMN_NAME as column_name, DATA_TYPE as data_type FROM information_schema.columns WHERE TABLE_NAME = ? AND TABLE_SCHEMA = DATABASE()", [tableName] ); return { contents: [ { uri: request.params.uri, mimeType: "application/json", text: JSON.stringify(rows, null, 2), }, ], }; } finally { connection.release(); } }); server.setRequestHandler(ListToolsRequestSchema, async () => { return { tools: [ { name: "query", description: "Run a read-only SQL query", inputSchema: { type: "object", properties: { sql: { type: "string" }, }, }, }, ], }; }); server.setRequestHandler(CallToolRequestSchema, async (request) => { if (request.params.name === "query") { const sql = request.params.arguments?.sql; const connection = await pool.getConnection(); try { await connection.beginTransaction(); // Set session to read only await connection.query("SET SESSION TRANSACTION READ ONLY"); const [rows] = await connection.query(sql); return { content: [{ type: "text", text: JSON.stringify(rows, null, 2) }], isError: false, }; } catch (error) { throw error; } finally { try { await connection.rollback(); } catch (error) { console.warn("Could not roll back transaction:", error); } connection.release(); } } throw new Error(`Unknown tool: ${request.params.name}`); }); async function runServer() { const transport = new StdioServerTransport(); await server.connect(transport); } runServer().catch(console.error);

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/shreyansh-ghl/mysql-mcp-server'

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