mirror of
https://github.com/czlonkowski/n8n-mcp.git
synced 2026-01-30 06:22:04 +00:00
Replace generic "Please try again later or contact support" error messages
with actionable guidance that directs users to use n8n_get_execution with
mode='preview' for efficient debugging.
## Changes
### Core Functionality
- Add formatExecutionError() to create execution-specific error messages
- Add formatNoExecutionError() for cases without execution context
- Update handleTriggerWebhookWorkflow to extract execution/workflow IDs from errors
- Modify getUserFriendlyErrorMessage to avoid generic SERVER_ERROR message
### Type Updates
- Add executionId and workflowId optional fields to McpToolResponse
- Add errorHandling optional field to ToolDocumentation.full
### Error Message Format
**With Execution ID:**
"Workflow {workflowId} execution {executionId} failed. Use n8n_get_execution({id: '{executionId}', mode: 'preview'}) to investigate the error."
**Without Execution ID:**
"Workflow failed to execute. Use n8n_list_executions to find recent executions, then n8n_get_execution with mode='preview' to investigate."
### Testing
- Add comprehensive tests in tests/unit/utils/n8n-errors.test.ts (20 tests)
- Add 10 new tests for handleTriggerWebhookWorkflow in handlers-n8n-manager.test.ts
- Update existing health check test to expect new error message format
- All tests passing (52 total tests)
### Documentation
- Update n8n-trigger-webhook-workflow tool documentation with errorHandling section
- Document why mode='preview' is recommended (fast, efficient, safe)
- Add example error responses and investigation workflow
## Why mode='preview'?
- Fast: <50ms response time
- Efficient: ~500 tokens (vs 50K+ for full mode)
- Safe: No timeout or token limit risks
- Informative: Shows structure, counts, and error details
## Breaking Changes
None - backward compatible improvement to error messages only.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
172 lines
6.3 KiB
TypeScript
172 lines
6.3 KiB
TypeScript
import { describe, it, expect } from 'vitest';
|
|
import {
|
|
formatExecutionError,
|
|
formatNoExecutionError,
|
|
getUserFriendlyErrorMessage,
|
|
N8nApiError,
|
|
N8nAuthenticationError,
|
|
N8nNotFoundError,
|
|
N8nValidationError,
|
|
N8nRateLimitError,
|
|
N8nServerError
|
|
} from '../../../src/utils/n8n-errors';
|
|
|
|
describe('formatExecutionError', () => {
|
|
it('should format error with both execution ID and workflow ID', () => {
|
|
const result = formatExecutionError('exec_12345', 'wf_abc');
|
|
|
|
expect(result).toBe("Workflow wf_abc execution exec_12345 failed. Use n8n_get_execution({id: 'exec_12345', mode: 'preview'}) to investigate the error.");
|
|
expect(result).toContain('mode: \'preview\'');
|
|
expect(result).toContain('exec_12345');
|
|
expect(result).toContain('wf_abc');
|
|
});
|
|
|
|
it('should format error with only execution ID', () => {
|
|
const result = formatExecutionError('exec_67890');
|
|
|
|
expect(result).toBe("Execution exec_67890 failed. Use n8n_get_execution({id: 'exec_67890', mode: 'preview'}) to investigate the error.");
|
|
expect(result).toContain('mode: \'preview\'');
|
|
expect(result).toContain('exec_67890');
|
|
expect(result).not.toContain('Workflow');
|
|
});
|
|
|
|
it('should include preview mode guidance', () => {
|
|
const result = formatExecutionError('test_id');
|
|
|
|
expect(result).toMatch(/mode:\s*'preview'/);
|
|
});
|
|
|
|
it('should format with undefined workflow ID (treated as missing)', () => {
|
|
const result = formatExecutionError('exec_123', undefined);
|
|
|
|
expect(result).toBe("Execution exec_123 failed. Use n8n_get_execution({id: 'exec_123', mode: 'preview'}) to investigate the error.");
|
|
});
|
|
|
|
it('should properly escape execution ID in suggestion', () => {
|
|
const result = formatExecutionError('exec-with-special_chars.123');
|
|
|
|
expect(result).toContain("id: 'exec-with-special_chars.123'");
|
|
});
|
|
});
|
|
|
|
describe('formatNoExecutionError', () => {
|
|
it('should provide guidance to check recent executions', () => {
|
|
const result = formatNoExecutionError();
|
|
|
|
expect(result).toBe("Workflow failed to execute. Use n8n_list_executions to find recent executions, then n8n_get_execution with mode='preview' to investigate.");
|
|
expect(result).toContain('n8n_list_executions');
|
|
expect(result).toContain('n8n_get_execution');
|
|
expect(result).toContain("mode='preview'");
|
|
});
|
|
|
|
it('should include preview mode in guidance', () => {
|
|
const result = formatNoExecutionError();
|
|
|
|
expect(result).toMatch(/mode\s*=\s*'preview'/);
|
|
});
|
|
});
|
|
|
|
describe('getUserFriendlyErrorMessage', () => {
|
|
it('should handle authentication error', () => {
|
|
const error = new N8nAuthenticationError('Invalid API key');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Failed to authenticate with n8n. Please check your API key.');
|
|
});
|
|
|
|
it('should handle not found error', () => {
|
|
const error = new N8nNotFoundError('Workflow', '123');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toContain('not found');
|
|
});
|
|
|
|
it('should handle validation error', () => {
|
|
const error = new N8nValidationError('Missing required field');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Invalid request: Missing required field');
|
|
});
|
|
|
|
it('should handle rate limit error', () => {
|
|
const error = new N8nRateLimitError(60);
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Too many requests. Please wait a moment and try again.');
|
|
});
|
|
|
|
it('should handle server error with custom message', () => {
|
|
const error = new N8nServerError('Database connection failed', 503);
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Database connection failed');
|
|
});
|
|
|
|
it('should handle server error without message', () => {
|
|
const error = new N8nApiError('', 500, 'SERVER_ERROR');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('n8n server error occurred');
|
|
});
|
|
|
|
it('should handle no response error', () => {
|
|
const error = new N8nApiError('Network error', undefined, 'NO_RESPONSE');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Unable to connect to n8n. Please check the server URL and ensure n8n is running.');
|
|
});
|
|
|
|
it('should handle unknown error with message', () => {
|
|
const error = new N8nApiError('Custom error message');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('Custom error message');
|
|
});
|
|
|
|
it('should handle unknown error without message', () => {
|
|
const error = new N8nApiError('');
|
|
const message = getUserFriendlyErrorMessage(error);
|
|
|
|
expect(message).toBe('An unexpected error occurred');
|
|
});
|
|
});
|
|
|
|
describe('Error message integration', () => {
|
|
it('should use formatExecutionError for webhook failures with execution ID', () => {
|
|
const executionId = 'exec_webhook_123';
|
|
const workflowId = 'wf_webhook_abc';
|
|
const message = formatExecutionError(executionId, workflowId);
|
|
|
|
expect(message).toContain('Workflow wf_webhook_abc execution exec_webhook_123 failed');
|
|
expect(message).toContain('n8n_get_execution');
|
|
expect(message).toContain("mode: 'preview'");
|
|
});
|
|
|
|
it('should use formatNoExecutionError for server errors without execution context', () => {
|
|
const message = formatNoExecutionError();
|
|
|
|
expect(message).toContain('Workflow failed to execute');
|
|
expect(message).toContain('n8n_list_executions');
|
|
expect(message).toContain('n8n_get_execution');
|
|
});
|
|
|
|
it('should not include "contact support" in any error message', () => {
|
|
const executionMessage = formatExecutionError('test');
|
|
const noExecutionMessage = formatNoExecutionError();
|
|
const serverError = new N8nServerError();
|
|
const serverErrorMessage = getUserFriendlyErrorMessage(serverError);
|
|
|
|
expect(executionMessage.toLowerCase()).not.toContain('contact support');
|
|
expect(noExecutionMessage.toLowerCase()).not.toContain('contact support');
|
|
expect(serverErrorMessage.toLowerCase()).not.toContain('contact support');
|
|
});
|
|
|
|
it('should always guide users to use preview mode first', () => {
|
|
const executionMessage = formatExecutionError('test');
|
|
const noExecutionMessage = formatNoExecutionError();
|
|
|
|
expect(executionMessage).toContain("mode: 'preview'");
|
|
expect(noExecutionMessage).toContain("mode='preview'");
|
|
});
|
|
});
|