Add support for webhook route parameters (#1343)

* 🚧 add webhookId to URL

* 🚧 add webhookId to webhook entity, 🔧 refactor migrations

* 🚧 🐘 postgres migration

* 🚧 add mySQL migration

* 🚧 refactor mongoDB

* 🚧 add webhookId to IWebhookDb

* 🚧 starting workflow with dynamic route works

*  production dynamic webhooks complete

* 🎨 fix lint issues

* 🔧 dynamic path for webhook-test complete

* 🎨 fix lint issues

* 🎨 fix typescript issue

*  add error message for dynamic webhook-test

* 🔨 improve handling of leading `/`

* 🚧 add webhookId to URL

* 🚧 add webhookId to webhook entity, 🔧 refactor migrations

* 🚧 🐘 postgres migration

* 🚧 add mySQL migration

* 🚧 refactor mongoDB

* 🚧 add webhookId to IWebhookDb

* 🚧 starting workflow with dynamic route works

*  production dynamic webhooks complete

* 🎨 fix lint issues

* 🔧 dynamic path for webhook-test complete

* 🎨 fix lint issues

* 🎨 fix typescript issue

*  add error message for dynamic webhook-test

* 🔨 improve handling of leading `/`

*  Fix issue that tab-title did not get reset on new workflow

* Revert " Fix issue that tab-title did not get reset on new workflow"

This reverts commit 699d0a8946.

* 🔧 reset params before extraction

* 🐘 removing unique constraint for webhookId

* 🚧 handle multiple webhooks per id

* 🔧 enable webhook-test for multiple WH with same id

* 🐘 add migration for postgres

*  add mysql migration

* 🎨 fix lint issue

Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
Ben Hesseldieck 2021-01-23 20:00:32 +01:00 committed by GitHub
parent 1a68303319
commit d395498882
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
26 changed files with 327 additions and 88 deletions

View file

@ -110,7 +110,7 @@ By default n8n uses SQLite to save credentials, past executions and workflows.
n8n however also supports MongoDB, PostgresDB and MySQL. To use them simply a few n8n however also supports MongoDB, PostgresDB and MySQL. To use them simply a few
environment variables have to be set. environment variables have to be set.
It is important to still persist the data in the `/root/.n8` folder. The reason It is important to still persist the data in the `/root/.n8n` folder. The reason
is that it contains n8n user data. That is the name of the webhook is that it contains n8n user data. That is the name of the webhook
(in case) the n8n tunnel gets used and even more important the encryption key (in case) the n8n tunnel gets used and even more important the encryption key
for the credentials. If none gets found n8n creates automatically one on for the credentials. If none gets found n8n creates automatically one on

View file

@ -114,7 +114,7 @@
"sqlite3": "^4.2.0", "sqlite3": "^4.2.0",
"sse-channel": "^3.1.1", "sse-channel": "^3.1.1",
"tslib": "1.11.2", "tslib": "1.11.2",
"typeorm": "^0.2.24" "typeorm": "^0.2.30"
}, },
"jest": { "jest": {
"transform": { "transform": {

View file

@ -116,12 +116,49 @@ export class ActiveWorkflowRunner {
throw new ResponseHelper.ResponseError('The "activeWorkflows" instance did not get initialized yet.', 404, 404); throw new ResponseHelper.ResponseError('The "activeWorkflows" instance did not get initialized yet.', 404, 404);
} }
const 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 | undefined;
// check if something exist // check if path is dynamic
if (webhook === undefined) { if (webhook === undefined) {
// The requested webhook is not registered // check if a dynamic webhook path exists
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404); const pathElements = path.split('/');
webhookId = pathElements.shift();
const dynamicWebhooks = await Db.collections.Webhook?.find({ webhookId, method: httpMethod, pathLength: pathElements.length });
if (dynamicWebhooks === undefined) {
// The requested webhook is not registered
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404);
}
// set webhook to the first webhook result
// if more results have been returned choose the one with the most route-matches
webhook = dynamicWebhooks[0];
if (dynamicWebhooks.length > 1) {
let maxMatches = 0;
const pathElementsSet = new Set(pathElements);
dynamicWebhooks.forEach(dynamicWebhook => {
const intersection =
dynamicWebhook.webhookPath
.split('/')
.reduce((acc, element) => pathElementsSet.has(element) ? acc += 1 : acc, 0);
if (intersection > maxMatches) {
maxMatches = intersection;
webhook = dynamicWebhook;
}
});
if (maxMatches === 0) {
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404);
}
}
path = webhook.webhookPath;
// extracting params from path
webhook.webhookPath.split('/').forEach((ele, index) => {
if (ele.startsWith(':')) {
// write params to req.params
req.params[ele.slice(1)] = pathElements[index];
}
});
} }
const workflowData = await Db.collections.Workflow!.findOne(webhook.workflowId); const workflowData = await Db.collections.Workflow!.findOne(webhook.workflowId);
@ -253,6 +290,15 @@ export class ActiveWorkflowRunner {
method: webhookData.httpMethod, method: webhookData.httpMethod,
} as IWebhookDb; } as IWebhookDb;
if (webhook.webhookPath.startsWith('/')) {
webhook.webhookPath = webhook.webhookPath.slice(1);
}
if ((path.startsWith(':') || path.includes('/:')) && node.webhookId) {
webhook.webhookId = node.webhookId;
webhook.pathLength = webhook.webhookPath.split('/').length;
}
try { try {
await Db.collections.Webhook?.insert(webhook); await Db.collections.Webhook?.insert(webhook);
@ -273,10 +319,9 @@ export class ActiveWorkflowRunner {
let errorMessage = ''; let errorMessage = '';
// if it's a workflow from the the insert // if it's a workflow from the the insert
// TODO check if there is standard error code for deplicate key violation that works // TODO check if there is standard error code for duplicate key violation that works
// with all databases // with all databases
if (error.name === 'MongoError' || error.name === 'QueryFailedError') { if (error.name === 'MongoError' || error.name === 'QueryFailedError') {
errorMessage = `The webhook path [${webhook.webhookPath}] and method [${webhook.method}] already exist.`; errorMessage = `The webhook path [${webhook.webhookPath}] and method [${webhook.method}] already exist.`;
} else if (error.detail) { } else if (error.detail) {

View file

@ -32,29 +32,10 @@ export let collections: IDatabaseCollections = {
Webhook: null, Webhook: null,
}; };
import { import { postgresMigrations } from './databases/postgresdb/migrations';
CreateIndexStoppedAt1594828256133, import { mongodbMigrations } from './databases/mongodb/migrations';
InitialMigration1587669153312, import { mysqlMigrations } from './databases/mysqldb/migrations';
WebhookModel1589476000887, import { sqliteMigrations } from './databases/sqlite/migrations';
} from './databases/postgresdb/migrations';
import {
CreateIndexStoppedAt1594910478695,
InitialMigration1587563438936,
WebhookModel1592679094242,
} from './databases/mongodb/migrations';
import {
CreateIndexStoppedAt1594902918301,
InitialMigration1588157391238,
WebhookModel1592447867632,
} from './databases/mysqldb/migrations';
import {
CreateIndexStoppedAt1594825041918,
InitialMigration1588102412422,
WebhookModel1592445003908,
} from './databases/sqlite/migrations';
import * as path from 'path'; import * as path from 'path';
@ -75,11 +56,7 @@ export async function init(): Promise<IDatabaseCollections> {
entityPrefix, entityPrefix,
url: await GenericHelpers.getConfigValue('database.mongodb.connectionUrl') as string, url: await GenericHelpers.getConfigValue('database.mongodb.connectionUrl') as string,
useNewUrlParser: true, useNewUrlParser: true,
migrations: [ migrations: mongodbMigrations,
InitialMigration1587563438936,
WebhookModel1592679094242,
CreateIndexStoppedAt1594910478695,
],
migrationsRun: true, migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`, migrationsTableName: `${entityPrefix}migrations`,
}; };
@ -112,11 +89,7 @@ export async function init(): Promise<IDatabaseCollections> {
port: await GenericHelpers.getConfigValue('database.postgresdb.port') as number, port: await GenericHelpers.getConfigValue('database.postgresdb.port') as number,
username: await GenericHelpers.getConfigValue('database.postgresdb.user') as string, username: await GenericHelpers.getConfigValue('database.postgresdb.user') as string,
schema: config.get('database.postgresdb.schema'), schema: config.get('database.postgresdb.schema'),
migrations: [ migrations: postgresMigrations,
InitialMigration1587669153312,
WebhookModel1589476000887,
CreateIndexStoppedAt1594828256133,
],
migrationsRun: true, migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`, migrationsTableName: `${entityPrefix}migrations`,
ssl, ssl,
@ -135,11 +108,7 @@ export async function init(): Promise<IDatabaseCollections> {
password: await GenericHelpers.getConfigValue('database.mysqldb.password') as string, password: await GenericHelpers.getConfigValue('database.mysqldb.password') as string,
port: await GenericHelpers.getConfigValue('database.mysqldb.port') as number, port: await GenericHelpers.getConfigValue('database.mysqldb.port') as number,
username: await GenericHelpers.getConfigValue('database.mysqldb.user') as string, username: await GenericHelpers.getConfigValue('database.mysqldb.user') as string,
migrations: [ migrations: mysqlMigrations,
InitialMigration1588157391238,
WebhookModel1592447867632,
CreateIndexStoppedAt1594902918301,
],
migrationsRun: true, migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`, migrationsTableName: `${entityPrefix}migrations`,
}; };
@ -151,11 +120,7 @@ export async function init(): Promise<IDatabaseCollections> {
type: 'sqlite', type: 'sqlite',
database: path.join(n8nFolder, 'database.sqlite'), database: path.join(n8nFolder, 'database.sqlite'),
entityPrefix, entityPrefix,
migrations: [ migrations: sqliteMigrations,
InitialMigration1588102412422,
WebhookModel1592445003908,
CreateIndexStoppedAt1594825041918,
],
migrationsRun: true, migrationsRun: true,
migrationsTableName: `${entityPrefix}migrations`, migrationsTableName: `${entityPrefix}migrations`,
}; };

View file

@ -57,6 +57,8 @@ export interface IWebhookDb {
webhookPath: string; webhookPath: string;
method: string; method: string;
node: string; node: string;
webhookId?: string;
pathLength?: number;
} }
export interface IWorkflowBase extends IWorkflowBaseWorkflow { export interface IWorkflowBase extends IWorkflowBaseWorkflow {

View file

@ -1693,6 +1693,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.activeWorkflowRunner.executeWebhook('HEAD', requestUrl, req, res); response = await this.activeWorkflowRunner.executeWebhook('HEAD', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
@ -1734,6 +1735,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.activeWorkflowRunner.executeWebhook('GET', requestUrl, req, res); response = await this.activeWorkflowRunner.executeWebhook('GET', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
@ -1755,6 +1757,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.activeWorkflowRunner.executeWebhook('POST', requestUrl, req, res); response = await this.activeWorkflowRunner.executeWebhook('POST', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
@ -1776,6 +1779,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.testWebhooks.callTestWebhook('HEAD', requestUrl, req, res); response = await this.testWebhooks.callTestWebhook('HEAD', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
@ -1817,6 +1821,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.testWebhooks.callTestWebhook('GET', requestUrl, req, res); response = await this.testWebhooks.callTestWebhook('GET', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);
@ -1838,6 +1843,7 @@ class App {
let response; let response;
try { try {
delete req.params[0];
response = await this.testWebhooks.callTestWebhook('POST', requestUrl, req, res); response = await this.testWebhooks.callTestWebhook('POST', requestUrl, req, res);
} catch (error) { } catch (error) {
ResponseHelper.sendErrorResponse(res, error); ResponseHelper.sendErrorResponse(res, error);

View file

@ -54,14 +54,28 @@ export class TestWebhooks {
* @memberof TestWebhooks * @memberof TestWebhooks
*/ */
async callTestWebhook(httpMethod: WebhookHttpMethod, path: string, request: express.Request, response: express.Response): Promise<IResponseCallbackData> { async callTestWebhook(httpMethod: WebhookHttpMethod, path: string, request: express.Request, response: express.Response): Promise<IResponseCallbackData> {
const webhookData: IWebhookData | undefined = this.activeWebhooks!.get(httpMethod, path); let webhookData: IWebhookData | undefined = this.activeWebhooks!.get(httpMethod, path);
// check if path is dynamic
if (webhookData === undefined) { if (webhookData === undefined) {
// The requested webhook is not registered const pathElements = path.split('/');
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404); const webhookId = pathElements.shift();
webhookData = this.activeWebhooks!.get(httpMethod, pathElements.join('/'), webhookId);
if (webhookData === undefined) {
// The requested webhook is not registered
throw new ResponseHelper.ResponseError(`The requested webhook "${httpMethod} ${path}" is not registered.`, 404, 404);
}
path = webhookData.path;
// extracting params from path
path.split('/').forEach((ele, index) => {
if (ele.startsWith(':')) {
// write params to req.params
request.params[ele.slice(1)] = pathElements[index];
}
});
} }
const webhookKey = this.activeWebhooks!.getWebhookKey(webhookData.httpMethod, webhookData.path); const webhookKey = this.activeWebhooks!.getWebhookKey(webhookData.httpMethod, webhookData.path, webhookData.webhookId) + `|${webhookData.workflowId}`;
// TODO: Clean that duplication up one day and improve code generally // TODO: Clean that duplication up one day and improve code generally
if (this.testWebhookData[webhookKey] === undefined) { if (this.testWebhookData[webhookKey] === undefined) {
@ -81,7 +95,7 @@ export class TestWebhooks {
return new Promise(async (resolve, reject) => { return new Promise(async (resolve, reject) => {
try { try {
const executionMode = 'manual'; const executionMode = 'manual';
const executionId = await WebhookHelpers.executeWebhook(workflow, webhookData, this.testWebhookData[webhookKey].workflowData, workflowStartNode, executionMode, this.testWebhookData[webhookKey].sessionId, request, response, (error: Error | null, data: IResponseCallbackData) => { const executionId = await WebhookHelpers.executeWebhook(workflow, webhookData!, this.testWebhookData[webhookKey].workflowData, workflowStartNode, executionMode, this.testWebhookData[webhookKey].sessionId, request, response, (error: Error | null, data: IResponseCallbackData) => {
if (error !== null) { if (error !== null) {
return reject(error); return reject(error);
} }
@ -98,7 +112,7 @@ export class TestWebhooks {
// Inform editor-ui that webhook got received // Inform editor-ui that webhook got received
if (this.testWebhookData[webhookKey].sessionId !== undefined) { if (this.testWebhookData[webhookKey].sessionId !== undefined) {
const pushInstance = Push.getInstance(); const pushInstance = Push.getInstance();
pushInstance.send('testWebhookReceived', { workflowId: webhookData.workflowId, executionId }, this.testWebhookData[webhookKey].sessionId!); pushInstance.send('testWebhookReceived', { workflowId: webhookData!.workflowId, executionId }, this.testWebhookData[webhookKey].sessionId!);
} }
} catch (error) { } catch (error) {
@ -158,7 +172,7 @@ export class TestWebhooks {
let key: string; let key: string;
const activatedKey: string[] = []; const activatedKey: string[] = [];
for (const webhookData of webhooks) { for (const webhookData of webhooks) {
key = this.activeWebhooks!.getWebhookKey(webhookData.httpMethod, webhookData.path); key = this.activeWebhooks!.getWebhookKey(webhookData.httpMethod, webhookData.path, webhookData.webhookId) + `|${workflowData.id}`;
activatedKey.push(key); activatedKey.push(key);

View file

@ -11,6 +11,8 @@ import {
} from '../../Interfaces'; } from '../../Interfaces';
@Entity() @Entity()
@Index(["webhookPath", "method"], { unique: true })
@Index(["webhookId", "method"], { unique: true })
export class WebhookEntity implements IWebhookDb { export class WebhookEntity implements IWebhookDb {
@ObjectIdColumn() @ObjectIdColumn()
@ -27,4 +29,10 @@ export class WebhookEntity implements IWebhookDb {
@Column() @Column()
node: string; node: string;
@Column()
webhookId: string;
@Column({ nullable: true })
pathLength: number;
} }

View file

@ -1,3 +1,9 @@
export * from './1587563438936-InitialMigration'; import { InitialMigration1587563438936 } from './1587563438936-InitialMigration';
export * from './1592679094242-WebhookModel'; import { WebhookModel1592679094242 } from './1592679094242-WebhookModel';
export * from './151594910478695-CreateIndexStoppedAt'; import { CreateIndexStoppedAt1594910478695 } from './151594910478695-CreateIndexStoppedAt';
export const mongodbMigrations = [
InitialMigration1587563438936,
WebhookModel1592679094242,
CreateIndexStoppedAt1594910478695,
];

View file

@ -1,6 +1,7 @@
import { import {
Column, Column,
Entity, Entity,
Index,
PrimaryColumn, PrimaryColumn,
} from 'typeorm'; } from 'typeorm';
@ -9,6 +10,7 @@ import {
} from '../../Interfaces'; } from '../../Interfaces';
@Entity() @Entity()
@Index(['webhookId', 'method', 'pathLength'])
export class WebhookEntity implements IWebhookDb { export class WebhookEntity implements IWebhookDb {
@Column() @Column()
@ -22,4 +24,10 @@ export class WebhookEntity implements IWebhookDb {
@Column() @Column()
node: string; node: string;
@Column({ nullable: true })
webhookId: string;
@Column({ nullable: true })
pathLength: number;
} }

View file

@ -0,0 +1,24 @@
import {MigrationInterface, QueryRunner} from "typeorm";
import * as config from '../../../../config';
export class AddWebhookId1611149998770 implements MigrationInterface {
name = 'AddWebhookId1611149998770';
async up(queryRunner: QueryRunner): Promise<void> {
const tablePrefix = config.get('database.tablePrefix');
await queryRunner.query('ALTER TABLE `' + tablePrefix + 'webhook_entity` ADD `webhookId` varchar(255) NULL');
await queryRunner.query('ALTER TABLE `' + tablePrefix + 'webhook_entity` ADD `pathLength` int NULL');
await queryRunner.query('CREATE INDEX `IDX_' + tablePrefix + '742496f199721a057051acf4c2` ON `' + tablePrefix + 'webhook_entity` (`webhookId`, `method`, `pathLength`)');
}
async down(queryRunner: QueryRunner): Promise<void> {
const tablePrefix = config.get('database.tablePrefix');
await queryRunner.query(
'DROP INDEX `IDX_' + tablePrefix + '742496f199721a057051acf4c2` ON `' + tablePrefix + 'webhook_entity`'
);
await queryRunner.query('ALTER TABLE `' + tablePrefix + 'webhook_entity` DROP COLUMN `pathLength`');
await queryRunner.query('ALTER TABLE `' + tablePrefix + 'webhook_entity` DROP COLUMN `webhookId`');
}
}

View file

@ -1,3 +1,11 @@
export * from './1588157391238-InitialMigration'; import { InitialMigration1588157391238 } from './1588157391238-InitialMigration';
export * from './1592447867632-WebhookModel'; import { WebhookModel1592447867632 } from './1592447867632-WebhookModel';
export * from './1594902918301-CreateIndexStoppedAt'; import { CreateIndexStoppedAt1594902918301 } from './1594902918301-CreateIndexStoppedAt';
import { AddWebhookId1611149998770 } from './1611149998770-AddWebhookId';
export const mysqlMigrations = [
InitialMigration1588157391238,
WebhookModel1592447867632,
CreateIndexStoppedAt1594902918301,
AddWebhookId1611149998770,
];

View file

@ -1,6 +1,7 @@
import { import {
Column, Column,
Entity, Entity,
Index,
PrimaryColumn, PrimaryColumn,
} from 'typeorm'; } from 'typeorm';
@ -9,6 +10,7 @@ import {
} from '../../'; } from '../../';
@Entity() @Entity()
@Index(['webhookId', 'method', 'pathLength'])
export class WebhookEntity implements IWebhookDb { export class WebhookEntity implements IWebhookDb {
@Column() @Column()
@ -22,4 +24,10 @@ export class WebhookEntity implements IWebhookDb {
@Column() @Column()
node: string; node: string;
@Column({ nullable: true })
webhookId: string;
@Column({ nullable: true })
pathLength: number;
} }

View file

@ -16,7 +16,7 @@ export class WebhookModel1589476000887 implements MigrationInterface {
tablePrefix = schema + '.' + tablePrefix; tablePrefix = schema + '.' + tablePrefix;
} }
await queryRunner.query(`CREATE TABLE ${tablePrefix}webhook_entity ("workflowId" integer NOT NULL, "webhookPath" character varying NOT NULL, "method" character varying NOT NULL, "node" character varying NOT NULL, CONSTRAINT "PK_${tablePrefixIndex}b21ace2e13596ccd87dc9bf4ea6" PRIMARY KEY ("webhookPath", "method"))`, undefined); await queryRunner.query(`CREATE TABLE IF NOT EXISTS ${tablePrefix}webhook_entity ("workflowId" integer NOT NULL, "webhookPath" character varying NOT NULL, "method" character varying NOT NULL, "node" character varying NOT NULL, CONSTRAINT "PK_${tablePrefixIndex}b21ace2e13596ccd87dc9bf4ea6" PRIMARY KEY ("webhookPath", "method"))`, undefined);
} }
async down(queryRunner: QueryRunner): Promise<void> { async down(queryRunner: QueryRunner): Promise<void> {

View file

@ -0,0 +1,33 @@
import {MigrationInterface, QueryRunner} from "typeorm";
import * as config from '../../../../config';
export class AddWebhookId1611144599516 implements MigrationInterface {
name = 'AddWebhookId1611144599516';
async up(queryRunner: QueryRunner): Promise<void> {
let tablePrefix = config.get('database.tablePrefix');
const tablePrefixPure = tablePrefix;
const schema = config.get('database.postgresdb.schema');
if (schema) {
tablePrefix = schema + '.' + tablePrefix;
}
await queryRunner.query(`ALTER TABLE ${tablePrefix}webhook_entity ADD "webhookId" character varying`);
await queryRunner.query(`ALTER TABLE ${tablePrefix}webhook_entity ADD "pathLength" integer`);
await queryRunner.query(`CREATE INDEX IF NOT EXISTS IDX_${tablePrefixPure}16f4436789e804e3e1c9eeb240 ON ${tablePrefix}webhook_entity ("webhookId", "method", "pathLength") `);
}
async down(queryRunner: QueryRunner): Promise<void> {
let tablePrefix = config.get('database.tablePrefix');
const tablePrefixPure = tablePrefix;
const schema = config.get('database.postgresdb.schema');
if (schema) {
tablePrefix = schema + '.' + tablePrefix;
}
await queryRunner.query(`DROP INDEX IDX_${tablePrefixPure}16f4436789e804e3e1c9eeb240`);
await queryRunner.query(`ALTER TABLE ${tablePrefix}webhook_entity DROP COLUMN "pathLength"`);
await queryRunner.query(`ALTER TABLE ${tablePrefix}webhook_entity DROP COLUMN "webhookId"`);
}
}

View file

@ -1,4 +1,11 @@
export * from './1587669153312-InitialMigration'; import { InitialMigration1587669153312 } from './1587669153312-InitialMigration';
export * from './1589476000887-WebhookModel'; import { WebhookModel1589476000887 } from './1589476000887-WebhookModel';
export * from './1594828256133-CreateIndexStoppedAt'; import { CreateIndexStoppedAt1594828256133 } from './1594828256133-CreateIndexStoppedAt';
import { AddWebhookId1611144599516 } from './1611144599516-AddWebhookId';
export const postgresMigrations = [
InitialMigration1587669153312,
WebhookModel1589476000887,
CreateIndexStoppedAt1594828256133,
AddWebhookId1611144599516,
];

View file

@ -1,6 +1,7 @@
import { import {
Column, Column,
Entity, Entity,
Index,
PrimaryColumn, PrimaryColumn,
} from 'typeorm'; } from 'typeorm';
@ -9,6 +10,7 @@ import {
} from '../../Interfaces'; } from '../../Interfaces';
@Entity() @Entity()
@Index(['webhookId', 'method', 'pathLength'])
export class WebhookEntity implements IWebhookDb { export class WebhookEntity implements IWebhookDb {
@Column() @Column()
@ -22,4 +24,10 @@ export class WebhookEntity implements IWebhookDb {
@Column() @Column()
node: string; node: string;
@Column({ nullable: true })
webhookId: string;
@Column({ nullable: true })
pathLength: number;
} }

View file

@ -8,7 +8,7 @@ export class CreateIndexStoppedAt1594825041918 implements MigrationInterface {
async up(queryRunner: QueryRunner): Promise<void> { async up(queryRunner: QueryRunner): Promise<void> {
const tablePrefix = config.get('database.tablePrefix'); const tablePrefix = config.get('database.tablePrefix');
await queryRunner.query(`CREATE INDEX "IDX_${tablePrefix}cefb067df2402f6aed0638a6c1" ON "execution_entity" ("stoppedAt") `); await queryRunner.query(`CREATE INDEX "IDX_${tablePrefix}cefb067df2402f6aed0638a6c1" ON "${tablePrefix}execution_entity" ("stoppedAt") `);
} }
async down(queryRunner: QueryRunner): Promise<void> { async down(queryRunner: QueryRunner): Promise<void> {

View file

@ -0,0 +1,26 @@
import {MigrationInterface, QueryRunner} from "typeorm";
import * as config from '../../../../config';
export class AddWebhookId1611071044839 implements MigrationInterface {
name = 'AddWebhookId1611071044839';
async up(queryRunner: QueryRunner): Promise<void> {
const tablePrefix = config.get('database.tablePrefix');
await queryRunner.query(`CREATE TABLE "temporary_webhook_entity" ("workflowId" integer NOT NULL, "webhookPath" varchar NOT NULL, "method" varchar NOT NULL, "node" varchar NOT NULL, "webhookId" varchar, "pathLength" integer, PRIMARY KEY ("webhookPath", "method"))`);
await queryRunner.query(`INSERT INTO "temporary_webhook_entity"("workflowId", "webhookPath", "method", "node") SELECT "workflowId", "webhookPath", "method", "node" FROM "${tablePrefix}webhook_entity"`);
await queryRunner.query(`DROP TABLE "${tablePrefix}webhook_entity"`);
await queryRunner.query(`ALTER TABLE "temporary_webhook_entity" RENAME TO "${tablePrefix}webhook_entity"`);
await queryRunner.query(`CREATE INDEX "IDX_${tablePrefix}742496f199721a057051acf4c2" ON "${tablePrefix}webhook_entity" ("webhookId", "method", "pathLength") `);
}
async down(queryRunner: QueryRunner): Promise<void> {
const tablePrefix = config.get('database.tablePrefix');
await queryRunner.query(`DROP INDEX "IDX_${tablePrefix}742496f199721a057051acf4c2"`);
await queryRunner.query(`ALTER TABLE "${tablePrefix}webhook_entity" RENAME TO "temporary_webhook_entity"`);
await queryRunner.query(`CREATE TABLE "${tablePrefix}webhook_entity" ("workflowId" integer NOT NULL, "webhookPath" varchar NOT NULL, "method" varchar NOT NULL, "node" varchar NOT NULL, PRIMARY KEY ("webhookPath", "method"))`);
await queryRunner.query(`INSERT INTO "${tablePrefix}webhook_entity"("workflowId", "webhookPath", "method", "node") SELECT "workflowId", "webhookPath", "method", "node" FROM "temporary_webhook_entity"`);
await queryRunner.query(`DROP TABLE "temporary_webhook_entity"`);
}
}

View file

@ -1,3 +1,11 @@
export * from './1588102412422-InitialMigration'; import { InitialMigration1588102412422 } from './1588102412422-InitialMigration';
export * from './1592445003908-WebhookModel'; import { WebhookModel1592445003908 } from './1592445003908-WebhookModel';
export * from './1594825041918-CreateIndexStoppedAt'; import { CreateIndexStoppedAt1594825041918 } from './1594825041918-CreateIndexStoppedAt';
import { AddWebhookId1611071044839 } from './1611071044839-AddWebhookId';
export const sqliteMigrations = [
InitialMigration1588102412422,
WebhookModel1592445003908,
CreateIndexStoppedAt1594825041918,
AddWebhookId1611071044839,
];

View file

@ -16,7 +16,7 @@ export class ActiveWebhooks {
} = {}; } = {};
private webhookUrls: { private webhookUrls: {
[key: string]: IWebhookData; [key: string]: IWebhookData[];
} = {}; } = {};
testWebhooks = false; testWebhooks = false;
@ -35,10 +35,10 @@ export class ActiveWebhooks {
throw new Error('Webhooks can only be added for saved workflows as an id is needed!'); throw new Error('Webhooks can only be added for saved workflows as an id is needed!');
} }
const webhookKey = this.getWebhookKey(webhookData.httpMethod, webhookData.path); const webhookKey = this.getWebhookKey(webhookData.httpMethod, webhookData.path, webhookData.webhookId);
//check that there is not a webhook already registed with that path/method //check that there is not a webhook already registed with that path/method
if (this.webhookUrls[webhookKey] !== undefined) { if (this.webhookUrls[webhookKey] && !webhookData.webhookId) {
throw new Error(`Test-Webhook can not be activated because another one with the same method "${webhookData.httpMethod}" and path "${webhookData.path}" is already active!`); throw new Error(`Test-Webhook can not be activated because another one with the same method "${webhookData.httpMethod}" and path "${webhookData.path}" is already active!`);
} }
@ -48,7 +48,10 @@ export class ActiveWebhooks {
// Make the webhook available directly because sometimes to create it successfully // Make the webhook available directly because sometimes to create it successfully
// it gets called // it gets called
this.webhookUrls[webhookKey] = webhookData; if (!this.webhookUrls[webhookKey]) {
this.webhookUrls[webhookKey] = [];
}
this.webhookUrls[webhookKey].push(webhookData);
try { try {
const webhookExists = await workflow.runWebhookMethod('checkExists', webhookData, NodeExecuteFunctions, mode, this.testWebhooks); const webhookExists = await workflow.runWebhookMethod('checkExists', webhookData, NodeExecuteFunctions, mode, this.testWebhooks);
@ -59,7 +62,11 @@ export class ActiveWebhooks {
} }
} catch (error) { } catch (error) {
// If there was a problem unregister the webhook again // If there was a problem unregister the webhook again
delete this.webhookUrls[webhookKey]; if (this.webhookUrls[webhookKey].length <= 1) {
delete this.webhookUrls[webhookKey];
} else {
this.webhookUrls[webhookKey] = this.webhookUrls[webhookKey].filter(webhook => webhook.path !== webhookData.path);
}
throw error; throw error;
} }
@ -72,16 +79,39 @@ export class ActiveWebhooks {
* *
* @param {WebhookHttpMethod} httpMethod * @param {WebhookHttpMethod} httpMethod
* @param {string} path * @param {string} path
* @param {(string | undefined)} webhookId
* @returns {(IWebhookData | undefined)} * @returns {(IWebhookData | undefined)}
* @memberof ActiveWebhooks * @memberof ActiveWebhooks
*/ */
get(httpMethod: WebhookHttpMethod, path: string): IWebhookData | undefined { get(httpMethod: WebhookHttpMethod, path: string, webhookId?: string): IWebhookData | undefined {
const webhookKey = this.getWebhookKey(httpMethod, path); const webhookKey = this.getWebhookKey(httpMethod, path, webhookId);
if (this.webhookUrls[webhookKey] === undefined) { if (this.webhookUrls[webhookKey] === undefined) {
return undefined; return undefined;
} }
return this.webhookUrls[webhookKey]; // set webhook to the first webhook result
// if more results have been returned choose the one with the most route-matches
let webhook = this.webhookUrls[webhookKey][0];
if (this.webhookUrls[webhookKey].length > 1) {
let maxMatches = 0;
const pathElementsSet = new Set(path.split('/'));
this.webhookUrls[webhookKey].forEach(dynamicWebhook => {
const intersection =
dynamicWebhook.path
.split('/')
.reduce((acc, element) => pathElementsSet.has(element) ? acc += 1 : acc, 0);
if (intersection > maxMatches) {
maxMatches = intersection;
webhook = dynamicWebhook;
}
});
if (maxMatches === 0) {
return undefined;
}
}
return webhook;
} }
/** /**
@ -116,10 +146,18 @@ export class ActiveWebhooks {
* *
* @param {WebhookHttpMethod} httpMethod * @param {WebhookHttpMethod} httpMethod
* @param {string} path * @param {string} path
* @param {(string | undefined)} webhookId
* @returns {string} * @returns {string}
* @memberof ActiveWebhooks * @memberof ActiveWebhooks
*/ */
getWebhookKey(httpMethod: WebhookHttpMethod, path: string): string { getWebhookKey(httpMethod: WebhookHttpMethod, path: string, webhookId?: string): string {
if (webhookId) {
if (path.startsWith(webhookId)) {
const cutFromIndex = path.indexOf('/') + 1;
path = path.slice(cutFromIndex);
}
return `${httpMethod}|${webhookId}|${path.split('/').length}`;
}
return `${httpMethod}|${path}`; return `${httpMethod}|${path}`;
} }
@ -147,7 +185,7 @@ export class ActiveWebhooks {
for (const webhookData of webhooks) { for (const webhookData of webhooks) {
await workflow.runWebhookMethod('delete', webhookData, NodeExecuteFunctions, mode, this.testWebhooks); await workflow.runWebhookMethod('delete', webhookData, NodeExecuteFunctions, mode, this.testWebhooks);
delete this.webhookUrls[this.getWebhookKey(webhookData.httpMethod, webhookData.path)]; delete this.webhookUrls[this.getWebhookKey(webhookData.httpMethod, webhookData.path, webhookData.webhookId)];
} }
// Remove also the workflow-webhook entry // Remove also the workflow-webhook entry

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;
@ -566,6 +567,7 @@ export interface IWebhookData {
webhookDescription: IWebhookDescription; webhookDescription: IWebhookDescription;
workflowId: string; workflowId: string;
workflowExecuteAdditionalData: IWorkflowExecuteAdditionalData; workflowExecuteAdditionalData: IWorkflowExecuteAdditionalData;
webhookId?: string;
} }
export interface IWebhookDescription { export interface IWebhookDescription {

View file

@ -641,13 +641,13 @@ export function getNodeParameters(nodePropertiesArray: INodeProperties[], nodeVa
} }
} }
// Itterate over all collections // Iterate over all collections
for (const itemName of Object.keys(propertyValues || {})) { for (const itemName of Object.keys(propertyValues || {})) {
if (nodeProperties.typeOptions !== undefined && nodeProperties.typeOptions.multipleValues === true) { if (nodeProperties.typeOptions !== undefined && nodeProperties.typeOptions.multipleValues === true) {
// Multiple can be set so will be an array // Multiple can be set so will be an array
const tempArrayValue: INodeParameters[] = []; const tempArrayValue: INodeParameters[] = [];
// Itterate over all items as it contains multiple ones // Iterate over all items as it contains multiple ones
for (const nodeValue of (propertyValues as INodeParameters)[itemName] as INodeParameters[]) { for (const nodeValue of (propertyValues as INodeParameters)[itemName] as INodeParameters[]) {
nodePropertyOptions = nodeProperties!.options!.find((nodePropertyOptions) => nodePropertyOptions.name === itemName) as INodePropertyCollection; nodePropertyOptions = nodeProperties!.options!.find((nodePropertyOptions) => nodePropertyOptions.name === itemName) as INodePropertyCollection;
@ -779,6 +779,11 @@ export function getNodeWebhooks(workflow: Workflow, node: INode, additionalData:
continue; continue;
} }
let webhookId: string | undefined;
if ((path.startsWith(':') || path.includes('/:')) && node.webhookId) {
webhookId = node.webhookId;
}
returnData.push({ returnData.push({
httpMethod: httpMethod.toString() as WebhookHttpMethod, httpMethod: httpMethod.toString() as WebhookHttpMethod,
node: node.name, node: node.name,
@ -786,6 +791,7 @@ export function getNodeWebhooks(workflow: Workflow, node: INode, additionalData:
webhookDescription, webhookDescription,
workflowId, workflowId,
workflowExecuteAdditionalData: additionalData, workflowExecuteAdditionalData: additionalData,
webhookId,
}); });
} }
@ -883,6 +889,13 @@ 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.startsWith(':') || path.includes('/:')) && node.webhookId) {
// setting this to false to prefix the webhookId
isFullPath = false;
}
if (path.startsWith('/')) {
path = path.slice(1);
}
return `${baseUrl}/${getNodeWebhookPath(workflowId, node, path, isFullPath)}`; return `${baseUrl}/${getNodeWebhookPath(workflowId, node, path, isFullPath)}`;
} }