mirror of
https://github.com/n8n-io/n8n.git
synced 2024-12-25 12:44:07 -08:00
241 lines
6.3 KiB
TypeScript
241 lines
6.3 KiB
TypeScript
import { readFileSync } from 'fs';
|
|
import { agent as testAgent } from 'supertest';
|
|
import type { INodeType, INodeTypeDescription, IWebhookFunctions } from 'n8n-workflow';
|
|
|
|
import { AbstractServer } from '@/AbstractServer';
|
|
import { ExternalHooks } from '@/ExternalHooks';
|
|
import { NodeTypes } from '@/NodeTypes';
|
|
import { Push } from '@/push';
|
|
import type { WorkflowEntity } from '@db/entities/WorkflowEntity';
|
|
|
|
import { mockInstance } from '../shared/mocking';
|
|
import { initActiveWorkflowManager } from './shared/utils';
|
|
import * as testDb from './shared/testDb';
|
|
import { createUser } from './shared/db/users';
|
|
import { createWorkflow } from './shared/db/workflows';
|
|
import type { SuperAgentTest } from './shared/types';
|
|
import { Telemetry } from '@/telemetry';
|
|
|
|
mockInstance(Telemetry);
|
|
|
|
describe('Webhook API', () => {
|
|
mockInstance(ExternalHooks);
|
|
mockInstance(Push);
|
|
|
|
let agent: SuperAgentTest;
|
|
|
|
beforeAll(async () => {
|
|
await testDb.init();
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await testDb.terminate();
|
|
});
|
|
|
|
describe('Content-Type support', () => {
|
|
beforeAll(async () => {
|
|
const node = new WebhookTestingNode();
|
|
const user = await createUser();
|
|
await createWorkflow(createWebhookWorkflow(node), user);
|
|
|
|
const nodeTypes = mockInstance(NodeTypes);
|
|
nodeTypes.getByName.mockReturnValue(node);
|
|
nodeTypes.getByNameAndVersion.mockReturnValue(node);
|
|
|
|
await initActiveWorkflowManager();
|
|
|
|
const server = new (class extends AbstractServer {})();
|
|
await server.start();
|
|
agent = testAgent(server.app);
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await testDb.truncate(['Workflow']);
|
|
});
|
|
|
|
test('should handle JSON', async () => {
|
|
const response = await agent.post('/webhook/abcd').send({ test: true });
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body).toEqual({
|
|
type: 'application/json',
|
|
body: { test: true },
|
|
params: {},
|
|
});
|
|
});
|
|
|
|
test('should handle XML', async () => {
|
|
const response = await agent
|
|
.post('/webhook/abcd')
|
|
.set('content-type', 'application/xml')
|
|
.send(
|
|
'<?xml version="1.0" encoding="UTF-8"?><Outer attr="test"><Inner>value</Inner></Outer>',
|
|
);
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body).toEqual({
|
|
type: 'application/xml',
|
|
body: {
|
|
outer: {
|
|
$: {
|
|
attr: 'test',
|
|
},
|
|
inner: 'value',
|
|
},
|
|
},
|
|
params: {},
|
|
});
|
|
});
|
|
|
|
test('should handle form-urlencoded', async () => {
|
|
const response = await agent
|
|
.post('/webhook/abcd')
|
|
.set('content-type', 'application/x-www-form-urlencoded')
|
|
.send('x=5&y=str&z=false');
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body).toEqual({
|
|
type: 'application/x-www-form-urlencoded',
|
|
body: { x: '5', y: 'str', z: 'false' },
|
|
params: {},
|
|
});
|
|
});
|
|
|
|
test('should handle plain text', async () => {
|
|
const response = await agent
|
|
.post('/webhook/abcd')
|
|
.set('content-type', 'text/plain')
|
|
.send('{"key": "value"}');
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body).toEqual({
|
|
type: 'text/plain',
|
|
body: '{"key": "value"}',
|
|
params: {},
|
|
});
|
|
});
|
|
|
|
test('should handle multipart/form-data', async () => {
|
|
const response = await agent
|
|
.post('/webhook/abcd')
|
|
.field('field1', 'value1')
|
|
.field('field2', 'value2')
|
|
.field('field2', 'value3')
|
|
.attach('file1', Buffer.from('random-text'))
|
|
.attach('file2', Buffer.from('random-text'))
|
|
.attach('file2', Buffer.from('random-text'))
|
|
.set('content-type', 'multipart/form-data');
|
|
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body.type).toEqual('multipart/form-data');
|
|
const { data, files } = response.body.body;
|
|
expect(data).toEqual({ field1: 'value1', field2: ['value2', 'value3'] });
|
|
|
|
expect(files.file1).not.toBeInstanceOf(Array);
|
|
expect(files.file1.mimetype).toEqual('application/octet-stream');
|
|
expect(readFileSync(files.file1.filepath, 'utf-8')).toEqual('random-text');
|
|
expect(files.file2).toBeInstanceOf(Array);
|
|
expect(files.file2.length).toEqual(2);
|
|
});
|
|
});
|
|
|
|
describe('Params support', () => {
|
|
beforeAll(async () => {
|
|
const node = new WebhookTestingNode();
|
|
const user = await createUser();
|
|
await createWorkflow(createWebhookWorkflow(node, ':variable', 'PATCH'), user);
|
|
|
|
const nodeTypes = mockInstance(NodeTypes);
|
|
nodeTypes.getByName.mockReturnValue(node);
|
|
nodeTypes.getByNameAndVersion.mockReturnValue(node);
|
|
|
|
await initActiveWorkflowManager();
|
|
|
|
const server = new (class extends AbstractServer {})();
|
|
await server.start();
|
|
agent = testAgent(server.app);
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await testDb.truncate(['Workflow']);
|
|
});
|
|
|
|
test('should handle params', async () => {
|
|
const response = await agent
|
|
.patch('/webhook/5ccef736-be16-4d10-b7fb-feed7a61ff22/test')
|
|
.send({ test: true });
|
|
expect(response.statusCode).toEqual(200);
|
|
expect(response.body).toEqual({
|
|
type: 'application/json',
|
|
body: { test: true },
|
|
params: {
|
|
variable: 'test',
|
|
},
|
|
});
|
|
|
|
await agent.post('/webhook/abcd').send({ test: true }).expect(404);
|
|
});
|
|
});
|
|
|
|
class WebhookTestingNode implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Webhook Testing Node',
|
|
name: 'webhook-testing-node',
|
|
group: ['trigger'],
|
|
version: 1,
|
|
description: '',
|
|
defaults: {},
|
|
inputs: [],
|
|
outputs: ['main'],
|
|
webhooks: [
|
|
{
|
|
name: 'default',
|
|
isFullPath: true,
|
|
httpMethod: '={{$parameter["httpMethod"]}}',
|
|
path: '={{$parameter["path"]}}',
|
|
},
|
|
],
|
|
properties: [
|
|
{
|
|
name: 'httpMethod',
|
|
type: 'string',
|
|
displayName: 'Method',
|
|
default: 'GET',
|
|
},
|
|
{
|
|
displayName: 'Path',
|
|
name: 'path',
|
|
type: 'string',
|
|
default: 'xyz',
|
|
},
|
|
],
|
|
};
|
|
|
|
async webhook(this: IWebhookFunctions) {
|
|
const req = this.getRequestObject();
|
|
return {
|
|
webhookResponse: {
|
|
type: req.contentType,
|
|
body: req.body,
|
|
params: req.params,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
const createWebhookWorkflow = (
|
|
node: WebhookTestingNode,
|
|
path = 'abcd',
|
|
httpMethod = 'POST',
|
|
): Partial<WorkflowEntity> => ({
|
|
active: true,
|
|
nodes: [
|
|
{
|
|
name: 'Webhook',
|
|
type: node.description.name,
|
|
typeVersion: 1,
|
|
parameters: { httpMethod, path },
|
|
id: '74786112-fb73-4d80-bd9a-43982939b801',
|
|
webhookId: '5ccef736-be16-4d10-b7fb-feed7a61ff22',
|
|
position: [740, 420],
|
|
},
|
|
],
|
|
});
|
|
});
|