webhook-handler.test.tsβ’15.5 kB
/**
* Unit tests for WebhookHandler
*/
import { describe, it, expect, vi, beforeEach } from 'vitest';
import { WebhookHandler } from '../../../../src/triggers/handlers/webhook-handler';
import { N8nApiClient } from '../../../../src/services/n8n-api-client';
import { InstanceContext } from '../../../../src/types/instance-context';
import { Workflow, WebhookRequest } from '../../../../src/types/n8n-api';
import { DetectedTrigger } from '../../../../src/triggers/types';
// Mock getN8nApiConfig
vi.mock('../../../../src/config/n8n-api', () => ({
getN8nApiConfig: vi.fn(() => ({
baseUrl: 'https://test.n8n.com/api/v1',
apiKey: 'test-api-key',
})),
}));
// Mock SSRFProtection
vi.mock('../../../../src/utils/ssrf-protection', () => ({
SSRFProtection: {
validateWebhookUrl: vi.fn(async () => ({ valid: true, reason: '' })),
},
}));
// Mock buildTriggerUrl
vi.mock('../../../../src/triggers/trigger-detector', () => ({
buildTriggerUrl: vi.fn((baseUrl: string, trigger: any, mode: string) => {
return `${baseUrl}/webhook/${trigger.webhookPath}`;
}),
}));
// Create mock client
const createMockClient = (): N8nApiClient => ({
getWorkflow: vi.fn(),
listWorkflows: vi.fn(),
createWorkflow: vi.fn(),
updateWorkflow: vi.fn(),
deleteWorkflow: vi.fn(),
triggerWebhook: vi.fn(),
getExecution: vi.fn(),
listExecutions: vi.fn(),
deleteExecution: vi.fn(),
} as unknown as N8nApiClient);
// Create test workflow
const createWorkflow = (): Workflow => ({
id: 'workflow-123',
name: 'Test Workflow',
active: true,
nodes: [
{
id: 'webhook-node',
name: 'Webhook',
type: 'n8n-nodes-base.webhook',
typeVersion: 1,
position: [0, 0],
parameters: {
path: 'test-webhook',
httpMethod: 'POST',
},
},
],
connections: {},
createdAt: new Date().toISOString(),
updatedAt: new Date().toISOString(),
settings: {},
staticData: undefined,
} as Workflow);
describe('WebhookHandler', () => {
let mockClient: N8nApiClient;
let handler: WebhookHandler;
beforeEach(async () => {
mockClient = createMockClient();
handler = new WebhookHandler(mockClient);
vi.clearAllMocks();
// Import and reset mock
const { SSRFProtection } = await import('../../../../src/utils/ssrf-protection');
vi.mocked(SSRFProtection.validateWebhookUrl).mockResolvedValue({
valid: true,
reason: '',
});
});
describe('initialization', () => {
it('should have correct trigger type', () => {
expect(handler.triggerType).toBe('webhook');
});
it('should have correct capabilities', () => {
expect(handler.capabilities.requiresActiveWorkflow).toBe(true);
expect(handler.capabilities.canPassInputData).toBe(true);
expect(handler.capabilities.supportedMethods).toEqual(['GET', 'POST', 'PUT', 'DELETE']);
});
});
describe('input validation', () => {
it('should validate correct webhook input', () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
httpMethod: 'POST' as const,
webhookPath: 'test-path',
};
const result = handler.validate(input);
expect(result).toEqual(input);
});
it('should validate minimal input', () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
};
const result = handler.validate(input);
expect(result.workflowId).toBe('workflow-123');
expect(result.triggerType).toBe('webhook');
});
it('should reject invalid trigger type', () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'chat',
};
expect(() => handler.validate(input)).toThrow();
});
it('should reject invalid HTTP method', () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook',
httpMethod: 'PATCH',
};
expect(() => handler.validate(input)).toThrow();
});
it('should accept optional fields', () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
data: { key: 'value' },
headers: { 'X-Custom': 'header' },
timeout: 60000,
waitForResponse: false,
};
const result = handler.validate(input);
expect(result.data).toEqual({ key: 'value' });
expect(result.headers).toEqual({ 'X-Custom': 'header' });
expect(result.timeout).toBe(60000);
expect(result.waitForResponse).toBe(false);
});
});
describe('execute', () => {
it('should execute webhook with provided path', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'custom-path',
httpMethod: 'POST' as const,
data: { test: 'data' },
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: { result: 'success' },
});
const response = await handler.execute(input, workflow);
expect(response.success).toBe(true);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
webhookUrl: expect.stringContaining('/webhook/custom-path'),
httpMethod: 'POST',
data: { test: 'data' },
})
);
});
it('should use trigger info when no explicit path provided', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
};
const workflow = createWorkflow();
const triggerInfo: DetectedTrigger = {
type: 'webhook',
node: workflow.nodes[0],
webhookPath: 'detected-path',
httpMethod: 'GET',
};
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: { result: 'success' },
});
const response = await handler.execute(input, workflow, triggerInfo);
expect(response.success).toBe(true);
expect(mockClient.triggerWebhook).toHaveBeenCalled();
});
it('should return error when no webhook path available', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
};
const workflow = createWorkflow();
const response = await handler.execute(input, workflow);
expect(response.success).toBe(false);
expect(response.error).toContain('No webhook path available');
});
it('should return error when base URL not available', async () => {
const handlerNoContext = new WebhookHandler(mockClient, {} as InstanceContext);
// Mock getN8nApiConfig to return null
const { getN8nApiConfig } = await import('../../../../src/config/n8n-api');
vi.mocked(getN8nApiConfig).mockReturnValue(null as any);
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test',
};
const workflow = createWorkflow();
const response = await handlerNoContext.execute(input, workflow);
expect(response.success).toBe(false);
expect(response.error).toContain('Cannot determine n8n base URL');
});
it('should handle SSRF protection rejection', async () => {
const { SSRFProtection } = await import('../../../../src/utils/ssrf-protection');
vi.mocked(SSRFProtection.validateWebhookUrl).mockResolvedValue({
valid: false,
reason: 'Private IP address not allowed',
});
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
const response = await handler.execute(input, workflow);
expect(response.success).toBe(false);
expect(response.error).toContain('SSRF protection');
expect(response.error).toContain('Private IP address not allowed');
});
it('should use default POST method when not specified', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
await handler.execute(input, workflow);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
httpMethod: 'POST',
})
);
});
it('should pass custom headers', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
headers: {
'X-Custom-Header': 'custom-value',
'Authorization': 'Bearer token',
},
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
await handler.execute(input, workflow);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
headers: {
'X-Custom-Header': 'custom-value',
'Authorization': 'Bearer token',
},
})
);
});
it('should set waitForResponse from input', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
waitForResponse: false,
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 202,
statusText: 'Accepted',
data: {},
});
await handler.execute(input, workflow);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
waitForResponse: false,
})
);
});
it('should default waitForResponse to true', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
await handler.execute(input, workflow);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
waitForResponse: true,
})
);
});
it('should return response with status and metadata', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
httpMethod: 'POST' as const,
};
const workflow = createWorkflow();
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: { result: 'webhook response' },
});
const response = await handler.execute(input, workflow);
expect(response.success).toBe(true);
expect(response.status).toBe(200);
expect(response.statusText).toBe('OK');
expect(response.data).toEqual({ status: 200, statusText: 'OK', data: { result: 'webhook response' } });
expect(response.metadata?.duration).toBeGreaterThanOrEqual(0);
expect(response.metadata?.webhookPath).toBe('test-path');
expect(response.metadata?.httpMethod).toBe('POST');
});
it('should handle API errors gracefully', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
const apiError = new Error('Webhook execution failed');
vi.mocked(mockClient.triggerWebhook).mockRejectedValue(apiError);
const response = await handler.execute(input, workflow);
expect(response.success).toBe(false);
expect(response.error).toBe('Webhook execution failed');
});
it('should extract execution ID from error details', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
const apiError: any = new Error('Execution error');
apiError.details = {
executionId: 'exec-456',
message: 'Node execution failed',
};
vi.mocked(mockClient.triggerWebhook).mockRejectedValue(apiError);
const response = await handler.execute(input, workflow);
expect(response.success).toBe(false);
expect(response.executionId).toBe('exec-456');
expect(response.details).toEqual({
executionId: 'exec-456',
message: 'Node execution failed',
});
});
it('should support all HTTP methods', async () => {
const workflow = createWorkflow();
const methods: Array<'GET' | 'POST' | 'PUT' | 'DELETE'> = ['GET', 'POST', 'PUT', 'DELETE'];
for (const method of methods) {
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
httpMethod: method,
};
const response = await handler.execute(input, workflow);
expect(response.success).toBe(true);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
httpMethod: method,
})
);
}
});
it('should use httpMethod from trigger info when not in input', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
};
const workflow = createWorkflow();
const triggerInfo: DetectedTrigger = {
type: 'webhook',
node: workflow.nodes[0],
webhookPath: 'detected-path',
httpMethod: 'PUT',
};
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
await handler.execute(input, workflow, triggerInfo);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
httpMethod: 'PUT',
})
);
});
it('should prefer input httpMethod over trigger info', async () => {
const input = {
workflowId: 'workflow-123',
triggerType: 'webhook' as const,
webhookPath: 'test-path',
httpMethod: 'DELETE' as const,
};
const workflow = createWorkflow();
const triggerInfo: DetectedTrigger = {
type: 'webhook',
node: workflow.nodes[0],
webhookPath: 'detected-path',
httpMethod: 'GET',
};
vi.mocked(mockClient.triggerWebhook).mockResolvedValue({
status: 200,
statusText: 'OK',
data: {},
});
await handler.execute(input, workflow, triggerInfo);
expect(mockClient.triggerWebhook).toHaveBeenCalledWith(
expect.objectContaining({
httpMethod: 'DELETE',
})
);
});
});
});