Add DELETE, PATCH and PUT request support to Webhooks

This commit is contained in:
Jan Oberhauser 2022-02-20 10:30:01 +01:00
parent ec5bfaf895
commit 9a06d0fffc
5 changed files with 103 additions and 249 deletions

View file

@ -72,6 +72,7 @@ import {
IWorkflowBase, IWorkflowBase,
LoggerProxy, LoggerProxy,
NodeHelpers, NodeHelpers,
WebhookHttpMethod,
Workflow, Workflow,
WorkflowExecuteMode, WorkflowExecuteMode,
} from 'n8n-workflow'; } from 'n8n-workflow';
@ -202,6 +203,8 @@ class App {
presetCredentialsLoaded: boolean; presetCredentialsLoaded: boolean;
webhookMethods: WebhookHttpMethod[];
constructor() { constructor() {
this.app = express(); this.app = express();
@ -237,6 +240,8 @@ class App {
this.presetCredentialsLoaded = false; this.presetCredentialsLoaded = false;
this.endpointPresetCredentials = config.get('credentials.overwrite.endpoint') as string; this.endpointPresetCredentials = config.get('credentials.overwrite.endpoint') as string;
this.webhookMethods = ['DELETE', 'GET', 'HEAD', 'PATCH', 'POST', 'PUT'];
const urlBaseWebhook = WebhookHelpers.getWebhookBaseUrl(); const urlBaseWebhook = WebhookHelpers.getWebhookBaseUrl();
const telemetrySettings: ITelemetrySettings = { const telemetrySettings: ITelemetrySettings = {
@ -2704,8 +2709,8 @@ class App {
WebhookServer.registerProductionWebhooks.apply(this); WebhookServer.registerProductionWebhooks.apply(this);
} }
// HEAD webhook requests (test for UI) // Register all webhook requests (test for UI)
this.app.head( this.app.all(
`/${this.endpointWebhookTest}/*`, `/${this.endpointWebhookTest}/*`,
async (req: express.Request, res: express.Response) => { async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-test/" to get the registred part of the url // Cut away the "/webhook-test/" to get the registred part of the url
@ -2713,38 +2718,9 @@ class App {
this.endpointWebhookTest.length + 2, this.endpointWebhookTest.length + 2,
); );
let response; const method = req.method.toUpperCase() as WebhookHttpMethod;
try {
response = await this.testWebhooks.callTestWebhook('HEAD', requestUrl, req, res);
} catch (error) {
ResponseHelper.sendErrorResponse(res, error);
return;
}
if (response.noWebhookResponse === true) {
// Nothing else to do as the response got already sent
return;
}
ResponseHelper.sendSuccessResponse(
res,
response.data,
true,
response.responseCode,
response.headers,
);
},
);
// HEAD webhook requests (test for UI)
this.app.options(
`/${this.endpointWebhookTest}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-test/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhookTest.length + 2,
);
if (method === 'OPTIONS') {
let allowedMethods: string[]; let allowedMethods: string[];
try { try {
allowedMethods = await this.testWebhooks.getWebhookMethods(requestUrl); allowedMethods = await this.testWebhooks.getWebhookMethods(requestUrl);
@ -2758,53 +2734,20 @@ class App {
} }
ResponseHelper.sendSuccessResponse(res, {}, true, 204); ResponseHelper.sendSuccessResponse(res, {}, true, 204);
},
);
// GET webhook requests (test for UI)
this.app.get(
`/${this.endpointWebhookTest}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-test/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhookTest.length + 2,
);
let response;
try {
response = await this.testWebhooks.callTestWebhook('GET', requestUrl, req, res);
} catch (error) {
ResponseHelper.sendErrorResponse(res, error);
return; return;
} }
if (response.noWebhookResponse === true) { if (!this.webhookMethods.includes(method)) {
// Nothing else to do as the response got already sent ResponseHelper.sendErrorResponse(
return;
}
ResponseHelper.sendSuccessResponse(
res, res,
response.data, new Error(`The method ${method} is not supported.`),
true,
response.responseCode,
response.headers,
);
},
);
// POST webhook requests (test for UI)
this.app.post(
`/${this.endpointWebhookTest}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-test/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhookTest.length + 2,
); );
return;
}
let response; let response;
try { try {
response = await this.testWebhooks.callTestWebhook('POST', requestUrl, req, res); response = await this.testWebhooks.callTestWebhook(method, requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
return; return;

View file

@ -16,6 +16,7 @@ import * as _ from 'lodash';
import * as compression from 'compression'; import * as compression from 'compression';
// eslint-disable-next-line import/no-extraneous-dependencies // eslint-disable-next-line import/no-extraneous-dependencies
import * as parseUrl from 'parseurl'; import * as parseUrl from 'parseurl';
import { WebhookHttpMethod } from 'n8n-workflow';
// eslint-disable-next-line import/no-cycle // eslint-disable-next-line import/no-cycle
import { import {
ActiveExecutions, ActiveExecutions,
@ -41,8 +42,8 @@ export function registerProductionWebhooks() {
// Regular Webhooks // Regular Webhooks
// ---------------------------------------- // ----------------------------------------
// HEAD webhook requests // Register all webhook requests
this.app.head( this.app.all(
`/${this.endpointWebhook}/*`, `/${this.endpointWebhook}/*`,
async (req: express.Request, res: express.Response) => { async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook/" to get the registred part of the url // Cut away the "/webhook/" to get the registred part of the url
@ -50,39 +51,9 @@ export function registerProductionWebhooks() {
this.endpointWebhook.length + 2, this.endpointWebhook.length + 2,
); );
let response; const method = req.method.toUpperCase() as WebhookHttpMethod;
try {
// eslint-disable-next-line @typescript-eslint/no-unsafe-call
response = await this.activeWorkflowRunner.executeWebhook('HEAD', requestUrl, req, res);
} catch (error) {
ResponseHelper.sendErrorResponse(res, error);
return;
}
if (response.noWebhookResponse === true) {
// Nothing else to do as the response got already sent
return;
}
ResponseHelper.sendSuccessResponse(
res,
response.data,
true,
response.responseCode,
response.headers,
);
},
);
// OPTIONS webhook requests
this.app.options(
`/${this.endpointWebhook}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhook.length + 2,
);
if (method === 'OPTIONS') {
let allowedMethods: string[]; let allowedMethods: string[];
try { try {
allowedMethods = await this.activeWorkflowRunner.getWebhookMethods(requestUrl); allowedMethods = await this.activeWorkflowRunner.getWebhookMethods(requestUrl);
@ -96,53 +67,18 @@ export function registerProductionWebhooks() {
} }
ResponseHelper.sendSuccessResponse(res, {}, true, 204); ResponseHelper.sendSuccessResponse(res, {}, true, 204);
},
);
// GET webhook requests
this.app.get(
`/${this.endpointWebhook}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhook.length + 2,
);
let response;
try {
response = await this.activeWorkflowRunner.executeWebhook('GET', requestUrl, req, res);
} catch (error) {
ResponseHelper.sendErrorResponse(res, error);
return; return;
} }
if (response.noWebhookResponse === true) { if (!this.webhookMethods.includes(method)) {
// Nothing else to do as the response got already sent ResponseHelper.sendErrorResponse(res, new Error(`The method ${method} is not supported.`));
return; return;
} }
ResponseHelper.sendSuccessResponse(
res,
response.data,
true,
response.responseCode,
response.headers,
);
},
);
// POST webhook requests
this.app.post(
`/${this.endpointWebhook}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhook.length + 2,
);
let response; let response;
try { try {
response = await this.activeWorkflowRunner.executeWebhook('POST', requestUrl, req, res); // eslint-disable-next-line @typescript-eslint/no-unsafe-call
response = await this.activeWorkflowRunner.executeWebhook(method, requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
return; return;
@ -169,8 +105,8 @@ export function registerProductionWebhooks() {
const waitingWebhooks = new WaitingWebhooks(); const waitingWebhooks = new WaitingWebhooks();
// HEAD webhook-waiting requests // Register all webhook-waiting requests
this.app.head( this.app.all(
`/${this.endpointWebhookWaiting}/*`, `/${this.endpointWebhookWaiting}/*`,
async (req: express.Request, res: express.Response) => { async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-waiting/" to get the registred part of the url // Cut away the "/webhook-waiting/" to get the registred part of the url
@ -178,73 +114,20 @@ export function registerProductionWebhooks() {
this.endpointWebhookWaiting.length + 2, this.endpointWebhookWaiting.length + 2,
); );
let response; const method = req.method.toUpperCase() as WebhookHttpMethod;
try {
response = await waitingWebhooks.executeWebhook('HEAD', requestUrl, req, res); // TOOD: Add support for OPTIONS in the future
} catch (error) { // if (method === 'OPTIONS') {
ResponseHelper.sendErrorResponse(res, error); // }
if (!this.webhookMethods.includes(method)) {
ResponseHelper.sendErrorResponse(res, new Error(`The method ${method} is not supported.`));
return; return;
} }
if (response.noWebhookResponse === true) {
// Nothing else to do as the response got already sent
return;
}
ResponseHelper.sendSuccessResponse(
res,
response.data,
true,
response.responseCode,
response.headers,
);
},
);
// GET webhook-waiting requests
this.app.get(
`/${this.endpointWebhookWaiting}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-waiting/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhookWaiting.length + 2,
);
let response; let response;
try { try {
response = await waitingWebhooks.executeWebhook('GET', requestUrl, req, res); response = await waitingWebhooks.executeWebhook(method, requestUrl, req, res);
} catch (error) {
ResponseHelper.sendErrorResponse(res, error);
return;
}
if (response.noWebhookResponse === true) {
// Nothing else to do as the response got already sent
return;
}
ResponseHelper.sendSuccessResponse(
res,
response.data,
true,
response.responseCode,
response.headers,
);
},
);
// POST webhook-waiting requests
this.app.post(
`/${this.endpointWebhookWaiting}/*`,
async (req: express.Request, res: express.Response) => {
// Cut away the "/webhook-waiting/" to get the registred part of the url
const requestUrl = (req as ICustomRequest).parsedUrl!.pathname!.slice(
this.endpointWebhookWaiting.length + 2,
);
let response;
try {
response = await waitingWebhooks.executeWebhook('POST', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
return; return;

View file

@ -250,6 +250,10 @@ export class Wait implements INodeType {
}, },
}, },
options: [ options: [
{
name: 'DELETE',
value: 'DELETE',
},
{ {
name: 'GET', name: 'GET',
value: 'GET', value: 'GET',
@ -258,10 +262,18 @@ export class Wait implements INodeType {
name: 'HEAD', name: 'HEAD',
value: 'HEAD', value: 'HEAD',
}, },
{
name: 'PATCH',
value: 'PATCH',
},
{ {
name: 'POST', name: 'POST',
value: 'POST', value: 'POST',
}, },
{
name: 'PUT',
value: 'PUT',
},
], ],
default: 'GET', default: 'GET',
description: 'The HTTP method of the Webhook call', description: 'The HTTP method of the Webhook call',
@ -514,6 +526,8 @@ export class Wait implements INodeType {
displayOptions: { displayOptions: {
show: { show: {
'/httpMethod': [ '/httpMethod': [
'PATCH',
'PUT',
'POST', 'POST',
], ],
}, },

View file

@ -120,6 +120,10 @@ export class Webhook implements INodeType {
name: 'httpMethod', name: 'httpMethod',
type: 'options', type: 'options',
options: [ options: [
{
name: 'DELETE',
value: 'DELETE',
},
{ {
name: 'GET', name: 'GET',
value: 'GET', value: 'GET',
@ -128,10 +132,18 @@ export class Webhook implements INodeType {
name: 'HEAD', name: 'HEAD',
value: 'HEAD', value: 'HEAD',
}, },
{
name: 'PATCH',
value: 'PATCH',
},
{ {
name: 'POST', name: 'POST',
value: 'POST', value: 'POST',
}, },
{
name: 'PUT',
value: 'PUT',
},
], ],
default: 'GET', default: 'GET',
description: 'The HTTP method to listen to.', description: 'The HTTP method to listen to.',
@ -265,6 +277,8 @@ export class Webhook implements INodeType {
displayOptions: { displayOptions: {
show: { show: {
'/httpMethod': [ '/httpMethod': [
'PATCH',
'PUT',
'POST', 'POST',
], ],
}, },

View file

@ -1178,7 +1178,7 @@ export interface IWorkflowMetadata {
active: boolean; active: boolean;
} }
export type WebhookHttpMethod = 'GET' | 'POST' | 'HEAD' | 'OPTIONS'; export type WebhookHttpMethod = 'DELETE' | 'GET' | 'HEAD' | 'PATCH' | 'POST' | 'PUT' | 'OPTIONS';
export interface IWebhookResponseData { export interface IWebhookResponseData {
workflowData?: INodeExecutionData[][]; workflowData?: INodeExecutionData[][];