UNPKG

@n8n/n8n-nodes-langchain

Version:

![Banner image](https://user-images.githubusercontent.com/10284570/173569848-c624317f-42b1-45a6-ab09-f0ea3c247648.png)

200 lines 9.41 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.McpTrigger = void 0; const McpServer_1 = require("./McpServer"); const error_1 = require("n8n-nodes-base/dist/nodes/Webhook/error"); const utils_1 = require("n8n-nodes-base/dist/nodes/Webhook/utils"); const n8n_workflow_1 = require("n8n-workflow"); const helpers_1 = require("../../../utils/helpers"); const MCP_SSE_SETUP_PATH = 'sse'; const MCP_SSE_MESSAGES_PATH = 'messages'; class McpTrigger extends n8n_workflow_1.Node { constructor() { super(...arguments); this.description = { displayName: 'MCP Server Trigger', name: 'mcpTrigger', icon: { light: 'file:../mcp.svg', dark: 'file:../mcp.dark.svg', }, group: ['trigger'], version: [1, 1.1, 2], description: 'Expose n8n tools as an MCP Server endpoint', activationMessage: 'You can now connect your MCP Clients to the URL, using SSE or Streamable HTTP transports.', defaults: { name: 'MCP Server Trigger', }, codex: { categories: ['AI', 'Core Nodes'], subcategories: { AI: ['Root Nodes', 'Model Context Protocol'], 'Core Nodes': ['Other Trigger Nodes'], }, alias: ['Model Context Protocol', 'MCP Server'], resources: { primaryDocumentation: [ { url: 'https://docs.n8n.io/integrations/builtin/core-nodes/n8n-nodes-langchain.mcptrigger/', }, ], }, }, triggerPanel: { header: 'Listen for MCP events', executionsHelp: { inactive: "This trigger has two modes: test and production.<br /><br /><b>Use test mode while you build your workflow</b>. Click the 'execute step' button, then make an MCP request to the test URL. The executions will show up in the editor.<br /><br /><b>Use production mode to run your workflow automatically</b>. Publish the workflow, then make requests to the production URL. These executions will show up in the <a data-key='executions'>executions list</a>, but not the editor.", active: "This trigger has two modes: test and production.<br /><br /><b>Use test mode while you build your workflow</b>. Click the 'execute step' button, then make an MCP request to the test URL. The executions will show up in the editor.<br /><br /><b>Use production mode to run your workflow automatically</b>. Since your workflow is activated, you can make requests to the production URL. These executions will show up in the <a data-key='executions'>executions list</a>, but not the editor.", }, activationHint: "Once you've finished building your workflow, run it without having to click this button by using the production URL.", }, inputs: [ { type: n8n_workflow_1.NodeConnectionTypes.AiTool, displayName: 'Tools', }, ], outputs: [], credentials: [ { name: 'httpBearerAuth', required: true, displayOptions: { show: { authentication: ['bearerAuth'], }, }, }, { name: 'httpHeaderAuth', required: true, displayOptions: { show: { authentication: ['headerAuth'], }, }, }, ], properties: [ { displayName: 'Authentication', name: 'authentication', type: 'options', options: [ { name: 'None', value: 'none' }, { name: 'Bearer Auth', value: 'bearerAuth' }, { name: 'Header Auth', value: 'headerAuth' }, ], default: 'none', description: 'The way to authenticate', }, { displayName: 'Path', name: 'path', type: 'string', default: '', placeholder: 'webhook', required: true, description: 'The base path for this MCP server', }, ], webhooks: [ { name: 'setup', httpMethod: 'GET', responseMode: 'onReceived', isFullPath: true, path: `={{$parameter["path"]}}{{parseFloat($nodeVersion)<2 ? '/${MCP_SSE_SETUP_PATH}' : ''}}`, nodeType: 'mcp', ndvHideMethod: true, ndvHideUrl: false, }, { name: 'default', httpMethod: 'POST', responseMode: 'onReceived', isFullPath: true, path: `={{$parameter["path"]}}{{parseFloat($nodeVersion)<2 ? '/${MCP_SSE_MESSAGES_PATH}' : ''}}`, nodeType: 'mcp', ndvHideMethod: true, ndvHideUrl: true, }, { name: 'default', httpMethod: 'DELETE', responseMode: 'onReceived', isFullPath: true, path: '={{$parameter["path"]}}', nodeType: 'mcp', ndvHideMethod: true, ndvHideUrl: true, }, ], }; } async webhook(context) { const webhookName = context.getWebhookName(); const req = context.getRequestObject(); const resp = context.getResponseObject(); try { await (0, utils_1.validateWebhookAuthentication)(context, 'authentication'); } catch (error) { if (error instanceof error_1.WebhookAuthorizationError) { resp.writeHead(error.responseCode); resp.end(error.message); return { noWebhookResponse: true }; } throw error; } const node = context.getNode(); const serverName = node.typeVersion > 1 ? (0, n8n_workflow_1.nodeNameToToolName)(node) : 'n8n-mcp-server'; const mcpServer = McpServer_1.McpServer.instance(context.logger); if (webhookName === 'setup') { const postUrl = node.typeVersion < 2 ? req.path.replace(new RegExp(`/${MCP_SSE_SETUP_PATH}$`), `/${MCP_SSE_MESSAGES_PATH}`) : req.path; const connectedTools = await (0, helpers_1.getConnectedTools)(context, true); await mcpServer.handleSetupRequest(req, resp, serverName, postUrl, connectedTools); return { noWebhookResponse: true }; } else if (webhookName === 'default') { if (req.method === 'DELETE') { await mcpServer.handleDeleteRequest(req, resp); } else { const sessionId = mcpServer.getSessionId(req); context.logger.debug('MCP POST request received for existing session'); if (sessionId) { const connectedTools = await (0, helpers_1.getConnectedTools)(context, true); const { wasToolCall, toolCallInfo, messageId, relaySessionId, needsListToolsRelay } = await mcpServer.handlePostMessage(req, resp, connectedTools, serverName); if (wasToolCall) { const workflowData = { ...(toolCallInfo && { mcpToolCall: toolCallInfo }), ...(messageId && { mcpMessageId: messageId }), }; return { noWebhookResponse: true, workflowData: [[{ json: workflowData }]] }; } if (needsListToolsRelay && relaySessionId && messageId) { const workflowData = { mcpListToolsRelay: { sessionId: relaySessionId, messageId, marker: McpServer_1.MCP_LIST_TOOLS_REQUEST_MARKER, }, }; return { noWebhookResponse: true, workflowData: [[{ json: workflowData }]] }; } } else { const connectedTools = await (0, helpers_1.getConnectedTools)(context, true); await mcpServer.handleStreamableHttpSetup(req, resp, serverName, connectedTools); } } return { noWebhookResponse: true }; } return { workflowData: [[{ json: {} }]] }; } } exports.McpTrigger = McpTrigger; //# sourceMappingURL=McpTrigger.node.js.map