Skip to main content
Glama
http-server.ts42.6 kB
/** * GoHighLevel MCP HTTP Server * HTTP version for ChatGPT web integration */ import express from 'express'; import cors from 'cors'; import { Server } from '@modelcontextprotocol/sdk/server/index.js'; import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js'; import { CallToolRequestSchema, ErrorCode, ListToolsRequestSchema, McpError } from '@modelcontextprotocol/sdk/types.js'; import * as dotenv from 'dotenv'; import { GHLApiClient } from './clients/ghl-api-client'; import { CredentialManager } from './auth/credential-manager'; import { ContactTools } from './tools/contact-tools'; import { ConversationTools } from './tools/conversation-tools'; import { BlogTools } from './tools/blog-tools'; import { OpportunityTools } from './tools/opportunity-tools'; import { CalendarTools } from './tools/calendar-tools'; import { EmailTools } from './tools/email-tools'; import { LocationTools } from './tools/location-tools'; import { EmailISVTools } from './tools/email-isv-tools'; import { SocialMediaTools } from './tools/social-media-tools'; import { MediaTools } from './tools/media-tools'; import { ObjectTools } from './tools/object-tools'; import { AssociationTools } from './tools/association-tools'; import { CustomFieldV2Tools } from './tools/custom-field-v2-tools'; import { WorkflowTools } from './tools/workflow-tools'; import { SurveyTools } from './tools/survey-tools'; import { StoreTools } from './tools/store-tools'; import { ProductsTools } from './tools/products-tools'; import { PaymentsTools } from './tools/payments-tools'; import { InvoicesTools } from './tools/invoices-tools'; import { GHLConfig } from './types/ghl-types'; // Load environment variables dotenv.config(); /** * HTTP MCP Server class for web deployment */ class GHLMCPHttpServer { private app: express.Application; private server: Server; private ghlClient: GHLApiClient; private credentialManager: CredentialManager; private contactTools: ContactTools; private conversationTools: ConversationTools; private blogTools: BlogTools; private opportunityTools: OpportunityTools; private calendarTools: CalendarTools; private emailTools: EmailTools; private locationTools: LocationTools; private emailISVTools: EmailISVTools; private socialMediaTools: SocialMediaTools; private mediaTools: MediaTools; private objectTools: ObjectTools; private associationTools: AssociationTools; private customFieldV2Tools: CustomFieldV2Tools; private workflowTools: WorkflowTools; private surveyTools: SurveyTools; private storeTools: StoreTools; private productsTools: ProductsTools; private paymentsTools: PaymentsTools; private invoicesTools: InvoicesTools; private port: number; constructor() { this.port = parseInt(process.env.PORT || process.env.MCP_SERVER_PORT || '8000'); // Initialize Express app this.app = express(); this.setupExpress(); // Initialize MCP server with capabilities this.server = new Server( { name: 'ghl-mcp-server', version: '1.0.0', }, { capabilities: { tools: {}, }, } ); // Initialize credential manager this.credentialManager = CredentialManager.getInstance(); // Initialize default GHL API client (for backward compatibility) try { this.ghlClient = this.initializeGHLClient(); } catch (error) { process.stderr.write(`[GHL MCP] No default credentials configured: ${error}\n`); process.stderr.write('[GHL MCP] Server will require credentials per request\n'); // @ts-ignore - Allow null for multi-tenant mode this.ghlClient = null; } // Initialize tools this.contactTools = new ContactTools(this.ghlClient); this.conversationTools = new ConversationTools(this.ghlClient); this.blogTools = new BlogTools(this.ghlClient); this.opportunityTools = new OpportunityTools(this.ghlClient); this.calendarTools = new CalendarTools(this.ghlClient); this.emailTools = new EmailTools(this.ghlClient); this.locationTools = new LocationTools(this.ghlClient); this.emailISVTools = new EmailISVTools(this.ghlClient); this.socialMediaTools = new SocialMediaTools(this.ghlClient); this.mediaTools = new MediaTools(this.ghlClient); this.objectTools = new ObjectTools(this.ghlClient); this.associationTools = new AssociationTools(this.ghlClient); this.customFieldV2Tools = new CustomFieldV2Tools(this.ghlClient); this.workflowTools = new WorkflowTools(this.ghlClient); this.surveyTools = new SurveyTools(this.ghlClient); this.storeTools = new StoreTools(this.ghlClient); this.productsTools = new ProductsTools(this.ghlClient); this.paymentsTools = new PaymentsTools(this.ghlClient); this.invoicesTools = new InvoicesTools(this.ghlClient); // Setup MCP handlers this.setupMCPHandlers(); this.setupRoutes(); } /** * Setup Express middleware and configuration */ private setupExpress(): void { // Enable CORS for ChatGPT integration this.app.use(cors({ origin: ['https://chatgpt.com', 'https://chat.openai.com', 'http://localhost:*'], methods: ['GET', 'POST', 'OPTIONS'], allowedHeaders: ['Content-Type', 'Authorization', 'Accept'], credentials: true })); // Parse JSON requests this.app.use(express.json()); // Request logging this.app.use((req, res, next) => { console.log(`[HTTP] ${req.method} ${req.path} - ${new Date().toISOString()}`); next(); }); } /** * Initialize GoHighLevel API client with configuration */ private initializeGHLClient(): GHLApiClient { // Load configuration from environment const config: GHLConfig = { accessToken: process.env.GHL_API_KEY || '', baseUrl: process.env.GHL_BASE_URL || 'https://services.leadconnectorhq.com', version: '2021-07-28', locationId: process.env.GHL_LOCATION_ID || '' }; // Validate required configuration if (!config.accessToken) { throw new Error('GHL_API_KEY environment variable is required'); } if (!config.locationId) { throw new Error('GHL_LOCATION_ID environment variable is required'); } console.log('[GHL MCP HTTP] Initializing GHL API client...'); console.log(`[GHL MCP HTTP] Base URL: ${config.baseUrl}`); console.log(`[GHL MCP HTTP] Version: ${config.version}`); console.log(`[GHL MCP HTTP] Location ID: ${config.locationId}`); return new GHLApiClient(config); } /** * Create tool instances with a specific GHL client */ private createToolInstances(client: GHLApiClient) { return { contactTools: new ContactTools(client), conversationTools: new ConversationTools(client), blogTools: new BlogTools(client), opportunityTools: new OpportunityTools(client), calendarTools: new CalendarTools(client), emailTools: new EmailTools(client), locationTools: new LocationTools(client), emailISVTools: new EmailISVTools(client), socialMediaTools: new SocialMediaTools(client), mediaTools: new MediaTools(client), objectTools: new ObjectTools(client), associationTools: new AssociationTools(client), customFieldV2Tools: new CustomFieldV2Tools(client), workflowTools: new WorkflowTools(client), surveyTools: new SurveyTools(client), storeTools: new StoreTools(client), productsTools: new ProductsTools(client), paymentsTools: new PaymentsTools(client), invoicesTools: new InvoicesTools(client) }; } /** * Setup MCP request handlers */ private setupMCPHandlers(): void { // Handle list tools requests this.server.setRequestHandler(ListToolsRequestSchema, async () => { console.log('[GHL MCP HTTP] Listing available tools...'); try { const contactToolDefinitions = this.contactTools.getToolDefinitions(); const conversationToolDefinitions = this.conversationTools.getToolDefinitions(); const blogToolDefinitions = this.blogTools.getToolDefinitions(); const opportunityToolDefinitions = this.opportunityTools.getToolDefinitions(); const calendarToolDefinitions = this.calendarTools.getToolDefinitions(); const emailToolDefinitions = this.emailTools.getToolDefinitions(); const locationToolDefinitions = this.locationTools.getToolDefinitions(); const emailISVToolDefinitions = this.emailISVTools.getToolDefinitions(); const socialMediaToolDefinitions = this.socialMediaTools.getTools(); const mediaToolDefinitions = this.mediaTools.getToolDefinitions(); const objectToolDefinitions = this.objectTools.getToolDefinitions(); const associationToolDefinitions = this.associationTools.getTools(); const customFieldV2ToolDefinitions = this.customFieldV2Tools.getTools(); const workflowToolDefinitions = this.workflowTools.getTools(); const surveyToolDefinitions = this.surveyTools.getTools(); const storeToolDefinitions = this.storeTools.getTools(); const productsToolDefinitions = this.productsTools.getTools(); const paymentsToolDefinitions = this.paymentsTools.getTools(); const invoicesToolDefinitions = this.invoicesTools.getTools(); const allTools = [ ...contactToolDefinitions, ...conversationToolDefinitions, ...blogToolDefinitions, ...opportunityToolDefinitions, ...calendarToolDefinitions, ...emailToolDefinitions, ...locationToolDefinitions, ...emailISVToolDefinitions, ...socialMediaToolDefinitions, ...mediaToolDefinitions, ...objectToolDefinitions, ...associationToolDefinitions, ...customFieldV2ToolDefinitions, ...workflowToolDefinitions, ...surveyToolDefinitions, ...storeToolDefinitions, ...productsToolDefinitions, ...paymentsToolDefinitions, ...invoicesToolDefinitions ]; console.log(`[GHL MCP HTTP] Registered ${allTools.length} tools total`); // Monotenant: return tools as-is return { tools: allTools }; } catch (error) { console.error('[GHL MCP HTTP] Error listing tools:', error); throw new McpError( ErrorCode.InternalError, `Failed to list tools: ${error}` ); } }); // Handle tool execution requests this.server.setRequestHandler(CallToolRequestSchema, async (request) => { const { name, arguments: args } = request.params; console.log(`[GHL MCP HTTP] Executing tool: ${name}`); try { // Extract credentials from request if provided const credentials = this.credentialManager.extractCredentials(args); // Get appropriate client based on credentials let client: GHLApiClient; try { client = this.credentialManager.getClient(credentials); } catch (error) { throw new McpError( ErrorCode.InvalidRequest, `Authentication failed: ${error}` ); } // Create tool instances with the appropriate client const toolInstances = this.createToolInstances(client); let result: any; // Route to appropriate tool handler if (this.isContactTool(name)) { result = await toolInstances.contactTools.executeTool(name, args || {}); } else if (this.isConversationTool(name)) { result = await toolInstances.conversationTools.executeTool(name, args || {}); } else if (this.isBlogTool(name)) { result = await toolInstances.blogTools.executeTool(name, args || {}); } else if (this.isOpportunityTool(name)) { result = await toolInstances.opportunityTools.executeTool(name, args || {}); } else if (this.isCalendarTool(name)) { result = await toolInstances.calendarTools.executeTool(name, args || {}); } else if (this.isEmailTool(name)) { result = await toolInstances.emailTools.executeTool(name, args || {}); } else if (this.isLocationTool(name)) { result = await toolInstances.locationTools.executeTool(name, args || {}); } else if (this.isEmailISVTool(name)) { result = await toolInstances.emailISVTools.executeTool(name, args || {}); } else if (this.isSocialMediaTool(name)) { result = await toolInstances.socialMediaTools.executeTool(name, args || {}); } else if (this.isMediaTool(name)) { result = await toolInstances.mediaTools.executeTool(name, args || {}); } else if (this.isObjectTool(name)) { result = await toolInstances.objectTools.executeTool(name, args || {}); } else if (this.isAssociationTool(name)) { result = await toolInstances.associationTools.executeAssociationTool(name, args || {}); } else if (this.isCustomFieldV2Tool(name)) { result = await toolInstances.customFieldV2Tools.executeCustomFieldV2Tool(name, args || {}); } else if (this.isWorkflowTool(name)) { result = await toolInstances.workflowTools.executeWorkflowTool(name, args || {}); } else if (this.isSurveyTool(name)) { result = await toolInstances.surveyTools.executeSurveyTool(name, args || {}); } else if (this.isStoreTool(name)) { result = await toolInstances.storeTools.executeStoreTool(name, args || {}); } else if (this.isProductsTool(name)) { result = await toolInstances.productsTools.executeProductsTool(name, args || {}); } else if (this.isPaymentsTool(name)) { result = await toolInstances.paymentsTools.handleToolCall(name, args || {}); } else if (this.isInvoicesTool(name)) { result = await toolInstances.invoicesTools.handleToolCall(name, args || {}); } else { throw new Error(`Unknown tool: ${name}`); } console.log(`[GHL MCP HTTP] Tool ${name} executed successfully`); return { content: [ { type: 'text', text: JSON.stringify(result, null, 2) } ] }; } catch (error) { console.error(`[GHL MCP HTTP] Error executing tool ${name}:`, error); throw new McpError( ErrorCode.InternalError, `Tool execution failed: ${error}` ); } }); } /** * Setup HTTP routes */ private setupRoutes(): void { // Optional auth middleware (enabled when AUTH_TOKEN or MCP_AUTH_TOKEN is set) const authToken = process.env.AUTH_TOKEN || process.env.MCP_AUTH_TOKEN; const customHeaderName = (process.env.AUTH_HEADER_NAME || 'x-mcp-auth').toLowerCase(); const authenticate: express.RequestHandler = (req, res, next) => { if (!authToken) { return next(); } const authorization = req.header('authorization'); const customHeader = req.header(customHeaderName); const xApiKey = req.header('x-api-key'); // Accept either: // - Authorization: Bearer <AUTH_TOKEN> // - X-MCP-Auth: <AUTH_TOKEN> (or custom header via AUTH_HEADER_NAME) // - X-API-Key: <AUTH_TOKEN> let isAuthorized = false; if (authorization && authorization.toLowerCase().startsWith('bearer ')) { const provided = authorization.slice(7).trim(); isAuthorized = provided === authToken; } if (!isAuthorized && typeof customHeader === 'string') { isAuthorized = customHeader.trim() === authToken; } if (!isAuthorized && typeof xApiKey === 'string') { isAuthorized = xApiKey.trim() === authToken; } if (!isAuthorized) { res.status(401).json({ error: 'Unauthorized' }); return; } next(); }; // Health check endpoint this.app.get('/health', (req, res) => { res.json({ status: 'healthy', server: 'ghl-mcp-server', version: '1.0.0', timestamp: new Date().toISOString(), tools: this.getToolsCount() }); }); // MCP capabilities endpoint this.app.get('/capabilities', (req, res) => { res.json({ capabilities: { tools: {}, }, server: { name: 'ghl-mcp-server', version: '1.0.0' } }); }); // Tools listing endpoint this.app.get('/tools', async (req, res) => { try { const contactTools = this.contactTools.getToolDefinitions(); const conversationTools = this.conversationTools.getToolDefinitions(); const blogTools = this.blogTools.getToolDefinitions(); const opportunityTools = this.opportunityTools.getToolDefinitions(); const calendarTools = this.calendarTools.getToolDefinitions(); const emailTools = this.emailTools.getToolDefinitions(); const locationTools = this.locationTools.getToolDefinitions(); const emailISVTools = this.emailISVTools.getToolDefinitions(); const socialMediaTools = this.socialMediaTools.getTools(); const mediaTools = this.mediaTools.getToolDefinitions(); const objectTools = this.objectTools.getToolDefinitions(); const associationTools = this.associationTools.getTools(); const customFieldV2Tools = this.customFieldV2Tools.getTools(); const workflowTools = this.workflowTools.getTools(); const surveyTools = this.surveyTools.getTools(); const storeTools = this.storeTools.getTools(); const productsTools = this.productsTools.getTools(); const paymentsTools = this.paymentsTools.getTools(); const invoicesTools = this.invoicesTools.getTools(); const allTools = [ ...contactTools, ...conversationTools, ...blogTools, ...opportunityTools, ...calendarTools, ...emailTools, ...locationTools, ...emailISVTools, ...socialMediaTools, ...mediaTools, ...objectTools, ...associationTools, ...customFieldV2Tools, ...workflowTools, ...surveyTools, ...storeTools, ...productsTools, ...paymentsTools, ...invoicesTools ]; res.json({ tools: allTools, count: allTools.length }); } catch (error) { res.status(500).json({ error: 'Failed to list tools' }); } }); // SSE endpoint for ChatGPT MCP connection const handleSSE = async (req: express.Request, res: express.Response) => { const sessionId = req.query.sessionId || 'unknown'; console.log(`[GHL MCP HTTP] New SSE connection from: ${req.ip}, sessionId: ${sessionId}, method: ${req.method}`); try { // Create SSE transport (this will set the headers) const transport = new SSEServerTransport('/sse', res); // Connect MCP server to transport await this.server.connect(transport); console.log(`[GHL MCP HTTP] SSE connection established for session: ${sessionId}`); // Handle client disconnect req.on('close', () => { console.log(`[GHL MCP HTTP] SSE connection closed for session: ${sessionId}`); }); } catch (error) { console.error(`[GHL MCP HTTP] SSE connection error for session ${sessionId}:`, error); // Only send error response if headers haven't been sent yet if (!res.headersSent) { res.status(500).json({ error: 'Failed to establish SSE connection' }); } else { // If headers were already sent, close the connection res.end(); } } }; // Handle both GET and POST for SSE (MCP protocol requirements) // Monotenant: no auth required this.app.get('/sse', handleSSE); this.app.post('/sse', handleSSE); // n8n compatibility: expose GET /mcp for protocol probe and SSE on same path this.app.get('/mcp', async (req: express.Request, res: express.Response) => { // If client requests SSE, upgrade to event stream const accept = (req.headers['accept'] || '').toString().toLowerCase(); if (accept.includes('text/event-stream')) { return handleSSE(req, res); } // Otherwise respond with minimal protocol info res.json({ protocol: { version: '2024-11-05' }, transport: 'sse', capabilities: { tools: {} }, server: { name: 'ghl-mcp-server', version: '1.0.0' } }); }); // Minimal HTTP MCP proxy for testing with curl and n8n HTTP client // Supports basic MCP methods used by clients: initialize, tools/list, tools/call this.app.post('/mcp', async (req, res) => { try { const { method, params } = req.body || {}; if (!method) { res.status(400).json({ error: 'Missing method' }); return; } // Handshake used by HTTP Streamable clients if (method === 'initialize') { res.json({ protocol: { version: '2024-11-05' }, capabilities: { tools: {} }, server: { name: 'ghl-mcp-server', version: '1.0.0' } }); return; } if (method === 'tools/list') { const contactTools = this.contactTools.getToolDefinitions(); const conversationTools = this.conversationTools.getToolDefinitions(); const blogTools = this.blogTools.getToolDefinitions(); const opportunityTools = this.opportunityTools.getToolDefinitions(); const calendarTools = this.calendarTools.getToolDefinitions(); const emailTools = this.emailTools.getToolDefinitions(); const locationTools = this.locationTools.getToolDefinitions(); const emailISVTools = this.emailISVTools.getToolDefinitions(); const socialMediaTools = this.socialMediaTools.getTools(); const mediaTools = this.mediaTools.getToolDefinitions(); const objectTools = this.objectTools.getToolDefinitions(); const associationTools = this.associationTools.getTools(); const customFieldV2Tools = this.customFieldV2Tools.getTools(); const workflowTools = this.workflowTools.getTools(); const surveyTools = this.surveyTools.getTools(); const storeTools = this.storeTools.getTools(); const productsTools = this.productsTools.getTools(); const paymentsTools = this.paymentsTools.getTools(); const invoicesTools = this.invoicesTools.getTools(); const allTools = [ ...contactTools, ...conversationTools, ...blogTools, ...opportunityTools, ...calendarTools, ...emailTools, ...locationTools, ...emailISVTools, ...socialMediaTools, ...mediaTools, ...objectTools, ...associationTools, ...customFieldV2Tools, ...workflowTools, ...surveyTools, ...storeTools, ...productsTools, ...paymentsTools, ...invoicesTools ]; res.json({ tools: allTools }); return; } if (method === 'tools/call') { const { name, arguments: args } = params || {}; if (!name) { res.status(400).json({ error: 'Missing tool name' }); return; } // Monotenant: always use default client const toolInstances = this.createToolInstances(this.ghlClient); let result: any; if (this.isContactTool(name)) { result = await toolInstances.contactTools.executeTool(name, args || {}); } else if (this.isConversationTool(name)) { result = await toolInstances.conversationTools.executeTool(name, args || {}); } else if (this.isBlogTool(name)) { result = await toolInstances.blogTools.executeTool(name, args || {}); } else if (this.isOpportunityTool(name)) { result = await toolInstances.opportunityTools.executeTool(name, args || {}); } else if (this.isCalendarTool(name)) { result = await toolInstances.calendarTools.executeTool(name, args || {}); } else if (this.isEmailTool(name)) { result = await toolInstances.emailTools.executeTool(name, args || {}); } else if (this.isLocationTool(name)) { result = await toolInstances.locationTools.executeTool(name, args || {}); } else if (this.isEmailISVTool(name)) { result = await toolInstances.emailISVTools.executeTool(name, args || {}); } else if (this.isSocialMediaTool(name)) { result = await toolInstances.socialMediaTools.executeTool(name, args || {}); } else if (this.isMediaTool(name)) { result = await toolInstances.mediaTools.executeTool(name, args || {}); } else if (this.isObjectTool(name)) { result = await toolInstances.objectTools.executeTool(name, args || {}); } else if (this.isAssociationTool(name)) { result = await toolInstances.associationTools.executeAssociationTool(name, args || {}); } else if (this.isCustomFieldV2Tool(name)) { result = await toolInstances.customFieldV2Tools.executeCustomFieldV2Tool(name, args || {}); } else if (this.isWorkflowTool(name)) { result = await toolInstances.workflowTools.executeWorkflowTool(name, args || {}); } else if (this.isSurveyTool(name)) { result = await toolInstances.surveyTools.executeSurveyTool(name, args || {}); } else if (this.isStoreTool(name)) { result = await toolInstances.storeTools.executeStoreTool(name, args || {}); } else if (this.isProductsTool(name)) { result = await toolInstances.productsTools.executeProductsTool(name, args || {}); } else if (this.isPaymentsTool(name)) { result = await toolInstances.paymentsTools.handleToolCall(name, args || {}); } else if (this.isInvoicesTool(name)) { result = await toolInstances.invoicesTools.handleToolCall(name, args || {}); } else { res.status(400).json({ error: `Unknown tool: ${name}` }); return; } res.json({ content: [{ type: 'text', text: JSON.stringify(result, null, 2) }] }); return; } res.status(400).json({ error: `Unsupported method: ${method}` }); return; } catch (err: any) { res.status(500).json({ error: `MCP proxy error: ${err?.message || err}` }); return; } }); // Root endpoint with server info this.app.get('/', (req, res) => { res.json({ name: 'GoHighLevel MCP Server', version: '1.0.0', status: 'running', endpoints: { health: '/health', capabilities: '/capabilities', tools: '/tools', sse: '/sse' }, tools: this.getToolsCount(), documentation: 'https://github.com/your-repo/ghl-mcp-server' }); }); } /** * Get tools count summary */ private getToolsCount() { return { contact: this.contactTools.getToolDefinitions().length, conversation: this.conversationTools.getToolDefinitions().length, blog: this.blogTools.getToolDefinitions().length, opportunity: this.opportunityTools.getToolDefinitions().length, calendar: this.calendarTools.getToolDefinitions().length, email: this.emailTools.getToolDefinitions().length, location: this.locationTools.getToolDefinitions().length, emailISV: this.emailISVTools.getToolDefinitions().length, socialMedia: this.socialMediaTools.getTools().length, media: this.mediaTools.getToolDefinitions().length, objects: this.objectTools.getToolDefinitions().length, associations: this.associationTools.getTools().length, customFieldsV2: this.customFieldV2Tools.getTools().length, workflows: this.workflowTools.getTools().length, surveys: this.surveyTools.getTools().length, store: this.storeTools.getTools().length, products: this.productsTools.getTools().length, total: this.contactTools.getToolDefinitions().length + this.conversationTools.getToolDefinitions().length + this.blogTools.getToolDefinitions().length + this.opportunityTools.getToolDefinitions().length + this.calendarTools.getToolDefinitions().length + this.emailTools.getToolDefinitions().length + this.locationTools.getToolDefinitions().length + this.emailISVTools.getToolDefinitions().length + this.socialMediaTools.getTools().length + this.mediaTools.getToolDefinitions().length + this.objectTools.getToolDefinitions().length + this.associationTools.getTools().length + this.customFieldV2Tools.getTools().length + this.workflowTools.getTools().length + this.surveyTools.getTools().length + this.storeTools.getTools().length + this.productsTools.getTools().length }; } /** * Tool name validation helpers */ private isContactTool(toolName: string): boolean { const contactToolNames = [ // Basic Contact Management 'create_contact', 'search_contacts', 'get_contact', 'update_contact', 'add_contact_tags', 'remove_contact_tags', 'delete_contact', // Task Management 'get_contact_tasks', 'create_contact_task', 'get_contact_task', 'update_contact_task', 'delete_contact_task', 'update_task_completion', // Note Management 'get_contact_notes', 'create_contact_note', 'get_contact_note', 'update_contact_note', 'delete_contact_note', // Advanced Operations 'upsert_contact', 'get_duplicate_contact', 'get_contacts_by_business', 'get_contact_appointments', // Bulk Operations 'bulk_update_contact_tags', 'bulk_update_contact_business', // Followers Management 'add_contact_followers', 'remove_contact_followers', // Campaign Management 'add_contact_to_campaign', 'remove_contact_from_campaign', 'remove_contact_from_all_campaigns', // Workflow Management 'add_contact_to_workflow', 'remove_contact_from_workflow' ]; return contactToolNames.includes(toolName); } private isConversationTool(toolName: string): boolean { const conversationToolNames = [ // Basic conversation operations 'send_sms', 'send_email', 'search_conversations', 'get_conversation', 'create_conversation', 'update_conversation', 'delete_conversation', 'get_recent_messages', // Message management 'get_email_message', 'get_message', 'upload_message_attachments', 'update_message_status', // Manual message creation 'add_inbound_message', 'add_outbound_call', // Call recordings & transcriptions 'get_message_recording', 'get_message_transcription', 'download_transcription', // Scheduling management 'cancel_scheduled_message', 'cancel_scheduled_email', // Live chat features 'live_chat_typing' ]; return conversationToolNames.includes(toolName); } private isBlogTool(toolName: string): boolean { const blogToolNames = [ 'create_blog_post', 'update_blog_post', 'get_blog_posts', 'get_blog_sites', 'get_blog_authors', 'get_blog_categories', 'check_url_slug' ]; return blogToolNames.includes(toolName); } private isOpportunityTool(toolName: string): boolean { const opportunityToolNames = [ 'search_opportunities', 'get_pipelines', 'get_opportunity', 'create_opportunity', 'update_opportunity_status', 'delete_opportunity', 'update_opportunity', 'upsert_opportunity', 'add_opportunity_followers', 'remove_opportunity_followers' ]; return opportunityToolNames.includes(toolName); } private isCalendarTool(toolName: string): boolean { const calendarToolNames = [ // Calendar Groups Management 'get_calendar_groups', 'create_calendar_group', 'validate_group_slug', 'update_calendar_group', 'delete_calendar_group', 'disable_calendar_group', // Calendars 'get_calendars', 'create_calendar', 'get_calendar', 'update_calendar', 'delete_calendar', // Events and Appointments 'get_calendar_events', 'get_free_slots', 'create_appointment', 'get_appointment', 'update_appointment', 'delete_appointment', // Appointment Notes 'get_appointment_notes', 'create_appointment_note', 'update_appointment_note', 'delete_appointment_note', // Calendar Resources 'get_calendar_resources', 'get_calendar_resource_by_id', 'update_calendar_resource', 'delete_calendar_resource', // Calendar Notifications 'get_calendar_notifications', 'create_calendar_notification', 'update_calendar_notification', 'delete_calendar_notification', // Blocked Slots 'create_block_slot', 'update_block_slot', 'get_blocked_slots', 'delete_blocked_slot' ]; return calendarToolNames.includes(toolName); } private isEmailTool(toolName: string): boolean { const emailToolNames = [ 'get_email_campaigns', 'create_email_template', 'get_email_templates', 'update_email_template', 'delete_email_template' ]; return emailToolNames.includes(toolName); } private isLocationTool(toolName: string): boolean { const locationToolNames = [ // Location Management 'search_locations', 'get_location', 'create_location', 'update_location', 'delete_location', // Location Tags 'get_location_tags', 'create_location_tag', 'get_location_tag', 'update_location_tag', 'delete_location_tag', // Location Tasks 'search_location_tasks', // Custom Fields 'get_location_custom_fields', 'create_location_custom_field', 'get_location_custom_field', 'update_location_custom_field', 'delete_location_custom_field', // Custom Values 'get_location_custom_values', 'create_location_custom_value', 'get_location_custom_value', 'update_location_custom_value', 'delete_location_custom_value', // Templates 'get_location_templates', 'delete_location_template', // Timezones 'get_timezones' ]; return locationToolNames.includes(toolName); } private isEmailISVTool(toolName: string): boolean { const emailISVToolNames = [ 'verify_email' ]; return emailISVToolNames.includes(toolName); } private isSocialMediaTool(toolName: string): boolean { const socialMediaToolNames = [ // Post Management 'search_social_posts', 'create_social_post', 'get_social_post', 'update_social_post', 'delete_social_post', 'bulk_delete_social_posts', // Account Management 'get_social_accounts', 'delete_social_account', // CSV Operations 'upload_social_csv', 'get_csv_upload_status', 'set_csv_accounts', // Categories & Tags 'get_social_categories', 'get_social_category', 'get_social_tags', 'get_social_tags_by_ids', // OAuth Integration 'start_social_oauth', 'get_platform_accounts' ]; return socialMediaToolNames.includes(toolName); } private isMediaTool(toolName: string): boolean { const mediaToolNames = [ 'get_media_files', 'upload_media_file', 'delete_media_file' ]; return mediaToolNames.includes(toolName); } private isObjectTool(toolName: string): boolean { const objectToolNames = [ 'get_all_objects', 'create_object_schema', 'get_object_schema', 'update_object_schema', 'create_object_record', 'get_object_record', 'update_object_record', 'delete_object_record', 'search_object_records' ]; return objectToolNames.includes(toolName); } private isAssociationTool(toolName: string): boolean { const associationToolNames = [ 'ghl_get_all_associations', 'ghl_create_association', 'ghl_get_association_by_id', 'ghl_update_association', 'ghl_delete_association', 'ghl_get_association_by_key', 'ghl_get_association_by_object_key', 'ghl_create_relation', 'ghl_get_relations_by_record', 'ghl_delete_relation' ]; return associationToolNames.includes(toolName); } private isCustomFieldV2Tool(toolName: string): boolean { const customFieldV2ToolNames = [ 'ghl_get_custom_field_by_id', 'ghl_create_custom_field', 'ghl_update_custom_field', 'ghl_delete_custom_field', 'ghl_get_custom_fields_by_object_key', 'ghl_create_custom_field_folder', 'ghl_update_custom_field_folder', 'ghl_delete_custom_field_folder' ]; return customFieldV2ToolNames.includes(toolName); } private isWorkflowTool(toolName: string): boolean { const workflowToolNames = [ 'ghl_get_workflows' ]; return workflowToolNames.includes(toolName); } private isSurveyTool(toolName: string): boolean { const surveyToolNames = [ 'ghl_get_surveys', 'ghl_get_survey_submissions' ]; return surveyToolNames.includes(toolName); } private isStoreTool(toolName: string): boolean { const storeToolNames = [ 'ghl_create_shipping_zone', 'ghl_list_shipping_zones', 'ghl_get_shipping_zone', 'ghl_update_shipping_zone', 'ghl_delete_shipping_zone', 'ghl_get_available_shipping_rates', 'ghl_create_shipping_rate', 'ghl_list_shipping_rates', 'ghl_get_shipping_rate', 'ghl_update_shipping_rate', 'ghl_delete_shipping_rate', 'ghl_create_shipping_carrier', 'ghl_list_shipping_carriers', 'ghl_get_shipping_carrier', 'ghl_update_shipping_carrier', 'ghl_delete_shipping_carrier', 'ghl_create_store_setting', 'ghl_get_store_setting' ]; return storeToolNames.includes(toolName); } private isProductsTool(toolName: string): boolean { const productsToolNames = [ 'ghl_create_product', 'ghl_list_products', 'ghl_get_product', 'ghl_update_product', 'ghl_delete_product', 'ghl_bulk_update_products', 'ghl_create_price', 'ghl_list_prices', 'ghl_get_price', 'ghl_update_price', 'ghl_delete_price', 'ghl_list_inventory', 'ghl_update_inventory', 'ghl_get_product_store_stats', 'ghl_update_product_store', 'ghl_create_product_collection', 'ghl_list_product_collections', 'ghl_get_product_collection', 'ghl_update_product_collection', 'ghl_delete_product_collection', 'ghl_list_product_reviews', 'ghl_get_reviews_count', 'ghl_update_product_review', 'ghl_delete_product_review', 'ghl_bulk_update_product_reviews' ]; return productsToolNames.includes(toolName); } private isPaymentsTool(toolName: string): boolean { const paymentsToolNames = [ 'create_whitelabel_integration_provider', 'list_whitelabel_integration_providers', 'list_orders', 'get_order_by_id', 'create_order_fulfillment', 'list_order_fulfillments', 'list_transactions', 'get_transaction_by_id', 'list_subscriptions', 'get_subscription_by_id', 'list_coupons', 'create_coupon', 'update_coupon', 'delete_coupon', 'get_coupon', 'create_custom_provider_integration', 'delete_custom_provider_integration', 'get_custom_provider_config', 'create_custom_provider_config', 'disconnect_custom_provider_config' ]; return paymentsToolNames.includes(toolName); } private isInvoicesTool(toolName: string): boolean { const invoicesToolNames = [ 'create_invoice_template', 'list_invoice_templates', 'get_invoice_template', 'update_invoice_template', 'delete_invoice_template', 'create_invoice_schedule', 'list_invoice_schedules', 'get_invoice_schedule', 'create_invoice', 'list_invoices', 'get_invoice', 'send_invoice', 'create_estimate', 'list_estimates', 'send_estimate', 'create_invoice_from_estimate', 'generate_invoice_number', 'generate_estimate_number' ]; return invoicesToolNames.includes(toolName); } /** * Test GHL API connection */ private async testGHLConnection(): Promise<void> { try { console.log('[GHL MCP HTTP] Testing GHL API connection...'); // Skip connection test in multi-tenant mode (when no default credentials) if (!this.ghlClient) { console.log('[GHL MCP HTTP] ⚠️ No default credentials configured - running in multi-tenant mode'); console.log('[GHL MCP HTTP] Credentials will be required per request'); return; } const result = await this.ghlClient.testConnection(); console.log('[GHL MCP HTTP] ✅ GHL API connection successful'); console.log(`[GHL MCP HTTP] Connected to location: ${result.data?.locationId}`); } catch (error) { console.error('[GHL MCP HTTP] ❌ GHL API connection failed:', error); throw new Error(`Failed to connect to GHL API: ${error}`); } } /** * Start the HTTP server */ async start(): Promise<void> { console.log('🚀 Starting GoHighLevel MCP HTTP Server...'); console.log('========================================='); try { // Test GHL API connection await this.testGHLConnection(); // Start HTTP server this.app.listen(this.port, '0.0.0.0', () => { console.log('✅ GoHighLevel MCP HTTP Server started successfully!'); console.log(`🌐 Server running on: http://0.0.0.0:${this.port}`); console.log(`🔗 SSE Endpoint: http://0.0.0.0:${this.port}/sse`); console.log(`📋 Tools Available: ${this.getToolsCount().total}`); console.log('🎯 Ready for ChatGPT integration!'); console.log('========================================='); }); } catch (error) { console.error('❌ Failed to start GHL MCP HTTP Server:', error); process.exit(1); } } } /** * Handle graceful shutdown */ function setupGracefulShutdown(): void { const shutdown = (signal: string) => { console.log(`\n[GHL MCP HTTP] Received ${signal}, shutting down gracefully...`); process.exit(0); }; process.on('SIGINT', () => shutdown('SIGINT')); process.on('SIGTERM', () => shutdown('SIGTERM')); } /** * Main entry point */ async function main(): Promise<void> { try { // Setup graceful shutdown setupGracefulShutdown(); // Create and start HTTP server const server = new GHLMCPHttpServer(); await server.start(); } catch (error) { console.error('💥 Fatal error:', error); process.exit(1); } } // Start the server main().catch((error) => { console.error('Unhandled error:', 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/clickmediapropy/gohighlevel-mcp-server'

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