production dynamic webhooks complete

This commit is contained in:
Ben Hesseldieck 2021-01-15 12:27:25 +01:00
parent 3fc0a21133
commit 56336b622b
6 changed files with 40 additions and 19 deletions

View file

@ -20,6 +20,7 @@ import {
} from 'n8n-core'; } from 'n8n-core';
import { import {
IDataObject,
IExecuteData, IExecuteData,
IGetExecutePollFunctions, IGetExecutePollFunctions,
IGetExecuteTriggerFunctions, IGetExecuteTriggerFunctions,
@ -117,23 +118,28 @@ export class ActiveWorkflowRunner {
} }
let webhook = await Db.collections.Webhook?.findOne({ webhookPath: path, method: httpMethod }) as IWebhookDb; let webhook = await Db.collections.Webhook?.findOne({ webhookPath: path, method: httpMethod }) as IWebhookDb;
let webhookId: string; let webhookId: string | undefined;
if (webhook === undefined) { if (webhook === undefined) {
// check if a dynamic webhook path exists
const pathElements = path.split('/'); const pathElements = path.split('/');
webhookId = pathElements[0]; webhookId = pathElements.shift();
webhook = await Db.collections.Webhook?.findOne({ webhookId, method: httpMethod }) as IWebhookDb; webhook = await Db.collections.Webhook?.findOne({ webhookId, method: httpMethod }) as IWebhookDb;
// write params to req.params if (webhook) {
} path = webhook.webhookPath;
// extracting params from path
// check if something exist const webhookPathParams: IDataObject = {};
if (webhook === undefined) { webhook.webhookPath.split('/').forEach((ele, index) => {
// The requested webhook is not registered if (ele.startsWith(':')) {
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404); webhookPathParams[ele.slice(1)] = pathElements[index];
} }
});
if (webhookId) { // write params to req.params
path = webhook.webhookPath; Object.assign(req.params, webhookPathParams);
} else {
// The requested webhook is not registered
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404);
}
} }
const workflowData = await Db.collections.Workflow!.findOne(webhook.workflowId); const workflowData = await Db.collections.Workflow!.findOne(webhook.workflowId);
@ -265,10 +271,14 @@ export class ActiveWorkflowRunner {
method: webhookData.httpMethod, method: webhookData.httpMethod,
} as IWebhookDb; } as IWebhookDb;
if (webhook.webhookPath.includes('/:') && node.webhookId) { if ((path.startsWith(':') || path.includes('/:')) && node.webhookId) {
webhook.webhookId = node.webhookId; webhook.webhookId = node.webhookId;
} }
if (webhook.webhookPath.charAt(0) === '/') {
webhook.webhookPath = webhook.webhookPath.slice(1);
}
try { try {
await Db.collections.Webhook?.insert(webhook); await Db.collections.Webhook?.insert(webhook);

View file

@ -993,6 +993,12 @@ export function getExecuteWebhookFunctions(workflow: Workflow, node: INode, addi
return getNodeParameter(workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, fallbackValue); return getNodeParameter(workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, fallbackValue);
}, },
getParamsData(): object {
if (additionalData.httpRequest === undefined) {
throw new Error('Request is missing!');
}
return additionalData.httpRequest.params;
},
getQueryData(): object { getQueryData(): object {
if (additionalData.httpRequest === undefined) { if (additionalData.httpRequest === undefined) {
throw new Error('Request is missing!'); throw new Error('Request is missing!');

View file

@ -54,9 +54,10 @@ export class ClassNameReplace implements INodeType {
const returnData: IDataObject[] = []; const returnData: IDataObject[] = [];
returnData.push( returnData.push(
{ {
body: this.getBodyData(),
headers: this.getHeaderData(), headers: this.getHeaderData(),
params: this.getParamsData(),
query: this.getQueryData(), query: this.getQueryData(),
body: this.getBodyData(),
} }
); );

View file

@ -412,9 +412,10 @@ export class Webhook implements INodeType {
const returnItem: INodeExecutionData = { const returnItem: INodeExecutionData = {
binary: {}, binary: {},
json: { json: {
body: data,
headers, headers,
params: this.getParamsData(),
query: this.getQueryData(), query: this.getQueryData(),
body: data,
}, },
}; };
@ -458,9 +459,10 @@ export class Webhook implements INodeType {
const returnItem: INodeExecutionData = { const returnItem: INodeExecutionData = {
binary: {}, binary: {},
json: { json: {
body: this.getBodyData(),
headers, headers,
params: this.getParamsData(),
query: this.getQueryData(), query: this.getQueryData(),
body: this.getBodyData(),
}, },
}; };
@ -483,9 +485,10 @@ export class Webhook implements INodeType {
const response: INodeExecutionData = { const response: INodeExecutionData = {
json: { json: {
body: this.getBodyData(),
headers, headers,
params: this.getParamsData(),
query: this.getQueryData(), query: this.getQueryData(),
body: this.getBodyData(),
}, },
}; };

View file

@ -311,6 +311,7 @@ export interface IWebhookFunctions {
getNode(): INode; getNode(): INode;
getNodeParameter(parameterName: string, fallbackValue?: any): NodeParameterValue | INodeParameters | NodeParameterValue[] | INodeParameters[] | object; //tslint:disable-line:no-any getNodeParameter(parameterName: string, fallbackValue?: any): NodeParameterValue | INodeParameters | NodeParameterValue[] | INodeParameters[] | object; //tslint:disable-line:no-any
getNodeWebhookUrl: (name: string) => string | undefined; getNodeWebhookUrl: (name: string) => string | undefined;
getParamsData(): object;
getQueryData(): object; getQueryData(): object;
getRequestObject(): express.Request; getRequestObject(): express.Request;
getResponseObject(): express.Response; getResponseObject(): express.Response;

View file

@ -883,7 +883,7 @@ export function getNodeWebhookPath(workflowId: string, node: INode, path: string
* @returns {string} * @returns {string}
*/ */
export function getNodeWebhookUrl(baseUrl: string, workflowId: string, node: INode, path: string, isFullPath?: boolean): string { export function getNodeWebhookUrl(baseUrl: string, workflowId: string, node: INode, path: string, isFullPath?: boolean): string {
if (path.includes('/:') && node.webhookId) { if ((path.startsWith(':') || path.includes('/:')) && node.webhookId) {
// setting this to false to prefix the webhookId // setting this to false to prefix the webhookId
isFullPath = false; isFullPath = false;
} }