Skip to main content
Glama

DataForSEO MCP Server

by cortex8
index-http.ts6.04 kB
#!/usr/bin/env node import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js"; import { DataForSEOClient, DataForSEOConfig } from '../core/client/dataforseo.client.js'; import { SerpApiModule } from '../core/modules/serp/serp-api.module.js'; import { KeywordsDataApiModule } from '../core/modules/keywords-data/keywords-data-api.module.js'; import { OnPageApiModule } from '../core/modules/onpage/onpage-api.module.js'; import { DataForSEOLabsApi } from '../core/modules/dataforseo-labs/dataforseo-labs-api.module.js'; import { EnabledModulesSchema, isModuleEnabled, defaultEnabledModules } from '../core/config/modules.config.js'; import { BaseModule, ToolDefinition } from '../core/modules/base.module.js'; import { z } from 'zod'; import { BacklinksApiModule } from "../core/modules/backlinks/backlinks-api.module.js"; import { BusinessDataApiModule } from "../core/modules/business-data-api/business-data-api.module.js"; import { DomainAnalyticsApiModule } from "../core/modules/domain-analytics/domain-analytics-api.module.js"; import { StreamableHTTPServerTransport } from "@modelcontextprotocol/sdk/server/streamableHttp.js"; import express, { Request as ExpressRequest, Response, NextFunction } from "express"; import { randomUUID } from "node:crypto"; import { GetPromptResult, isInitializeRequest, ReadResourceResult, ServerNotificationSchema } from "@modelcontextprotocol/sdk/types.js" import { name, version } from '../core/utils/version.js'; import { ModuleLoaderService } from "../core/utils/module-loader.js"; import { initializeFieldConfiguration } from '../core/config/field-configuration.js'; import { initMcpServer } from "./init-mcp-server.js"; // Initialize field configuration if provided initializeFieldConfiguration(); // Extended request interface to include auth properties interface Request extends ExpressRequest { username?: string; password?: string; } console.error('Starting DataForSEO MCP Server...'); console.error(`Server name: ${name}, version: ${version}`); function getSessionId() { return randomUUID().toString(); } async function main() { const app = express(); app.use(express.json()); // Basic Auth Middleware const basicAuth = (req: Request, res: Response, next: NextFunction) => { // Check for Authorization header const authHeader = req.headers.authorization; console.error(authHeader) if (!authHeader || !authHeader.startsWith('Basic ')) { next(); return; } // Extract credentials const base64Credentials = authHeader.split(' ')[1]; const credentials = Buffer.from(base64Credentials, 'base64').toString('utf-8'); const [username, password] = credentials.split(':'); if (!username || !password) { console.error('Invalid credentials'); res.status(401).json({ jsonrpc: "2.0", error: { code: -32001, message: "Invalid credentials" }, id: null }); return; } // Add credentials to request req.username = username; req.password = password; next(); }; const handleMcpRequest = async (req: Request, res: Response) => { // In stateless mode, create a new instance of transport and server for each request // to ensure complete isolation. A single instance would cause request ID collisions // when multiple clients connect concurrently. try { // Check if we have valid credentials if (!req.username && !req.password) { // If no request auth, check environment variables const envUsername = process.env.DATAFORSEO_USERNAME; const envPassword = process.env.DATAFORSEO_PASSWORD; if (!envUsername || !envPassword) { console.error('No DataForSEO credentials provided'); res.status(401).json({ jsonrpc: "2.0", error: { code: -32001, message: "Authentication required. Provide DataForSEO credentials." }, id: null }); return; } // Use environment variables req.username = envUsername; req.password = envPassword; } const server = initMcpServer(req.username, req.password); console.error(Date.now().toLocaleString()) const transport: StreamableHTTPServerTransport = new StreamableHTTPServerTransport({ sessionIdGenerator: undefined }); await server.connect(transport); console.error('handle request'); await transport.handleRequest(req , res, req.body); console.error('end handle request'); req.on('close', () => { console.error('Request closed'); transport.close(); server.close(); }); } catch (error) { console.error('Error handling MCP request:', error); if (!res.headersSent) { res.status(500).json({ jsonrpc: '2.0', error: { code: -32603, message: 'Internal server error', }, id: null, }); } } }; const handleNotAllowed = (method: string) => async (req: Request, res: Response) => { console.error(`Received ${method} request`); res.status(405).json({ jsonrpc: "2.0", error: { code: -32000, message: "Method not allowed." }, id: null }); }; // Apply basic auth and shared handler to both endpoints app.post('/http', basicAuth, handleMcpRequest); app.post('/mcp', basicAuth, handleMcpRequest); app.get('/http', handleNotAllowed('GET HTTP')); app.get('/mcp', handleNotAllowed('GET MCP')); app.delete('/http', handleNotAllowed('DELETE HTTP')); app.delete('/mcp', handleNotAllowed('DELETE MCP')); // Start the server const PORT = process.env.PORT ? parseInt(process.env.PORT, 10) : 3000; app.listen(PORT, () => { console.log(`MCP Stateless Streamable HTTP Server listening on port ${PORT}`); }); } main().catch((error) => { console.error("Fatal error in main():", error); process.exit(1); });

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/cortex8/oyt-dataforseo-mcp-worker'

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