mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-16 17:46:45 +00:00
187 lines
6.7 KiB
TypeScript
187 lines
6.7 KiB
TypeScript
import { WebhookAuthorizationError } from 'n8n-nodes-base/dist/nodes/Webhook/error';
|
||
import { validateWebhookAuthentication } from 'n8n-nodes-base/dist/nodes/Webhook/utils';
|
||
import type { INodeTypeDescription, IWebhookFunctions, IWebhookResponseData } from 'n8n-workflow';
|
||
import { NodeConnectionTypes, Node } from 'n8n-workflow';
|
||
|
||
import { getConnectedTools, nodeNameToToolName } from '@utils/helpers';
|
||
|
||
import type { CompressionResponse } from './FlushingTransport';
|
||
import { McpServerManager } from './McpServer';
|
||
|
||
const MCP_SSE_SETUP_PATH = 'sse';
|
||
const MCP_SSE_MESSAGES_PATH = 'messages';
|
||
|
||
export class McpTrigger extends Node {
|
||
description: INodeTypeDescription = {
|
||
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>. <a data-key='activate'>Activate</a> 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: NodeConnectionTypes.AiTool,
|
||
displayName: 'Tools',
|
||
},
|
||
],
|
||
outputs: [],
|
||
credentials: [
|
||
{
|
||
// eslint-disable-next-line n8n-nodes-base/node-class-description-credentials-name-unsuffixed
|
||
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,
|
||
},
|
||
],
|
||
};
|
||
|
||
async webhook(context: IWebhookFunctions): Promise<IWebhookResponseData> {
|
||
const webhookName = context.getWebhookName();
|
||
const req = context.getRequestObject();
|
||
const resp = context.getResponseObject() as unknown as CompressionResponse;
|
||
|
||
try {
|
||
await validateWebhookAuthentication(context, 'authentication');
|
||
} catch (error) {
|
||
if (error instanceof WebhookAuthorizationError) {
|
||
resp.writeHead(error.responseCode);
|
||
resp.end(error.message);
|
||
return { noWebhookResponse: true };
|
||
}
|
||
throw error;
|
||
}
|
||
const node = context.getNode();
|
||
// Get a url/tool friendly name for the server, based on the node name
|
||
const serverName = node.typeVersion > 1 ? nodeNameToToolName(node) : 'n8n-mcp-server';
|
||
|
||
const mcpServerManager: McpServerManager = McpServerManager.instance(context.logger);
|
||
|
||
if (webhookName === 'setup') {
|
||
// Sets up the transport and opens the long-lived connection. This resp
|
||
// will stay streaming, and is the channel that sends the events
|
||
|
||
// Prior to version 2.0, we use different paths for the setup and messages.
|
||
const postUrl =
|
||
node.typeVersion < 2
|
||
? req.path.replace(new RegExp(`/${MCP_SSE_SETUP_PATH}$`), `/${MCP_SSE_MESSAGES_PATH}`)
|
||
: req.path;
|
||
await mcpServerManager.createServerWithSSETransport(serverName, postUrl, resp);
|
||
|
||
return { noWebhookResponse: true };
|
||
} else if (webhookName === 'default') {
|
||
// Here we handle POST requests. These can be either
|
||
// 1) Client calls in an established session using the SSE transport, or
|
||
// 2) Client calls in an established session using the StreamableHTTPServerTransport
|
||
// 3) Session setup requests using the StreamableHTTPServerTransport
|
||
|
||
// Check if there is a session and a transport is already established
|
||
const sessionId = mcpServerManager.getSessionId(req);
|
||
if (sessionId && mcpServerManager.getTransport(sessionId)) {
|
||
const connectedTools = await getConnectedTools(context, true);
|
||
const wasToolCall = await mcpServerManager.handlePostMessage(req, resp, connectedTools);
|
||
if (wasToolCall) return { noWebhookResponse: true, workflowData: [[{ json: {} }]] };
|
||
} else {
|
||
// If no session is established, this is a setup request
|
||
// for the StreamableHTTPServerTransport, so we create a new transport
|
||
await mcpServerManager.createServerWithStreamableHTTPTransport(serverName, resp, req);
|
||
}
|
||
return { noWebhookResponse: true };
|
||
}
|
||
|
||
return { workflowData: [[{ json: {} }]] };
|
||
}
|
||
}
|