diff --git a/packages/cli/package.json b/packages/cli/package.json index f69a2719f5..48bbacb418 100644 --- a/packages/cli/package.json +++ b/packages/cli/package.json @@ -1,6 +1,6 @@ { "name": "n8n", - "version": "0.41.0", + "version": "0.42.0", "description": "n8n Workflow Automation Tool", "license": "SEE LICENSE IN LICENSE.md", "homepage": "https://n8n.io", @@ -92,10 +92,10 @@ "localtunnel": "^2.0.0", "lodash.get": "^4.4.2", "mongodb": "^3.2.3", - "n8n-core": "~0.18.0", - "n8n-editor-ui": "~0.29.0", - "n8n-nodes-base": "~0.36.0", - "n8n-workflow": "~0.18.0", + "n8n-core": "~0.19.0", + "n8n-editor-ui": "~0.30.0", + "n8n-nodes-base": "~0.37.0", + "n8n-workflow": "~0.19.0", "open": "^7.0.0", "pg": "^7.11.0", "request-promise-native": "^1.0.7", diff --git a/packages/cli/src/ActiveWorkflowRunner.ts b/packages/cli/src/ActiveWorkflowRunner.ts index 6a0e5cb02b..9604fd9de6 100644 --- a/packages/cli/src/ActiveWorkflowRunner.ts +++ b/packages/cli/src/ActiveWorkflowRunner.ts @@ -21,6 +21,7 @@ import { import { IExecuteData, + IGetExecutePollFunctions, IGetExecuteTriggerFunctions, INode, INodeExecutionData, @@ -218,6 +219,73 @@ export class ActiveWorkflowRunner { } + /** + * Runs the given workflow + * + * @param {IWorkflowDb} workflowData + * @param {INode} node + * @param {INodeExecutionData[][]} data + * @param {IWorkflowExecuteAdditionalDataWorkflow} additionalData + * @param {WorkflowExecuteMode} mode + * @returns + * @memberof ActiveWorkflowRunner + */ + runWorkflow(workflowData: IWorkflowDb, node: INode, data: INodeExecutionData[][], additionalData: IWorkflowExecuteAdditionalDataWorkflow, mode: WorkflowExecuteMode) { + const nodeExecutionStack: IExecuteData[] = [ + { + node, + data: { + main: data, + } + } + ]; + + const executionData: IRunExecutionData = { + startData: {}, + resultData: { + runData: {}, + }, + executionData: { + contextData: {}, + nodeExecutionStack, + waitingExecution: {}, + }, + }; + + // Start the workflow + const runData: IWorkflowExecutionDataProcess = { + credentials: additionalData.credentials, + executionMode: mode, + executionData, + workflowData, + }; + + const workflowRunner = new WorkflowRunner(); + return workflowRunner.run(runData, true); + } + + + /** + * Return poll function which gets the global functions from n8n-core + * and overwrites the __emit to be able to start it in subprocess + * + * @param {IWorkflowDb} workflowData + * @param {IWorkflowExecuteAdditionalDataWorkflow} additionalData + * @param {WorkflowExecuteMode} mode + * @returns {IGetExecutePollFunctions} + * @memberof ActiveWorkflowRunner + */ + getExecutePollFunctions(workflowData: IWorkflowDb, additionalData: IWorkflowExecuteAdditionalDataWorkflow, mode: WorkflowExecuteMode): IGetExecutePollFunctions { + return ((workflow: Workflow, node: INode) => { + const returnFunctions = NodeExecuteFunctions.getExecutePollFunctions(workflow, node, additionalData, mode); + returnFunctions.__emit = (data: INodeExecutionData[][]): void => { + this.runWorkflow(workflowData, node, data, additionalData, mode); + }; + return returnFunctions; + }); + } + + /** * Return trigger function which gets the global functions from n8n-core * and overwrites the emit to be able to start it in subprocess @@ -232,43 +300,13 @@ export class ActiveWorkflowRunner { return ((workflow: Workflow, node: INode) => { const returnFunctions = NodeExecuteFunctions.getExecuteTriggerFunctions(workflow, node, additionalData, mode); returnFunctions.emit = (data: INodeExecutionData[][]): void => { - - const nodeExecutionStack: IExecuteData[] = [ - { - node, - data: { - main: data, - } - } - ]; - - const executionData: IRunExecutionData = { - startData: {}, - resultData: { - runData: {}, - }, - executionData: { - contextData: {}, - nodeExecutionStack, - waitingExecution: {}, - }, - }; - - // Start the workflow - const runData: IWorkflowExecutionDataProcess = { - credentials: additionalData.credentials, - executionMode: mode, - executionData, - workflowData, - }; - - const workflowRunner = new WorkflowRunner(); - workflowRunner.run(runData, true); + this.runWorkflow(workflowData, node, data, additionalData, mode); }; return returnFunctions; }); } + /** * Makes a workflow active * @@ -303,10 +341,11 @@ export class ActiveWorkflowRunner { const credentials = await WorkflowCredentials(workflowData.nodes); const additionalData = await WorkflowExecuteAdditionalData.getBase(credentials); const getTriggerFunctions = this.getExecuteTriggerFunctions(workflowData, additionalData, mode); + const getPollFunctions = this.getExecutePollFunctions(workflowData, additionalData, mode); // Add the workflows which have webhooks defined await this.addWorkflowWebhooks(workflowInstance, additionalData, mode); - await this.activeWorkflows.add(workflowId, workflowInstance, additionalData, getTriggerFunctions); + await this.activeWorkflows.add(workflowId, workflowInstance, additionalData, getTriggerFunctions, getPollFunctions); if (this.activationErrors[workflowId] !== undefined) { // If there were any activation errors delete them diff --git a/packages/cli/src/NodeTypes.ts b/packages/cli/src/NodeTypes.ts index aaae63effd..66b2363bfe 100644 --- a/packages/cli/src/NodeTypes.ts +++ b/packages/cli/src/NodeTypes.ts @@ -2,6 +2,7 @@ import { INodeType, INodeTypes, INodeTypeData, + NodeHelpers, } from 'n8n-workflow'; @@ -11,6 +12,15 @@ class NodeTypesClass implements INodeTypes { async init(nodeTypes: INodeTypeData): Promise { + // Some nodeTypes need to get special parameters applied like the + // polling nodes the polling times + for (const nodeTypeData of Object.values(nodeTypes)) { + const applyParameters = NodeHelpers.getSpecialNodeParameters(nodeTypeData.type) + + if (applyParameters.length) { + nodeTypeData.type.description.properties.unshift.apply(nodeTypeData.type.description.properties, applyParameters); + } + } this.nodeTypes = nodeTypes; } diff --git a/packages/core/package.json b/packages/core/package.json index 716f466f02..31eed262ca 100644 --- a/packages/core/package.json +++ b/packages/core/package.json @@ -1,6 +1,6 @@ { "name": "n8n-core", - "version": "0.18.0", + "version": "0.19.0", "description": "Core functionality of n8n", "license": "SEE LICENSE IN LICENSE.md", "homepage": "https://n8n.io", @@ -39,10 +39,11 @@ "typescript": "~3.7.4" }, "dependencies": { + "cron": "^1.7.2", "crypto-js": "^3.1.9-1", "lodash.get": "^4.4.2", "mmmagic": "^0.5.2", - "n8n-workflow": "~0.18.0", + "n8n-workflow": "~0.19.0", "request-promise-native": "^1.0.7" }, "jest": { diff --git a/packages/core/src/ActiveWorkflows.ts b/packages/core/src/ActiveWorkflows.ts index 0a7b3df268..8ae57291d4 100644 --- a/packages/core/src/ActiveWorkflows.ts +++ b/packages/core/src/ActiveWorkflows.ts @@ -1,19 +1,23 @@ +import { CronJob } from 'cron'; + import { + IGetExecutePollFunctions, IGetExecuteTriggerFunctions, + INode, + IPollResponse, ITriggerResponse, IWorkflowExecuteAdditionalData, Workflow, } from 'n8n-workflow'; - -export interface WorkflowData { - workflow: Workflow; - triggerResponse?: ITriggerResponse; -} +import { + ITriggerTime, + IWorkflowData, +} from './'; export class ActiveWorkflows { private workflowData: { - [key: string]: WorkflowData; + [key: string]: IWorkflowData; } = {}; @@ -48,7 +52,7 @@ export class ActiveWorkflows { * @returns {(WorkflowData | undefined)} * @memberof ActiveWorkflows */ - get(id: string): WorkflowData | undefined { + get(id: string): IWorkflowData | undefined { return this.workflowData[id]; } @@ -62,7 +66,7 @@ export class ActiveWorkflows { * @returns {Promise} * @memberof ActiveWorkflows */ - async add(id: string, workflow: Workflow, additionalData: IWorkflowExecuteAdditionalData, getTriggerFunctions: IGetExecuteTriggerFunctions): Promise { + async add(id: string, workflow: Workflow, additionalData: IWorkflowExecuteAdditionalData, getTriggerFunctions: IGetExecuteTriggerFunctions, getPollFunctions: IGetExecutePollFunctions): Promise { console.log('ADD ID (active): ' + id); this.workflowData[id] = { @@ -78,9 +82,118 @@ export class ActiveWorkflows { this.workflowData[id].triggerResponse = triggerResponse; } } + + const pollNodes = workflow.getPollNodes(); + for (const pollNode of pollNodes) { + this.workflowData[id].pollResponse = await this.activatePolling(pollNode, workflow, additionalData, getPollFunctions); + } } + /** + * Activates polling for the given node + * + * @param {INode} node + * @param {Workflow} workflow + * @param {IWorkflowExecuteAdditionalData} additionalData + * @param {IGetExecutePollFunctions} getPollFunctions + * @returns {Promise} + * @memberof ActiveWorkflows + */ + async activatePolling(node: INode, workflow: Workflow, additionalData: IWorkflowExecuteAdditionalData, getPollFunctions: IGetExecutePollFunctions): Promise { + const mode = 'trigger'; + + const pollFunctions = getPollFunctions(workflow, node, additionalData, mode); + + const pollTimes = pollFunctions.getNodeParameter('pollTimes') as unknown as { + item: ITriggerTime[]; + }; + + // Define the order the cron-time-parameter appear + const parameterOrder = [ + 'second', // 0 - 59 + 'minute', // 0 - 59 + 'hour', // 0 - 23 + 'dayOfMonth', // 1 - 31 + 'month', // 0 - 11(Jan - Dec) + 'weekday', // 0 - 6(Sun - Sat) + ]; + + // Get all the trigger times + const cronTimes: string[] = []; + let cronTime: string[]; + let parameterName: string; + if (pollTimes.item !== undefined) { + for (const item of pollTimes.item) { + cronTime = []; + if (item.mode === 'custom') { + cronTimes.push(item.cronExpression as string); + continue; + } + if (item.mode === 'everyMinute') { + cronTimes.push(`${Math.floor(Math.random() * 60).toString()} * * * * *`); + continue; + } + if (item.mode === 'everyX') { + if (item.unit === 'minutes') { + cronTimes.push(`${Math.floor(Math.random() * 60).toString()} */${item.value} * * * *`); + } else if (item.unit === 'hours') { + cronTimes.push(`${Math.floor(Math.random() * 60).toString()} 0 */${item.value} * * *`); + } + continue; + } + + for (parameterName of parameterOrder) { + if (item[parameterName] !== undefined) { + // Value is set so use it + cronTime.push(item[parameterName] as string); + } else if (parameterName === 'second') { + // For seconds we use by default a random one to make sure to + // balance the load a little bit over time + cronTime.push(Math.floor(Math.random() * 60).toString()); + } else { + // For all others set "any" + cronTime.push('*'); + } + } + + cronTimes.push(cronTime.join(' ')); + } + } + + // The trigger function to execute when the cron-time got reached + const executeTrigger = async () => { + const pollResponse = await workflow.runPoll(node, pollFunctions); + + if (pollResponse !== null) { + // TODO: Run workflow + pollFunctions.__emit(pollResponse); + } + }; + + // Execute the trigger directly to be able to know if it works + await executeTrigger(); + + const timezone = pollFunctions.getTimezone(); + + // Start the cron-jobs + const cronJobs: CronJob[] = []; + for (const cronTime of cronTimes) { + cronJobs.push(new CronJob(cronTime, executeTrigger, undefined, true, timezone)); + } + + // Stop the cron-jobs + async function closeFunction() { + for (const cronJob of cronJobs) { + cronJob.stop(); + } + } + + return { + closeFunction, + }; + } + /** * Makes a workflow inactive @@ -103,6 +216,10 @@ export class ActiveWorkflows { await workflowData.triggerResponse.closeFunction(); } + if (workflowData.pollResponse && workflowData.pollResponse.closeFunction) { + await workflowData.pollResponse.closeFunction(); + } + delete this.workflowData[id]; } diff --git a/packages/core/src/Interfaces.ts b/packages/core/src/Interfaces.ts index 76b6eea932..3b9a501969 100644 --- a/packages/core/src/Interfaces.ts +++ b/packages/core/src/Interfaces.ts @@ -8,13 +8,16 @@ import { ILoadOptionsFunctions as ILoadOptionsFunctionsBase, INodeExecutionData, INodeType, + IPollFunctions as IPollFunctionsBase, + IPollResponse, ITriggerFunctions as ITriggerFunctionsBase, + ITriggerResponse, IWebhookFunctions as IWebhookFunctionsBase, IWorkflowSettings as IWorkflowSettingsWorkflow, + Workflow, } from 'n8n-workflow'; -import * as request from 'request'; import * as requestPromise from 'request-promise-native'; interface Constructable { @@ -31,7 +34,7 @@ export interface IProcessMessage { export interface IExecuteFunctions extends IExecuteFunctionsBase { helpers: { prepareBinaryData(binaryData: Buffer, filePath?: string, mimeType?: string): Promise; - request: request.RequestAPI, + request: requestPromise.RequestPromiseAPI, returnJsonArray(jsonData: IDataObject | IDataObject[]): INodeExecutionData[]; }; } @@ -40,7 +43,16 @@ export interface IExecuteFunctions extends IExecuteFunctionsBase { export interface IExecuteSingleFunctions extends IExecuteSingleFunctionsBase { helpers: { prepareBinaryData(binaryData: Buffer, filePath?: string, mimeType?: string): Promise; - request: request.RequestAPI < requestPromise.RequestPromise, requestPromise.RequestPromiseOptions, request.RequiredUriUrl >, + request: requestPromise.RequestPromiseAPI, + }; +} + + +export interface IPollFunctions extends IPollFunctionsBase { + helpers: { + prepareBinaryData(binaryData: Buffer, filePath?: string, mimeType?: string): Promise; + request: requestPromise.RequestPromiseAPI, + returnJsonArray(jsonData: IDataObject | IDataObject[]): INodeExecutionData[]; }; } @@ -48,12 +60,22 @@ export interface IExecuteSingleFunctions extends IExecuteSingleFunctionsBase { export interface ITriggerFunctions extends ITriggerFunctionsBase { helpers: { prepareBinaryData(binaryData: Buffer, filePath?: string, mimeType?: string): Promise; - request: request.RequestAPI, + request: requestPromise.RequestPromiseAPI, returnJsonArray(jsonData: IDataObject | IDataObject[]): INodeExecutionData[]; }; } +export interface ITriggerTime { + mode: string; + hour: number; + minute: number; + dayOfMonth: number; + weekeday: number; + [key: string]: string | number; +} + + export interface IUserSettings { encryptionKey?: string; tunnelSubdomain?: string; @@ -61,14 +83,14 @@ export interface IUserSettings { export interface ILoadOptionsFunctions extends ILoadOptionsFunctionsBase { helpers: { - request?: request.RequestAPI, + request?: requestPromise.RequestPromiseAPI, }; } export interface IHookFunctions extends IHookFunctionsBase { helpers: { - request: request.RequestAPI, + request: requestPromise.RequestPromiseAPI, }; } @@ -76,7 +98,7 @@ export interface IHookFunctions extends IHookFunctionsBase { export interface IWebhookFunctions extends IWebhookFunctionsBase { helpers: { prepareBinaryData(binaryData: Buffer, filePath?: string, mimeType?: string): Promise; - request: request.RequestAPI, + request: requestPromise.RequestPromiseAPI, returnJsonArray(jsonData: IDataObject | IDataObject[]): INodeExecutionData[]; }; } @@ -98,3 +120,10 @@ export interface INodeDefinitionFile { export interface INodeInputDataConnections { [key: string]: INodeExecutionData[][]; } + + +export interface IWorkflowData { + pollResponse?: IPollResponse; + triggerResponse?: ITriggerResponse; + workflow: Workflow; +} diff --git a/packages/core/src/NodeExecuteFunctions.ts b/packages/core/src/NodeExecuteFunctions.ts index 775c2fd754..4fc7c2dd40 100644 --- a/packages/core/src/NodeExecuteFunctions.ts +++ b/packages/core/src/NodeExecuteFunctions.ts @@ -17,6 +17,7 @@ import { INodeExecutionData, INodeParameters, INodeType, + IPollFunctions, IRunExecutionData, ITaskDataConnections, ITriggerFunctions, @@ -310,6 +311,57 @@ export function getWebhookDescription(name: string, workflow: Workflow, node: IN +/** + * Returns the execute functions the poll nodes have access to. + * + * @export + * @param {Workflow} workflow + * @param {INode} node + * @param {IWorkflowExecuteAdditionalData} additionalData + * @param {WorkflowExecuteMode} mode + * @returns {ITriggerFunctions} + */ +// TODO: Check if I can get rid of: additionalData, and so then maybe also at ActiveWorkflowRunner.add +export function getExecutePollFunctions(workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode): IPollFunctions { + return ((workflow: Workflow, node: INode) => { + return { + __emit: (data: INodeExecutionData[][]): void => { + throw new Error('Overwrite NodeExecuteFunctions.getExecutePullFunctions.__emit function!'); + }, + getCredentials(type: string): ICredentialDataDecryptedObject | undefined { + return getCredentials(workflow, node, type, additionalData); + }, + getMode: (): WorkflowExecuteMode => { + return mode; + }, + getNodeParameter: (parameterName: string, fallbackValue?: any): NodeParameterValue | INodeParameters | NodeParameterValue[] | INodeParameters[] | object => { //tslint:disable-line:no-any + const runExecutionData: IRunExecutionData | null = null; + const itemIndex = 0; + const runIndex = 0; + const connectionInputData: INodeExecutionData[] = []; + + return getNodeParameter(workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, fallbackValue); + }, + getRestApiUrl: (): string => { + return additionalData.restApiUrl; + }, + getTimezone: (): string => { + return getTimezone(workflow, additionalData); + }, + getWorkflowStaticData(type: string): IDataObject { + return workflow.getStaticData(type, node); + }, + helpers: { + prepareBinaryData, + request: requestPromise, + returnJsonArray, + }, + }; + })(workflow, node); +} + + + /** * Returns the execute functions the trigger nodes have access to. * diff --git a/packages/editor-ui/package.json b/packages/editor-ui/package.json index 333d9fde9e..4da273419a 100644 --- a/packages/editor-ui/package.json +++ b/packages/editor-ui/package.json @@ -1,6 +1,6 @@ { "name": "n8n-editor-ui", - "version": "0.29.0", + "version": "0.30.1", "description": "Workflow Editor UI for n8n", "license": "SEE LICENSE IN LICENSE.md", "homepage": "https://n8n.io", @@ -43,7 +43,7 @@ "@vue/cli-plugin-eslint": "^4.1.2", "@vue/cli-plugin-typescript": "~4.1.2", "@vue/cli-plugin-unit-jest": "^4.1.2", - "@vue/cli-service": "^4.1.2", + "@vue/cli-service": "^3.11.0", "@vue/eslint-config-standard": "^5.0.1", "@vue/eslint-config-typescript": "~5.0.1", "@vue/test-utils": "^1.0.0-beta.24", @@ -63,7 +63,7 @@ "lodash.debounce": "^4.0.8", "lodash.get": "^4.4.2", "lodash.set": "^4.3.2", - "n8n-workflow": "~0.18.0", + "n8n-workflow": "~0.19.0", "node-sass": "^4.12.0", "prismjs": "^1.17.1", "quill": "^2.0.0-dev.3", diff --git a/packages/editor-ui/src/main.ts b/packages/editor-ui/src/main.ts index 048a718d96..f9a7f879a0 100644 --- a/packages/editor-ui/src/main.ts +++ b/packages/editor-ui/src/main.ts @@ -33,6 +33,7 @@ import { faCogs, faClone, faCloud, + faCloudDownloadAlt, faCopy, faCut, faDotCircle, @@ -106,6 +107,7 @@ library.add(faCog); library.add(faCogs); library.add(faClone); library.add(faCloud); +library.add(faCloudDownloadAlt); library.add(faCopy); library.add(faCut); library.add(faDotCircle); diff --git a/packages/nodes-base/credentials/TogglApi.credentials.ts b/packages/nodes-base/credentials/TogglApi.credentials.ts new file mode 100644 index 0000000000..d2af640c46 --- /dev/null +++ b/packages/nodes-base/credentials/TogglApi.credentials.ts @@ -0,0 +1,24 @@ +import { + ICredentialType, + NodePropertyTypes, +} from 'n8n-workflow'; + + +export class TogglApi implements ICredentialType { + name = 'togglApi'; + displayName = 'Toggl API'; + properties = [ + { + displayName: 'Username', + name: 'username', + type: 'string' as NodePropertyTypes, + default: '', + }, + { + displayName: 'Password', + name: 'password', + type: 'string' as NodePropertyTypes, + default: '', + }, + ]; +} diff --git a/packages/nodes-base/nodes/Cron.node.ts b/packages/nodes-base/nodes/Cron.node.ts index 90ad6214ce..7f91b401ee 100644 --- a/packages/nodes-base/nodes/Cron.node.ts +++ b/packages/nodes-base/nodes/Cron.node.ts @@ -55,27 +55,31 @@ export class Cron implements INodeType { options: [ { name: 'Every Minute', - value: 'everyMinute' + value: 'everyMinute', }, { name: 'Every Hour', - value: 'everyHour' + value: 'everyHour', }, { name: 'Every Day', - value: 'everyDay' + value: 'everyDay', }, { name: 'Every Week', - value: 'everyWeek' + value: 'everyWeek', }, { name: 'Every Month', - value: 'everyMonth' + value: 'everyMonth', + }, + { + name: 'Every X', + value: 'everyX', }, { name: 'Custom', - value: 'custom' + value: 'custom', }, ], default: 'everyDay', @@ -94,7 +98,8 @@ export class Cron implements INodeType { mode: [ 'custom', 'everyHour', - 'everyMinute' + 'everyMinute', + 'everyX', ], }, }, @@ -113,7 +118,8 @@ export class Cron implements INodeType { hide: { mode: [ 'custom', - 'everyMinute' + 'everyMinute', + 'everyX', ], }, }, @@ -196,6 +202,48 @@ export class Cron implements INodeType { default: '* * * * * *', description: 'Use custom cron expression. Values and ranges as follows:
  • Seconds: 0-59
  • Minutes: 0 - 59
  • Hours: 0 - 23
  • Day of Month: 1 - 31
  • Months: 0 - 11 (Jan - Dec)
  • Day of Week: 0 - 6 (Sun - Sat)
', }, + { + displayName: 'Value', + name: 'value', + type: 'number', + typeOptions: { + minValue: 0, + maxValue: 1000, + }, + displayOptions: { + show: { + mode: [ + 'everyX', + ], + }, + }, + default: 2, + description: 'All how many X minutes/hours it should trigger.', + }, + { + displayName: 'Unit', + name: 'unit', + type: 'options', + displayOptions: { + show: { + mode: [ + 'everyX', + ], + }, + }, + options: [ + { + name: 'Minutes', + value: 'minutes' + }, + { + name: 'Hours', + value: 'hours' + }, + ], + default: 'hours', + description: 'If it should trigger all X minutes or hours.', + }, ] }, ], @@ -236,6 +284,14 @@ export class Cron implements INodeType { cronTimes.push(`${Math.floor(Math.random() * 60).toString()} * * * * *`); continue; } + if (item.mode === 'everyX') { + if (item.unit === 'minutes') { + cronTimes.push(`${Math.floor(Math.random() * 60).toString()} */${item.value} * * * *`); + } else if (item.unit === 'hours') { + cronTimes.push(`${Math.floor(Math.random() * 60).toString()} 0 */${item.value} * * *`); + } + continue; + } for (parameterName of parameterOrder) { if (item[parameterName] !== undefined) { diff --git a/packages/nodes-base/nodes/SseTrigger.node.ts b/packages/nodes-base/nodes/SseTrigger.node.ts new file mode 100644 index 0000000000..c31e08c321 --- /dev/null +++ b/packages/nodes-base/nodes/SseTrigger.node.ts @@ -0,0 +1,57 @@ +import * as EventSource from 'eventsource'; +import { ITriggerFunctions } from 'n8n-core'; +import { + INodeType, + INodeTypeDescription, + ITriggerResponse, +} from 'n8n-workflow'; + + +export class SseTrigger implements INodeType { + description: INodeTypeDescription = { + displayName: 'SSE Trigger', + name: 'sseTrigger', + icon: 'fa:cloud-download-alt', + group: ['trigger'], + version: 1, + description: 'Triggers worklfow on a new Server-Sent Event', + defaults: { + name: 'SSE Trigger', + color: '#225577', + }, + inputs: [], + outputs: ['main'], + properties: [ + { + displayName: 'URL', + name: 'url', + type: 'string', + default: '', + placeholder: 'http://example.com', + description: 'The URL to receive the SSE from.', + required: true, + }, + ] + }; + + + async trigger(this: ITriggerFunctions): Promise { + const url = this.getNodeParameter('url') as string; + + const eventSource = new EventSource(url); + + eventSource.onmessage = (event) => { + const eventData = JSON.parse(event.data); + this.emit([this.helpers.returnJsonArray([eventData])]); + }; + + async function closeFunction() { + eventSource.close(); + } + + return { + closeFunction, + }; + + } +} diff --git a/packages/nodes-base/nodes/Toggl/GenericFunctions.ts b/packages/nodes-base/nodes/Toggl/GenericFunctions.ts new file mode 100644 index 0000000000..37c6bdb80b --- /dev/null +++ b/packages/nodes-base/nodes/Toggl/GenericFunctions.ts @@ -0,0 +1,49 @@ +import { OptionsWithUri } from 'request'; + +import { + IExecuteFunctions, + IHookFunctions, + ILoadOptionsFunctions, + IExecuteSingleFunctions, + IPollFunctions, + ITriggerFunctions, +} from 'n8n-core'; + +import { + IDataObject, +} from 'n8n-workflow'; + +export async function togglApiRequest(this: ITriggerFunctions | IPollFunctions | IHookFunctions | IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions, method: string, resource: string, body: any = {}, query?: IDataObject, uri?: string): Promise { // tslint:disable-line:no-any + const credentials = this.getCredentials('togglApi'); + if (credentials === undefined) { + throw new Error('No credentials got returned!'); + } + const headerWithAuthentication = Object.assign({}, + { Authorization: ` Basic ${Buffer.from(`${credentials.username}:${credentials.password}`).toString('base64')}` }); + + const options: OptionsWithUri = { + headers: headerWithAuthentication, + method, + qs: query, + uri: uri || `https://www.toggl.com/api/v8${resource}`, + body, + json: true + }; + if (Object.keys(options.body).length === 0) { + delete options.body; + } + try { + return await this.helpers.request!(options); + } catch (error) { + if (error.statusCode === 403) { + throw new Error('The Toggle credentials are probably invalid!'); + } + + const errorMessage = error.response.body && (error.response.body.message || error.response.body.Message); + if (errorMessage !== undefined) { + throw new Error(errorMessage); + } + + throw error; + } +} diff --git a/packages/nodes-base/nodes/Toggl/TogglTrigger.node.ts b/packages/nodes-base/nodes/Toggl/TogglTrigger.node.ts new file mode 100644 index 0000000000..6080f13421 --- /dev/null +++ b/packages/nodes-base/nodes/Toggl/TogglTrigger.node.ts @@ -0,0 +1,79 @@ +import { IPollFunctions } from 'n8n-core'; +import { + INodeExecutionData, + INodeType, + INodeTypeDescription, + IDataObject, +} from 'n8n-workflow'; + +import * as moment from 'moment'; +import { togglApiRequest } from './GenericFunctions'; + +export class TogglTrigger implements INodeType { + description: INodeTypeDescription = { + displayName: 'Toggl', + name: 'toggl', + icon: 'file:toggl.png', + group: ['trigger'], + version: 1, + description: 'Starts the workflow when Toggl events occure', + defaults: { + name: 'Toggl', + color: '#00FF00', + }, + credentials: [ + { + name: 'togglApi', + required: true, + } + ], + polling: true, + inputs: [], + outputs: ['main'], + properties: [ + { + displayName: 'Event', + name: 'event', + type: 'options', + options: [ + { + name: 'New Time Entry', + value: 'newTimeEntry', + } + ], + required: true, + default: 'newTimeEntry', + }, + ] + }; + + async poll(this: IPollFunctions): Promise { + const webhookData = this.getWorkflowStaticData('node'); + const event = this.getNodeParameter('event') as string; + let endpoint: string; + + if (event === 'newTimeEntry') { + endpoint = '/time_entries'; + } else { + throw new Error(`The defined event "${event}" is not supported`); + } + + const qs: IDataObject = {}; + let timeEntries = []; + qs.start_date = webhookData.lastTimeChecked; + qs.end_date = moment().format(); + + try { + timeEntries = await togglApiRequest.call(this, 'GET', endpoint, {}, qs); + webhookData.lastTimeChecked = qs.end_date; + } catch (err) { + throw new Error(`Toggl Trigger Error: ${err}`); + } + if (Array.isArray(timeEntries) && timeEntries.length !== 0) { + return [this.helpers.returnJsonArray(timeEntries)]; + } + + return null; + } + +} diff --git a/packages/nodes-base/nodes/Toggl/toggl.png b/packages/nodes-base/nodes/Toggl/toggl.png new file mode 100644 index 0000000000..06cbdb7caa Binary files /dev/null and b/packages/nodes-base/nodes/Toggl/toggl.png differ diff --git a/packages/nodes-base/package.json b/packages/nodes-base/package.json index 8930d1304c..6dfb99e72d 100644 --- a/packages/nodes-base/package.json +++ b/packages/nodes-base/package.json @@ -1,6 +1,6 @@ { "name": "n8n-nodes-base", - "version": "0.36.0", + "version": "0.37.0", "description": "Base nodes of n8n", "license": "SEE LICENSE IN LICENSE.md", "homepage": "https://n8n.io", @@ -75,6 +75,7 @@ "dist/credentials/TypeformApi.credentials.js", "dist/credentials/VeroApi.credentials.js", "dist/credentials/WordpressApi.credentials.js" + "dist/credentials/TogglApi.credentials.js", ], "nodes": [ "dist/nodes/ActiveCampaign/ActiveCampaign.node.js", @@ -144,6 +145,7 @@ "dist/nodes/RenameKeys.node.js", "dist/nodes/RssFeedRead.node.js", "dist/nodes/Set.node.js", + "dist/nodes/SseTrigger.node.js", "dist/nodes/SplitInBatches.node.js", "dist/nodes/Slack/Slack.node.js", "dist/nodes/SpreadsheetFile.node.js", @@ -158,6 +160,7 @@ "dist/nodes/Trello/TrelloTrigger.node.js", "dist/nodes/Twilio/Twilio.node.js", "dist/nodes/Typeform/TypeformTrigger.node.js", + "dist/nodes/Toggl/TogglTrigger.node.js", "dist/nodes/Vero/Vero.node.js", "dist/nodes/WriteBinaryFile.node.js", "dist/nodes/Webhook.node.js", @@ -170,6 +173,7 @@ "@types/basic-auth": "^1.1.2", "@types/cheerio": "^0.22.15", "@types/cron": "^1.6.1", + "@types/eventsource": "^1.1.2", "@types/express": "^4.16.1", "@types/gm": "^1.18.2", "@types/imap-simple": "^4.2.0", @@ -183,7 +187,7 @@ "@types/xml2js": "^0.4.3", "gulp": "^4.0.0", "jest": "^24.9.0", - "n8n-workflow": "~0.18.0", + "n8n-workflow": "~0.19.0", "ts-jest": "^24.0.2", "tslint": "^5.17.0", "typescript": "~3.7.4" @@ -192,7 +196,8 @@ "aws4": "^1.8.0", "basic-auth": "^2.0.1", "cheerio": "^1.0.0-rc.3", - "cron": "^1.6.0", + "cron": "^1.7.2", + "eventsource": "^1.0.7", "glob-promise": "^3.4.0", "gm": "^1.23.1", "googleapis": "^46.0.0", @@ -202,7 +207,7 @@ "lodash.unset": "^4.5.2", "mongodb": "^3.3.2", "mysql2": "^2.0.1", - "n8n-core": "~0.18.0", + "n8n-core": "~0.19.0", "nodemailer": "^5.1.1", "pdf-parse": "^1.1.1", "pg-promise": "^9.0.3", diff --git a/packages/workflow/package.json b/packages/workflow/package.json index 62f7bb446e..16b0d49956 100644 --- a/packages/workflow/package.json +++ b/packages/workflow/package.json @@ -1,6 +1,6 @@ { "name": "n8n-workflow", - "version": "0.18.0", + "version": "0.19.0", "description": "Workflow base code of n8n", "license": "SEE LICENSE IN LICENSE.md", "homepage": "https://n8n.io", diff --git a/packages/workflow/src/Interfaces.ts b/packages/workflow/src/Interfaces.ts index 7824ee52cb..be0330d810 100644 --- a/packages/workflow/src/Interfaces.ts +++ b/packages/workflow/src/Interfaces.ts @@ -107,6 +107,10 @@ export interface IDataObject { } +export interface IGetExecutePollFunctions { + (workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode): IPollFunctions; +} + export interface IGetExecuteTriggerFunctions { (workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode): ITriggerFunctions; } @@ -208,6 +212,19 @@ export interface IHookFunctions { }; } +export interface IPollFunctions { + __emit(data: INodeExecutionData[][]): void; + getCredentials(type: string): ICredentialDataDecryptedObject | undefined; + getMode(): WorkflowExecuteMode; + getNodeParameter(parameterName: string, fallbackValue?: any): NodeParameterValue | INodeParameters | NodeParameterValue[] | INodeParameters[] | object; //tslint:disable-line:no-any + getRestApiUrl(): string; + getTimezone(): string; + getWorkflowStaticData(type: string): IDataObject; + helpers: { + [key: string]: (...args: any[]) => any //tslint:disable-line:no-any + }; +} + export interface ITriggerFunctions { emit(data: INodeExecutionData[][]): void; getCredentials(type: string): ICredentialDataDecryptedObject | undefined; @@ -285,6 +302,7 @@ export interface INodeExecutionData { export interface INodeExecuteFunctions { + getExecutePollFunctions: IGetExecutePollFunctions; getExecuteTriggerFunctions: IGetExecuteTriggerFunctions; getExecuteFunctions: IGetExecuteFunctions; getExecuteSingleFunctions: IGetExecuteSingleFunctions; @@ -363,6 +381,10 @@ export interface IParameterDependencies { [key: string]: string[]; } +export interface IPollResponse { + closeFunction?: () => Promise; +} + export interface ITriggerResponse { closeFunction?: () => Promise; // To manually trigger the run @@ -376,6 +398,7 @@ export interface INodeType { description: INodeTypeDescription; execute?(this: IExecuteFunctions): Promise; executeSingle?(this: IExecuteSingleFunctions): Promise; + poll?(this: IPollFunctions): Promise; trigger?(this: ITriggerFunctions): Promise; webhook?(this: IWebhookFunctions): Promise; hooks?: { @@ -447,6 +470,7 @@ export interface INodeTypeDescription { properties: INodeProperties[]; credentials?: INodeCredentialDescription[]; maxNodes?: number; // How many nodes of that type can be created in a workflow + polling?: boolean; subtitle?: string; hooks?: { [key: string]: INodeHookDescription[] | undefined; diff --git a/packages/workflow/src/NodeHelpers.ts b/packages/workflow/src/NodeHelpers.ts index a9f22441da..a1b631a19f 100644 --- a/packages/workflow/src/NodeHelpers.ts +++ b/packages/workflow/src/NodeHelpers.ts @@ -23,6 +23,242 @@ import { import { get } from 'lodash'; + + +/** + * Gets special parameters which should be added to nodeTypes depending + * on their type or configuration + * + * @export + * @param {INodeType} nodeType + * @returns + */ +export function getSpecialNodeParameters(nodeType: INodeType) { + if (nodeType.description.polling === true) { + return [ + { + displayName: 'Poll Times', + name: 'pollTimes', + type: 'fixedCollection', + typeOptions: { + multipleValues: true, + multipleValueButtonText: 'Add Poll Time', + }, + default: {}, + description: 'Time at which polling should occur.', + placeholder: 'Add Poll Time', + options: [ + { + name: 'item', + displayName: 'Item', + values: [ + { + displayName: 'Mode', + name: 'mode', + type: 'options', + options: [ + { + name: 'Every Minute', + value: 'everyMinute', + }, + { + name: 'Every Hour', + value: 'everyHour', + }, + { + name: 'Every Day', + value: 'everyDay', + }, + { + name: 'Every Week', + value: 'everyWeek', + }, + { + name: 'Every Month', + value: 'everyMonth', + }, + { + name: 'Every X', + value: 'everyX', + }, + { + name: 'Custom', + value: 'custom', + }, + ], + default: 'everyDay', + description: 'How often to trigger.', + }, + { + displayName: 'Hour', + name: 'hour', + type: 'number', + typeOptions: { + minValue: 0, + maxValue: 23, + }, + displayOptions: { + hide: { + mode: [ + 'custom', + 'everyHour', + 'everyMinute', + 'everyX', + ], + }, + }, + default: 14, + description: 'The hour of the day to trigger (24h format).', + }, + { + displayName: 'Minute', + name: 'minute', + type: 'number', + typeOptions: { + minValue: 0, + maxValue: 59, + }, + displayOptions: { + hide: { + mode: [ + 'custom', + 'everyMinute', + 'everyX', + ], + }, + }, + default: 0, + description: 'The minute of the day to trigger.', + }, + { + displayName: 'Day of Month', + name: 'dayOfMonth', + type: 'number', + displayOptions: { + show: { + mode: [ + 'everyMonth', + ], + }, + }, + typeOptions: { + minValue: 1, + maxValue: 31, + }, + default: 1, + description: 'The day of the month to trigger.', + }, + { + displayName: 'Weekday', + name: 'weekday', + type: 'options', + displayOptions: { + show: { + mode: [ + 'everyWeek', + ], + }, + }, + options: [ + { + name: 'Monday', + value: '1', + }, + { + name: 'Tuesday', + value: '2', + }, + { + name: 'Wednesday', + value: '3', + }, + { + name: 'Thursday', + value: '4', + }, + { + name: 'Friday', + value: '5', + }, + { + name: 'Saturday', + value: '6', + }, + { + name: 'Sunday', + value: '0', + }, + ], + default: '1', + description: 'The weekday to trigger.', + }, + { + displayName: 'Cron Expression', + name: 'cronExpression', + type: 'string', + displayOptions: { + show: { + mode: [ + 'custom', + ], + }, + }, + default: '* * * * * *', + description: 'Use custom cron expression. Values and ranges as follows:
  • Seconds: 0-59
  • Minutes: 0 - 59
  • Hours: 0 - 23
  • Day of Month: 1 - 31
  • Months: 0 - 11 (Jan - Dec)
  • Day of Week: 0 - 6 (Sun - Sat)
', + }, + { + displayName: 'Value', + name: 'value', + type: 'number', + typeOptions: { + minValue: 0, + maxValue: 1000, + }, + displayOptions: { + show: { + mode: [ + 'everyX', + ], + }, + }, + default: 2, + description: 'All how many X minutes/hours it should trigger.', + }, + { + displayName: 'Unit', + name: 'unit', + type: 'options', + displayOptions: { + show: { + mode: [ + 'everyX', + ], + }, + }, + options: [ + { + name: 'Minutes', + value: 'minutes' + }, + { + name: 'Hours', + value: 'hours' + }, + ], + default: 'hours', + description: 'If it should trigger all X minutes or hours.', + }, + ], + }, + ], + }, + ]; + } + + return []; +} + + /** * Returns if the parameter should be displayed or not * diff --git a/packages/workflow/src/Workflow.ts b/packages/workflow/src/Workflow.ts index 9efaaa67ec..1ee7991391 100644 --- a/packages/workflow/src/Workflow.ts +++ b/packages/workflow/src/Workflow.ts @@ -3,27 +3,28 @@ import { IConnections, IGetExecuteTriggerFunctions, INode, - NodeHelpers, INodes, INodeExecuteFunctions, INodeExecutionData, - INodeParameters, INodeIssues, - NodeParameterValue, + INodeParameters, INodeType, INodeTypes, - ObservableObject, + IPollFunctions, IRunExecutionData, ITaskDataConnections, ITriggerResponse, IWebhookData, IWebhookResponseData, - WebhookSetupMethodNames, - WorkflowDataProxy, IWorfklowIssues, IWorkflowExecuteAdditionalData, - WorkflowExecuteMode, IWorkflowSettings, + NodeHelpers, + NodeParameterValue, + ObservableObject, + WebhookSetupMethodNames, + WorkflowDataProxy, + WorkflowExecuteMode, } from './'; // @ts-ignore @@ -188,7 +189,7 @@ export class Workflow { continue; } - if (nodeType.trigger !== undefined || nodeType.webhook !== undefined) { + if (nodeType.poll !== undefined || nodeType.trigger !== undefined || nodeType.webhook !== undefined) { // Is a trigger node. So workflow can be activated. return true; } @@ -289,6 +290,30 @@ export class Workflow { * @memberof Workflow */ getTriggerNodes(): INode[] { + return this.queryNodes((nodeType: INodeType) => !!nodeType.trigger ); + } + + + /** + * Returns all the poll nodes in the workflow + * + * @returns {INode[]} + * @memberof Workflow + */ + getPollNodes(): INode[] { + return this.queryNodes((nodeType: INodeType) => !!nodeType.poll ); + } + + + /** + * Returns all the nodes in the workflow for which the given + * checkFunction return true + * + * @param {(nodeType: INodeType) => boolean} checkFunction + * @returns {INode[]} + * @memberof Workflow + */ + queryNodes(checkFunction: (nodeType: INodeType) => boolean): INode[] { const returnNodes: INode[] = []; // Check if it has any of them @@ -304,7 +329,7 @@ export class Workflow { nodeType = this.nodeTypes.getByName(node.type); - if (nodeType !== undefined && nodeType.trigger) { + if (nodeType !== undefined && checkFunction(nodeType)) { returnNodes.push(node); } } @@ -729,14 +754,14 @@ export class Workflow { // Check which node to return as start node - // Check if there are any trigger nodes and then return the first one + // Check if there are any trigger or poll nodes and then return the first one let node: INode; let nodeType: INodeType; for (const nodeName of nodeNames) { node = this.nodes[nodeName]; nodeType = this.nodeTypes.getByName(node.type) as INodeType; - if (nodeType.trigger !== undefined) { + if (nodeType.trigger !== undefined || nodeType.poll !== undefined) { return node; } } @@ -994,6 +1019,30 @@ export class Workflow { } + /** + * Runs the given trigger node so that it can trigger the workflow + * when the node has data. + * + * @param {INode} node + * @param {IPollFunctions} pollFunctions + * @returns + * @memberof Workflow + */ + async runPoll(node: INode, pollFunctions: IPollFunctions): Promise { + const nodeType = this.nodeTypes.getByName(node.type); + + if (nodeType === undefined) { + throw new Error(`The node type "${node.type}" of node "${node.name}" is not known.`); + } + + if (!nodeType.poll) { + throw new Error(`The node type "${node.type}" of node "${node.name}" does not have a poll function defined.`); + } + + return nodeType.poll!.call(pollFunctions); + } + + /** * Executes the webhook data to see what it should return and if the * workflow should be started or not @@ -1096,6 +1145,9 @@ export class Workflow { } else if (nodeType.execute) { const thisArgs = nodeExecuteFunctions.getExecuteFunctions(this, runExecutionData, runIndex, connectionInputData, inputData, node, additionalData, mode); return nodeType.execute.call(thisArgs); + } else if (nodeType.poll) { + const thisArgs = nodeExecuteFunctions.getExecutePollFunctions(this, node, additionalData, mode); + return nodeType.poll.call(thisArgs); } else if (nodeType.trigger) { if (mode === 'manual') { // In manual mode start the trigger