mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 10:02:05 +00:00
fix(core): Revert back to the extended query-parser on express 5 (#15016)
Co-authored-by: Iván Ovejero <ivov.src@gmail.com>
This commit is contained in:
committed by
GitHub
parent
c0b54832b3
commit
9541b5bb07
@@ -66,7 +66,7 @@ export abstract class AbstractServer {
|
|||||||
constructor() {
|
constructor() {
|
||||||
this.app = express();
|
this.app = express();
|
||||||
this.app.disable('x-powered-by');
|
this.app.disable('x-powered-by');
|
||||||
|
this.app.set('query parser', 'extended');
|
||||||
this.app.engine('handlebars', expressHandlebars({ defaultLayout: false }));
|
this.app.engine('handlebars', expressHandlebars({ defaultLayout: false }));
|
||||||
this.app.set('view engine', 'handlebars');
|
this.app.set('view engine', 'handlebars');
|
||||||
this.app.set('views', TEMPLATES_DIR);
|
this.app.set('views', TEMPLATES_DIR);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { readFileSync } from 'fs';
|
import { readFileSync } from 'fs';
|
||||||
import type { IWorkflowBase } from 'n8n-workflow';
|
import { mock } from 'jest-mock-extended';
|
||||||
|
import type { INode, IWorkflowBase } from 'n8n-workflow';
|
||||||
import {
|
import {
|
||||||
NodeConnectionTypes,
|
NodeConnectionTypes,
|
||||||
type INodeType,
|
type INodeType,
|
||||||
@@ -8,10 +9,8 @@ import {
|
|||||||
} from 'n8n-workflow';
|
} from 'n8n-workflow';
|
||||||
import { agent as testAgent } from 'supertest';
|
import { agent as testAgent } from 'supertest';
|
||||||
|
|
||||||
import { ExternalHooks } from '@/external-hooks';
|
import type { User } from '@/databases/entities/user';
|
||||||
import { NodeTypes } from '@/node-types';
|
import { NodeTypes } from '@/node-types';
|
||||||
import { Push } from '@/push';
|
|
||||||
import { Telemetry } from '@/telemetry';
|
|
||||||
import { WebhookServer } from '@/webhooks/webhook-server';
|
import { WebhookServer } from '@/webhooks/webhook-server';
|
||||||
|
|
||||||
import { createUser } from './shared/db/users';
|
import { createUser } from './shared/db/users';
|
||||||
@@ -23,16 +22,78 @@ import { mockInstance } from '../shared/mocking';
|
|||||||
|
|
||||||
jest.unmock('node:fs');
|
jest.unmock('node:fs');
|
||||||
|
|
||||||
mockInstance(Telemetry);
|
class WebhookTestingNode implements INodeType {
|
||||||
|
description: INodeTypeDescription = {
|
||||||
|
displayName: 'Webhook Testing Node',
|
||||||
|
name: 'webhook-testing-node',
|
||||||
|
group: ['trigger'],
|
||||||
|
version: 1,
|
||||||
|
description: '',
|
||||||
|
defaults: {},
|
||||||
|
inputs: [],
|
||||||
|
outputs: [NodeConnectionTypes.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 { contentType, body, params, query } = this.getRequestObject();
|
||||||
|
const webhookResponse: Record<string, any> = { contentType, body };
|
||||||
|
if (Object.keys(params).length) webhookResponse.params = params;
|
||||||
|
if (Object.keys(query).length) webhookResponse.query = query;
|
||||||
|
return { webhookResponse };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
describe('Webhook API', () => {
|
describe('Webhook API', () => {
|
||||||
mockInstance(ExternalHooks);
|
const nodeInstance = new WebhookTestingNode();
|
||||||
mockInstance(Push);
|
const node = mock<INode>({
|
||||||
|
name: 'Webhook',
|
||||||
|
type: nodeInstance.description.name,
|
||||||
|
webhookId: '5ccef736-be16-4d10-b7fb-feed7a61ff22',
|
||||||
|
});
|
||||||
|
const workflowData = { active: true, nodes: [node] } as IWorkflowBase;
|
||||||
|
|
||||||
|
const nodeTypes = mockInstance(NodeTypes);
|
||||||
|
nodeTypes.getByName.mockReturnValue(nodeInstance);
|
||||||
|
nodeTypes.getByNameAndVersion.mockReturnValue(nodeInstance);
|
||||||
|
|
||||||
|
let user: User;
|
||||||
let agent: SuperAgentTest;
|
let agent: SuperAgentTest;
|
||||||
|
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
await testDb.init();
|
await testDb.init();
|
||||||
|
user = await createUser();
|
||||||
|
|
||||||
|
const server = new WebhookServer();
|
||||||
|
await server.start();
|
||||||
|
agent = testAgent(server.app);
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
await testDb.truncate(['Workflow']);
|
||||||
|
await createWorkflow(workflowData, user);
|
||||||
|
await initActiveWorkflowManager();
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(async () => {
|
afterAll(async () => {
|
||||||
@@ -41,32 +102,15 @@ describe('Webhook API', () => {
|
|||||||
|
|
||||||
describe('Content-Type support', () => {
|
describe('Content-Type support', () => {
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
const node = new WebhookTestingNode();
|
node.parameters = { httpMethod: 'POST', path: 'abcd' };
|
||||||
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 WebhookServer();
|
|
||||||
await server.start();
|
|
||||||
agent = testAgent(server.app);
|
|
||||||
});
|
|
||||||
|
|
||||||
afterAll(async () => {
|
|
||||||
await testDb.truncate(['Workflow']);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should handle JSON', async () => {
|
test('should handle JSON', async () => {
|
||||||
const response = await agent.post('/webhook/abcd').send({ test: true });
|
const response = await agent.post('/webhook/abcd').send({ test: true });
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body).toEqual({
|
expect(response.body).toEqual({
|
||||||
type: 'application/json',
|
contentType: 'application/json',
|
||||||
body: { test: true },
|
body: { test: true },
|
||||||
params: {},
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -79,7 +123,7 @@ describe('Webhook API', () => {
|
|||||||
);
|
);
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body).toEqual({
|
expect(response.body).toEqual({
|
||||||
type: 'application/xml',
|
contentType: 'application/xml',
|
||||||
body: {
|
body: {
|
||||||
outer: {
|
outer: {
|
||||||
$: {
|
$: {
|
||||||
@@ -88,7 +132,6 @@ describe('Webhook API', () => {
|
|||||||
inner: 'value',
|
inner: 'value',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
params: {},
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -99,9 +142,8 @@ describe('Webhook API', () => {
|
|||||||
.send('x=5&y=str&z=false');
|
.send('x=5&y=str&z=false');
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body).toEqual({
|
expect(response.body).toEqual({
|
||||||
type: 'application/x-www-form-urlencoded',
|
contentType: 'application/x-www-form-urlencoded',
|
||||||
body: { x: '5', y: 'str', z: 'false' },
|
body: { x: '5', y: 'str', z: 'false' },
|
||||||
params: {},
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -112,9 +154,8 @@ describe('Webhook API', () => {
|
|||||||
.send('{"key": "value"}');
|
.send('{"key": "value"}');
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body).toEqual({
|
expect(response.body).toEqual({
|
||||||
type: 'text/plain',
|
contentType: 'text/plain',
|
||||||
body: '{"key": "value"}',
|
body: '{"key": "value"}',
|
||||||
params: {},
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -130,7 +171,7 @@ describe('Webhook API', () => {
|
|||||||
.set('content-type', 'multipart/form-data');
|
.set('content-type', 'multipart/form-data');
|
||||||
|
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body.type).toEqual('multipart/form-data');
|
expect(response.body.contentType).toEqual('multipart/form-data');
|
||||||
const { data, files } = response.body.body;
|
const { data, files } = response.body.body;
|
||||||
expect(data).toEqual({ field1: 'value1', field2: ['value2', 'value3'] });
|
expect(data).toEqual({ field1: 'value1', field2: ['value2', 'value3'] });
|
||||||
|
|
||||||
@@ -142,25 +183,9 @@ describe('Webhook API', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('Params support', () => {
|
describe('Route-parameters support', () => {
|
||||||
beforeAll(async () => {
|
beforeAll(async () => {
|
||||||
const node = new WebhookTestingNode();
|
node.parameters = { httpMethod: 'PATCH', path: ':variable' };
|
||||||
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 WebhookServer();
|
|
||||||
await server.start();
|
|
||||||
agent = testAgent(server.app);
|
|
||||||
});
|
|
||||||
|
|
||||||
afterAll(async () => {
|
|
||||||
await testDb.truncate(['Workflow']);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should handle params', async () => {
|
test('should handle params', async () => {
|
||||||
@@ -169,7 +194,7 @@ describe('Webhook API', () => {
|
|||||||
.send({ test: true });
|
.send({ test: true });
|
||||||
expect(response.statusCode).toEqual(200);
|
expect(response.statusCode).toEqual(200);
|
||||||
expect(response.body).toEqual({
|
expect(response.body).toEqual({
|
||||||
type: 'application/json',
|
contentType: 'application/json',
|
||||||
body: { test: true },
|
body: { test: true },
|
||||||
params: {
|
params: {
|
||||||
variable: 'test',
|
variable: 'test',
|
||||||
@@ -180,68 +205,21 @@ describe('Webhook API', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
class WebhookTestingNode implements INodeType {
|
describe('Query-parameters support', () => {
|
||||||
description: INodeTypeDescription = {
|
beforeAll(async () => {
|
||||||
displayName: 'Webhook Testing Node',
|
node.parameters = { httpMethod: 'GET', path: 'testing' };
|
||||||
name: 'webhook-testing-node',
|
});
|
||||||
group: ['trigger'],
|
|
||||||
version: 1,
|
|
||||||
description: '',
|
|
||||||
defaults: {},
|
|
||||||
inputs: [],
|
|
||||||
outputs: [NodeConnectionTypes.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) {
|
test('should use the extended query parser', async () => {
|
||||||
const req = this.getRequestObject();
|
const response = await agent.get('/webhook/testing?filter[field]=value');
|
||||||
return {
|
expect(response.statusCode).toEqual(200);
|
||||||
webhookResponse: {
|
expect(response.body).toEqual({
|
||||||
type: req.contentType,
|
query: {
|
||||||
body: req.body,
|
filter: {
|
||||||
params: req.params,
|
field: 'value',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
});
|
||||||
}
|
});
|
||||||
}
|
|
||||||
|
|
||||||
const createWebhookWorkflow = (
|
|
||||||
node: WebhookTestingNode,
|
|
||||||
path = 'abcd',
|
|
||||||
httpMethod = 'POST',
|
|
||||||
): Partial<IWorkflowBase> => ({
|
|
||||||
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],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user