mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
⚡ production dynamic webhooks complete
This commit is contained in:
parent
3fc0a21133
commit
56336b622b
|
@ -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);
|
||||||
|
|
|
@ -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!');
|
||||||
|
|
|
@ -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(),
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -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(),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue