From e3aff74f6b0ecbf87dccac9be99405fde52b87b2 Mon Sep 17 00:00:00 2001 From: Jan Date: Wed, 23 Dec 2020 08:05:02 +0100 Subject: [PATCH] :sparkles: Add RabbitMQ and RabbitMQ Trigger Node (#1258) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * :sparkles: Add RabbitMQ-Node * :sparkles: Add RabbitMQ-Trigger Node * :zap: Fix issue that connection errors did not get caught * 🔨Fix name and description for RabbitMQ Trigger node Co-authored-by: Harshil --- .../credentials/RabbitMQ.credentials.ts | 157 ++++++++++++++++ .../nodes/RabbitMQ/DefaultOptions.ts | 60 ++++++ .../nodes/RabbitMQ/GenericFunctions.ts | 62 +++++++ .../nodes/RabbitMQ/RabbitMQ.node.ts | 147 +++++++++++++++ .../nodes/RabbitMQ/RabbitMQTrigger.node.ts | 172 ++++++++++++++++++ .../nodes-base/nodes/RabbitMQ/rabbitmq.png | Bin 0 -> 527 bytes packages/nodes-base/package.json | 4 + 7 files changed, 602 insertions(+) create mode 100644 packages/nodes-base/credentials/RabbitMQ.credentials.ts create mode 100644 packages/nodes-base/nodes/RabbitMQ/DefaultOptions.ts create mode 100644 packages/nodes-base/nodes/RabbitMQ/GenericFunctions.ts create mode 100644 packages/nodes-base/nodes/RabbitMQ/RabbitMQ.node.ts create mode 100644 packages/nodes-base/nodes/RabbitMQ/RabbitMQTrigger.node.ts create mode 100644 packages/nodes-base/nodes/RabbitMQ/rabbitmq.png diff --git a/packages/nodes-base/credentials/RabbitMQ.credentials.ts b/packages/nodes-base/credentials/RabbitMQ.credentials.ts new file mode 100644 index 0000000000..046650f58a --- /dev/null +++ b/packages/nodes-base/credentials/RabbitMQ.credentials.ts @@ -0,0 +1,157 @@ +import { + ICredentialType, + NodePropertyTypes, +} from 'n8n-workflow'; + +export class RabbitMQ implements ICredentialType { + name = 'rabbitmq'; + displayName = 'RabbitMQ'; + documentationUrl = 'rabbitmq'; + properties = [ + { + displayName: 'Hostname', + name: 'hostname', + type: 'string' as NodePropertyTypes, + default: '', + placeholder: 'localhost', + }, + { + displayName: 'Port', + name: 'port', + type: 'number' as NodePropertyTypes, + default: 5672, + }, + { + displayName: 'User', + name: 'username', + type: 'string' as NodePropertyTypes, + default: '', + placeholder: 'guest', + }, + { + displayName: 'Password', + name: 'password', + type: 'string' as NodePropertyTypes, + typeOptions: { + password: true, + }, + default: '', + placeholder: 'guest', + }, + { + displayName: 'Vhost', + name: 'vhost', + type: 'string' as NodePropertyTypes, + default: '/', + }, + { + displayName: 'SSL', + name: 'ssl', + type: 'boolean' as NodePropertyTypes, + default: false, + }, + { + displayName: 'Client Certificate', + name: 'cert', + type: 'string' as NodePropertyTypes, + typeOptions: { + password: true, + }, + displayOptions: { + show: { + ssl: [ + true, + ], + }, + }, + default: '', + description: 'SSL Client Certificate to use.', + }, + { + displayName: 'Client Key', + name: 'key', + type: 'string' as NodePropertyTypes, + typeOptions: { + password: true, + }, + displayOptions: { + show: { + ssl: [ + true, + ], + }, + }, + default: '', + description: 'SSL Client Key to use.', + }, + { + displayName: 'Passphrase', + name: 'passphrase', + type: 'string' as NodePropertyTypes, + typeOptions: { + password: true, + }, + displayOptions: { + show: { + ssl: [ + true, + ], + }, + }, + default: '', + description: 'SSL passphrase to use.', + }, + { + displayName: 'CA Certificates', + name: 'ca', + type: 'string' as NodePropertyTypes, + typeOptions: { + password: true, + }, + // typeOptions: { + // multipleValues: true, + // multipleValueButtonText: 'Add Certificate', + // }, + displayOptions: { + show: { + ssl: [ + true, + ], + }, + }, + default: '', + description: 'SSL CA Certificates to use.', + }, + // { + // displayName: 'Client ID', + // name: 'clientId', + // type: 'string' as NodePropertyTypes, + // default: '', + // placeholder: 'my-app', + // }, + // { + // displayName: 'Brokers', + // name: 'brokers', + // type: 'string' as NodePropertyTypes, + // default: '', + // placeholder: 'kafka1:9092,kafka2:9092', + // }, + // { + // displayName: 'Username', + // name: 'username', + // type: 'string' as NodePropertyTypes, + // default: '', + // description: 'Optional username if authenticated is required.', + // }, + // { + // displayName: 'Password', + // name: 'password', + // type: 'string' as NodePropertyTypes, + // typeOptions: { + // password: true, + // }, + // default: '', + // description: 'Optional password if authenticated is required.', + // }, + ]; +} diff --git a/packages/nodes-base/nodes/RabbitMQ/DefaultOptions.ts b/packages/nodes-base/nodes/RabbitMQ/DefaultOptions.ts new file mode 100644 index 0000000000..10d295dc03 --- /dev/null +++ b/packages/nodes-base/nodes/RabbitMQ/DefaultOptions.ts @@ -0,0 +1,60 @@ +import { + INodeProperties, + INodePropertyCollection, + INodePropertyOptions, +} from 'n8n-workflow'; + +export const rabbitDefaultOptions: Array = [ + { + displayName: 'Arguments', + name: 'arguments', + placeholder: 'Add Argument', + description: 'Arguments to add.', + type: 'fixedCollection', + typeOptions: { + multipleValues: true, + }, + default: {}, + options: [ + { + name: 'argument', + displayName: 'Argument', + values: [ + { + displayName: 'Key', + name: 'key', + type: 'string', + default: '', + }, + { + displayName: 'Value', + name: 'value', + type: 'string', + default: '', + }, + ], + }, + ], + }, + { + displayName: 'Auto Delete', + name: 'autoDelete', + type: 'boolean', + default: false, + description: 'The queue will be deleted when the number of consumers drops to zero .', + }, + { + displayName: 'Durable', + name: 'durable', + type: 'boolean', + default: true, + description: 'The queue will survive broker restarts.', + }, + { + displayName: 'Exclusive', + name: 'exclusive', + type: 'boolean', + default: false, + description: 'Scopes the queue to the connection.', + }, +]; diff --git a/packages/nodes-base/nodes/RabbitMQ/GenericFunctions.ts b/packages/nodes-base/nodes/RabbitMQ/GenericFunctions.ts new file mode 100644 index 0000000000..13659704a1 --- /dev/null +++ b/packages/nodes-base/nodes/RabbitMQ/GenericFunctions.ts @@ -0,0 +1,62 @@ +import { + IDataObject, + IExecuteFunctions, + ITriggerFunctions, +} from 'n8n-workflow'; + +const amqplib = require('amqplib'); + +export async function rabbitmqConnect(this: IExecuteFunctions | ITriggerFunctions, queue: string, options: IDataObject): Promise { // tslint:disable-line:no-any + const credentials = this.getCredentials('rabbitmq') as IDataObject; + + const credentialKeys = [ + 'hostname', + 'port', + 'username', + 'password', + 'vhost', + ]; + const credentialData: IDataObject = {}; + credentialKeys.forEach(key => { + credentialData[key] = credentials[key] === '' ? undefined : credentials[key]; + }); + + const optsData: IDataObject = {}; + if (credentials.ssl === true) { + credentialData.protocol = 'amqps'; + + optsData.cert = credentials.cert === '' ? undefined : Buffer.from(credentials.cert as string); + optsData.key = credentials.key === '' ? undefined : Buffer.from(credentials.key as string); + optsData.passphrase = credentials.passphrase === '' ? undefined : credentials.passphrase; + optsData.ca = credentials.ca === '' ? undefined : [Buffer.from(credentials.ca as string)]; + optsData.credentials = amqplib.credentials.external(); + } + + + return new Promise(async (resolve, reject) => { + try { + const connection = await amqplib.connect(credentialData, optsData); + + connection.on('error', (error: Error) => { + reject(error); + }); + + const channel = await connection.createChannel().catch(console.warn); + + if (options.arguments && ((options.arguments as IDataObject).argument! as IDataObject[]).length) { + const additionalArguments: IDataObject = {}; + ((options.arguments as IDataObject).argument as IDataObject[]).forEach((argument: IDataObject) => { + additionalArguments[argument.key as string] = argument.value; + }); + options.arguments = additionalArguments; + } + + await channel.assertQueue(queue, options); + + resolve(channel); + } catch (error) { + reject(error); + } + }); + +} diff --git a/packages/nodes-base/nodes/RabbitMQ/RabbitMQ.node.ts b/packages/nodes-base/nodes/RabbitMQ/RabbitMQ.node.ts new file mode 100644 index 0000000000..9103fa5027 --- /dev/null +++ b/packages/nodes-base/nodes/RabbitMQ/RabbitMQ.node.ts @@ -0,0 +1,147 @@ +import { + IExecuteFunctions, +} from 'n8n-core'; + +import { + IDataObject, + INodeExecutionData, + INodeType, + INodeTypeDescription, +} from 'n8n-workflow'; + +import { + rabbitDefaultOptions, +} from './DefaultOptions'; + +import { + rabbitmqConnect, +} from './GenericFunctions'; + +export class RabbitMQ implements INodeType { + description: INodeTypeDescription = { + displayName: 'RabbitMQ', + name: 'rabbitmq', + icon: 'file:rabbitmq.png', + group: ['transform'], + version: 1, + description: 'Sends messages to a RabbitMQ topic', + defaults: { + name: 'RabbitMQ', + color: '#ff6600', + }, + inputs: ['main'], + outputs: ['main'], + credentials: [ + { + name: 'rabbitmq', + required: true, + }, + ], + properties: [ + { + displayName: 'Queue / Topic', + name: 'queue', + type: 'string', + default: '', + placeholder: 'queue-name', + description: 'Name of the queue to publish to.', + }, + { + displayName: 'Send Input Data', + name: 'sendInputData', + type: 'boolean', + default: true, + description: 'Send the the data the node receives as JSON to Kafka.', + }, + { + displayName: 'Message', + name: 'message', + type: 'string', + displayOptions: { + show: { + sendInputData: [ + false, + ], + }, + }, + default: '', + description: 'The message to be sent.', + }, + { + displayName: 'Options', + name: 'options', + type: 'collection', + default: {}, + placeholder: 'Add Option', + options: rabbitDefaultOptions, + }, + ], + }; + + async execute(this: IExecuteFunctions): Promise { + let channel; + try { + const items = this.getInputData(); + + const queue = this.getNodeParameter('queue', 0) as string; + + const options = this.getNodeParameter('options', 0, {}) as IDataObject; + + channel = await rabbitmqConnect.call(this, queue, options); + + const sendInputData = this.getNodeParameter('sendInputData', 0) as boolean; + + let message: string; + + const queuePromises = []; + for (let i = 0; i < items.length; i++) { + if (sendInputData === true) { + message = JSON.stringify(items[i].json); + } else { + message = this.getNodeParameter('message', i) as string; + } + + queuePromises.push(channel.sendToQueue(queue, Buffer.from(message))); + } + + // @ts-ignore + const promisesResponses = await Promise.allSettled(queuePromises); + + const returnItems: INodeExecutionData[] = []; + + promisesResponses.forEach((response: IDataObject) => { + if (response!.status !== 'fulfilled') { + + if (this.continueOnFail() !== true) { + throw new Error(response!.reason as string); + } else { + // Return the actual reason as error + returnItems.push( + { + json: { + error: response.reason, + }, + }, + ); + return; + } + } + + returnItems.push({ + json: { + success: response.value, + }, + }); + }); + + await channel.close(); + + return this.prepareOutputData(returnItems); + } catch (error) { + if (channel) { + await channel.close(); + } + throw error; + } + } +} diff --git a/packages/nodes-base/nodes/RabbitMQ/RabbitMQTrigger.node.ts b/packages/nodes-base/nodes/RabbitMQ/RabbitMQTrigger.node.ts new file mode 100644 index 0000000000..67deb8d882 --- /dev/null +++ b/packages/nodes-base/nodes/RabbitMQ/RabbitMQTrigger.node.ts @@ -0,0 +1,172 @@ +import { + IDataObject, + INodeExecutionData, + INodeProperties, + INodeType, + INodeTypeDescription, + ITriggerFunctions, + ITriggerResponse, +} from 'n8n-workflow'; + +import { + rabbitDefaultOptions, +} from './DefaultOptions'; + +import { + rabbitmqConnect, +} from './GenericFunctions'; + +export class RabbitMQTrigger implements INodeType { + description: INodeTypeDescription = { + displayName: 'RabbitMQ Trigger', + name: 'rabbitmqTrigger', + icon: 'file:rabbitmq.png', + group: ['trigger'], + version: 1, + description: 'Listens to RabbitMQ messages', + defaults: { + name: 'RabbitMQ', + color: '#ff6600', + }, + inputs: [], + outputs: ['main'], + credentials: [ + { + name: 'rabbitmq', + required: true, + }, + ], + properties: [ + { + displayName: 'Queue / Topic', + name: 'queue', + type: 'string', + default: '', + placeholder: 'queue-name', + description: 'Name of the queue to publish to.', + }, + + { + displayName: 'Options', + name: 'options', + type: 'collection', + default: {}, + placeholder: 'Add Option', + options: [ + { + displayName: 'Content is Binary', + name: 'contentIsBinary', + type: 'boolean', + default: false, + description: 'Saves the content as binary.', + }, + { + displayName: 'JSON Parse Body', + name: 'jsonParseBody', + type: 'boolean', + displayOptions: { + hide: { + contentIsBinary: [ + true, + ], + }, + }, + default: false, + description: 'Parse the body to an object.', + }, + { + displayName: 'Only Content', + name: 'onlyContent', + type: 'boolean', + displayOptions: { + hide: { + contentIsBinary: [ + true, + ], + }, + }, + default: false, + description: 'Returns only the content property.', + }, + ...rabbitDefaultOptions, + ].sort((a, b) => { + if ((a as INodeProperties).displayName.toLowerCase() < (b as INodeProperties).displayName.toLowerCase()) { return -1; } + if ((a as INodeProperties).displayName.toLowerCase() > (b as INodeProperties).displayName.toLowerCase()) { return 1; } + return 0; + }) as INodeProperties[], + }, + ], + }; + + + async trigger(this: ITriggerFunctions): Promise { + const queue = this.getNodeParameter('queue') as string; + const options = this.getNodeParameter('options', {}) as IDataObject; + + const channel = await rabbitmqConnect.call(this, queue, options); + + const self = this; + + const item: INodeExecutionData = { + json: {}, + }; + + const startConsumer = async () => { + await channel.consume(queue, async (message: IDataObject) => { + if (message !== null) { + let content: IDataObject | string = message!.content!.toString(); + + if (options.contentIsBinary === true) { + item.binary = { + data: await this.helpers.prepareBinaryData(message.content), + }; + + item.json = message; + message.content = undefined; + } else { + if (options.jsonParseBody === true) { + content = JSON.parse(content as string); + } + if (options.onlyContent === true) { + item.json = content as IDataObject; + } else { + message.content = content; + item.json = message; + } + } + + self.emit([ + [ + item, + ], + ]); + channel.ack(message); + } + }); + }; + + startConsumer(); + + // The "closeFunction" function gets called by n8n whenever + // the workflow gets deactivated and can so clean up. + async function closeFunction() { + await channel.close(); + } + + // The "manualTriggerFunction" function gets called by n8n + // when a user is in the workflow editor and starts the + // workflow manually. So the function has to make sure that + // the emit() gets called with similar data like when it + // would trigger by itself so that the user knows what data + // to expect. + async function manualTriggerFunction() { + startConsumer(); + } + + return { + closeFunction, + manualTriggerFunction, + }; + } + +} diff --git a/packages/nodes-base/nodes/RabbitMQ/rabbitmq.png b/packages/nodes-base/nodes/RabbitMQ/rabbitmq.png new file mode 100644 index 0000000000000000000000000000000000000000..d37a2a0fbeeb23a1db89fefbf9f0988ed88d4273 GIT binary patch literal 527 zcmV+q0`UEbP)H{rhJG{bmCBX9oWN{r>#)|M>3y{`&EQHvazi{r>;;X$t@K>G^63{PE!ZW&`~H z{`~s#`u_azt#|Ts9sB(D`|;)a?b!Ee3H|o$^lT6PX94>C`u_g%`RCI6Wd-*3?)Tor z^S_zwmsjtUQ1N;x_WSer;>h&Kp76(=^u3Srs&4IzNA`v~>V7TwaTxM&7dUJ~-v9ss zF-b&0R7l6|(`j#mARNc>BKWTq>;|%GJ$fIzulN7rhGL9M8YN`QX6pBW$o z%QGLFJjr4O9%WGf00nxkqz&dQdb-mcf?7w<<$PS(V7YolW#pY!GhS3n9AoImF!d`e zwn2pxWJ;B?#|Y2M$rEg5&MdH3V}j?PC)^^_4do*q@jt>+Pt{;1?J-I-s*ls8xf$%S zz#9%4?+b%*V2^d@UCdarAoV=nvsMbA6AM2KL%gjQA!*p^yz9oOKtzX{NUQWDJ8)0M z4BRoEdG4581s1*z