Files
n8n-mcp/src/mcp/tools-n8n-friendly.ts
czlonkowski 3fec6813f3 feat: implement n8n integration improvements and protocol version negotiation
- Add intelligent protocol version negotiation (2024-11-05 for n8n, 2025-03-26 for standard clients)
- Fix memory leak potential with async cleanup and connection close handling
- Enhance error sanitization for production environments
- Add schema validation for n8n nested output workaround
- Improve Docker security with unpredictable UIDs/GIDs
- Create n8n-friendly tool descriptions to reduce schema validation errors
- Add comprehensive protocol negotiation test suite

Addresses code review feedback:
- Protocol version inconsistency resolved
- Memory management improved
- Error information leakage fixed
- Docker security enhanced

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-08-01 14:23:48 +02:00

175 lines
6.2 KiB
TypeScript

/**
* n8n-friendly tool descriptions
* These descriptions are optimized to reduce schema validation errors in n8n's AI Agent
*
* Key principles:
* 1. Use exact JSON examples in descriptions
* 2. Be explicit about data types
* 3. Keep descriptions short and directive
* 4. Avoid ambiguity
*/
export const n8nFriendlyDescriptions: Record<string, {
description: string;
params: Record<string, string>;
}> = {
// Validation tools - most prone to errors
validate_node_operation: {
description: 'Validate n8n node. ALWAYS pass two parameters: nodeType (string) and config (object). Example call: {"nodeType": "nodes-base.slack", "config": {"resource": "channel", "operation": "create"}}',
params: {
nodeType: 'String value like "nodes-base.slack"',
config: 'Object value like {"resource": "channel", "operation": "create"} or empty object {}',
profile: 'Optional string: "minimal" or "runtime" or "ai-friendly" or "strict"'
}
},
validate_node_minimal: {
description: 'Check required fields. MUST pass: nodeType (string) and config (object). Example: {"nodeType": "nodes-base.webhook", "config": {}}',
params: {
nodeType: 'String like "nodes-base.webhook"',
config: 'Object, use {} for empty'
}
},
// Search and info tools
search_nodes: {
description: 'Search nodes. Pass query (string). Example: {"query": "webhook"}',
params: {
query: 'String keyword like "webhook" or "database"',
limit: 'Optional number, default 20'
}
},
get_node_info: {
description: 'Get node details. Pass nodeType (string). Example: {"nodeType": "nodes-base.httpRequest"}',
params: {
nodeType: 'String with prefix like "nodes-base.httpRequest"'
}
},
get_node_essentials: {
description: 'Get node basics. Pass nodeType (string). Example: {"nodeType": "nodes-base.slack"}',
params: {
nodeType: 'String with prefix like "nodes-base.slack"'
}
},
// Task tools
get_node_for_task: {
description: 'Find node for task. Pass task (string). Example: {"task": "send_http_request"}',
params: {
task: 'String task name like "send_http_request"'
}
},
list_tasks: {
description: 'List tasks by category. Pass category (string). Example: {"category": "HTTP/API"}',
params: {
category: 'String: "HTTP/API" or "Webhooks" or "Database" or "AI/LangChain" or "Data Processing" or "Communication"'
}
},
// Workflow validation
validate_workflow: {
description: 'Validate workflow. Pass workflow object. MUST have: {"workflow": {"nodes": [array of node objects], "connections": {object with node connections}}}. Each node needs: name, type, typeVersion, position.',
params: {
workflow: 'Object with two required fields: nodes (array) and connections (object). Example: {"nodes": [{"name": "Webhook", "type": "n8n-nodes-base.webhook", "typeVersion": 2, "position": [250, 300], "parameters": {}}], "connections": {}}',
options: 'Optional object. Example: {"validateNodes": true, "profile": "runtime"}'
}
},
validate_workflow_connections: {
description: 'Validate workflow connections only. Pass workflow object. Example: {"workflow": {"nodes": [...], "connections": {}}}',
params: {
workflow: 'Object with nodes array and connections object. Minimal example: {"nodes": [{"name": "Webhook"}], "connections": {}}'
}
},
validate_workflow_expressions: {
description: 'Validate n8n expressions in workflow. Pass workflow object. Example: {"workflow": {"nodes": [...], "connections": {}}}',
params: {
workflow: 'Object with nodes array and connections object containing n8n expressions like {{ $json.data }}'
}
},
// Property tools
get_property_dependencies: {
description: 'Get field dependencies. Pass nodeType (string) and optional config (object). Example: {"nodeType": "nodes-base.httpRequest", "config": {}}',
params: {
nodeType: 'String like "nodes-base.httpRequest"',
config: 'Optional object, use {} for empty'
}
},
// AI tool info
get_node_as_tool_info: {
description: 'Get AI tool usage. Pass nodeType (string). Example: {"nodeType": "nodes-base.slack"}',
params: {
nodeType: 'String with prefix like "nodes-base.slack"'
}
},
// Template tools
search_templates: {
description: 'Search workflow templates. Pass query (string). Example: {"query": "chatbot"}',
params: {
query: 'String keyword like "chatbot" or "webhook"',
limit: 'Optional number, default 20'
}
},
get_template: {
description: 'Get template by ID. Pass templateId (number). Example: {"templateId": 1234}',
params: {
templateId: 'Number ID like 1234'
}
},
// Documentation tool
tools_documentation: {
description: 'Get tool docs. Pass optional depth (string). Example: {"depth": "essentials"} or {}',
params: {
depth: 'Optional string: "essentials" or "overview" or "detailed"',
topic: 'Optional string topic name'
}
}
};
/**
* Apply n8n-friendly descriptions to tools
* This function modifies tool descriptions to be more explicit for n8n's AI agent
*/
export function makeToolsN8nFriendly(tools: any[]): any[] {
return tools.map(tool => {
const toolName = tool.name as string;
const friendlyDesc = n8nFriendlyDescriptions[toolName];
if (friendlyDesc) {
// Clone the tool to avoid mutating the original
const updatedTool = { ...tool };
// Update the main description
updatedTool.description = friendlyDesc.description;
// Clone inputSchema if it exists
if (tool.inputSchema?.properties) {
updatedTool.inputSchema = {
...tool.inputSchema,
properties: { ...tool.inputSchema.properties }
};
// Update parameter descriptions
Object.keys(updatedTool.inputSchema.properties).forEach(param => {
if (friendlyDesc.params[param]) {
updatedTool.inputSchema.properties[param] = {
...updatedTool.inputSchema.properties[param],
description: friendlyDesc.params[param]
};
}
});
}
return updatedTool;
}
return tool;
});
}