From 128c3b83dfff3a3c21b2169da010698a3f4d20de Mon Sep 17 00:00:00 2001
From: agobrech <45268029+agobrech@users.noreply.github.com>
Date: Tue, 18 Oct 2022 13:59:17 +0200
Subject: [PATCH] feat(Node): add the Scheduler Node (#4223)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* ✨ Create Schedule node with MVP structure
* ✨ Add 24 increments for hours picker
* 🚨 Lintfix
* Add timestamp, add hour minute and cron expression
* Fix bug where there was one extra interval object
* Fix default value from fixedCollection
* 🐛 UI fixes
* 🎨 Changed logic to reflect UI fixes
* Fix auto intitialising
* Deprecated interval and cron in favor of schedule node
* 🐛 Ui fixes
* 🐛 Fix issue with week intervals
* 🚨 Lint fixes
* change order of days in the week to chronological order
---
packages/nodes-base/nodes/Cron/Cron.node.json | 2 +-
packages/nodes-base/nodes/Cron/Cron.node.ts | 5 +-
.../nodes/Interval/Interval.node.ts | 1 +
.../nodes/Schedule/CronInterface.ts | 9 +
.../nodes/Schedule/ScheduleTrigger.node.json | 18 +
.../nodes/Schedule/ScheduleTrigger.node.ts | 520 ++++++++++++++++++
.../nodes-base/nodes/Schedule/schedule.svg | 1 +
packages/nodes-base/package.json | 1 +
8 files changed, 555 insertions(+), 2 deletions(-)
create mode 100644 packages/nodes-base/nodes/Schedule/CronInterface.ts
create mode 100644 packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.json
create mode 100644 packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.ts
create mode 100644 packages/nodes-base/nodes/Schedule/schedule.svg
diff --git a/packages/nodes-base/nodes/Cron/Cron.node.json b/packages/nodes-base/nodes/Cron/Cron.node.json
index f6ef185226..fc761179ee 100644
--- a/packages/nodes-base/nodes/Cron/Cron.node.json
+++ b/packages/nodes-base/nodes/Cron/Cron.node.json
@@ -128,7 +128,7 @@
}
]
},
- "alias": ["Time", "Scheduler", "Polling"],
+ "alias": ["Time", "Scheduler", "Polling", "Cron", "Interval"],
"subcategories": {
"Core Nodes": ["Flow"]
}
diff --git a/packages/nodes-base/nodes/Cron/Cron.node.ts b/packages/nodes-base/nodes/Cron/Cron.node.ts
index 718d15f8f0..4607b1b3f6 100644
--- a/packages/nodes-base/nodes/Cron/Cron.node.ts
+++ b/packages/nodes-base/nodes/Cron/Cron.node.ts
@@ -17,6 +17,7 @@ export class Cron implements INodeType {
icon: 'fa:calendar',
group: ['trigger', 'schedule'],
version: 1,
+ hidden: true,
description: 'Triggers the workflow at a specific time',
eventTriggerDescription: '',
activationMessage:
@@ -69,7 +70,9 @@ export class Cron implements INodeType {
const timezone = this.getTimezone();
// Start the cron-jobs
- const cronJobs = cronTimes.map(cronTime => new CronJob(cronTime, executeTrigger, undefined, true, timezone));
+ const cronJobs = cronTimes.map(
+ (cronTime) => new CronJob(cronTime, executeTrigger, undefined, true, timezone),
+ );
// Stop the cron-jobs
async function closeFunction() {
diff --git a/packages/nodes-base/nodes/Interval/Interval.node.ts b/packages/nodes-base/nodes/Interval/Interval.node.ts
index cf36eb15af..57693b38fc 100644
--- a/packages/nodes-base/nodes/Interval/Interval.node.ts
+++ b/packages/nodes-base/nodes/Interval/Interval.node.ts
@@ -13,6 +13,7 @@ export class Interval implements INodeType {
icon: 'fa:hourglass',
group: ['trigger', 'schedule'],
version: 1,
+ hidden: true,
description: 'Triggers the workflow in a given interval',
eventTriggerDescription: '',
activationMessage:
diff --git a/packages/nodes-base/nodes/Schedule/CronInterface.ts b/packages/nodes-base/nodes/Schedule/CronInterface.ts
new file mode 100644
index 0000000000..22b6473cd1
--- /dev/null
+++ b/packages/nodes-base/nodes/Schedule/CronInterface.ts
@@ -0,0 +1,9 @@
+import { IDataObject } from 'n8n-workflow';
+
+export type ICronExpression = [
+ string | Date,
+ string | Date,
+ string | Date,
+ string | Date,
+ string | Date,
+];
diff --git a/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.json b/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.json
new file mode 100644
index 0000000000..fcaab357f9
--- /dev/null
+++ b/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.json
@@ -0,0 +1,18 @@
+{
+ "node": "n8n-nodes-base.scheduleTrigger",
+ "nodeVersion": "1.0",
+ "codexVersion": "1.0",
+ "categories": ["Core Nodes"],
+ "resources": {
+ "primaryDocumentation": [
+ {
+ "url": "https://docs.n8n.io/nodes/n8n-nodes-base.scheduleTrigger/"
+ }
+ ],
+ "generic": []
+ },
+ "alias": ["Time", "Scheduler", "Polling", "Cron", "Interval"],
+ "subcategories": {
+ "Core Nodes": ["Flow"]
+ }
+}
diff --git a/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.ts b/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.ts
new file mode 100644
index 0000000000..34833f2e8b
--- /dev/null
+++ b/packages/nodes-base/nodes/Schedule/ScheduleTrigger.node.ts
@@ -0,0 +1,520 @@
+import { ITriggerFunctions } from 'n8n-core';
+import {
+ IDataObject,
+ INodeType,
+ INodeTypeDescription,
+ ITriggerResponse,
+ NodeOperationError,
+} from 'n8n-workflow';
+
+import { CronJob } from 'cron';
+import { ICronExpression } from './CronInterface';
+import moment from 'moment';
+
+export class ScheduleTrigger implements INodeType {
+ description: INodeTypeDescription = {
+ displayName: 'Schedule Trigger',
+ name: 'scheduleTrigger',
+ icon: 'file:schedule.svg',
+ group: ['trigger', 'schedule'],
+ version: 1,
+ description: 'Triggers the workflow on a given schedule',
+ eventTriggerDescription: '',
+ activationMessage:
+ 'Your schedule trigger will now trigger executions on the schedule you have defined.',
+ defaults: {
+ name: 'Schedule Trigger',
+ color: '#00FF00',
+ },
+ // eslint-disable-next-line n8n-nodes-base/node-class-description-inputs-wrong-regular-node
+ inputs: [],
+ outputs: ['main'],
+ properties: [
+ {
+ displayName:
+ 'This workflow will run on the schedule you define here once you activate it.
For testing, you can also trigger it manually: by going back to the canvas and clicking ‘execute workflow’',
+ name: 'notice',
+ type: 'notice',
+ default: '',
+ },
+ {
+ displayName: 'Trigger Rules',
+ name: 'rule',
+ placeholder: 'Add Rule',
+ type: 'fixedCollection',
+ typeOptions: {
+ multipleValues: true,
+ },
+ default: {
+ interval: [
+ {
+ field: 'days',
+ },
+ ],
+ },
+ options: [
+ {
+ name: 'interval',
+ displayName: 'Trigger Interval',
+ values: [
+ {
+ displayName: 'Trigger Interval',
+ name: 'field',
+ type: 'options',
+ default: 'days',
+ options: [
+ {
+ name: 'Custom (Cron)',
+ value: 'cronExpression',
+ },
+ {
+ name: 'Days',
+ value: 'days',
+ },
+ {
+ name: 'Hours',
+ value: 'hours',
+ },
+ {
+ name: 'Minutes',
+ value: 'minutes',
+ },
+ {
+ name: 'Months',
+ value: 'months',
+ },
+ {
+ name: 'Seconds',
+ value: 'seconds',
+ },
+ {
+ name: 'Weeks',
+ value: 'weeks',
+ },
+ ],
+ },
+ {
+ displayName: 'Seconds Between Triggers',
+ name: 'secondsInterval',
+ type: 'number',
+ default: 30,
+ displayOptions: {
+ show: {
+ field: ['seconds'],
+ },
+ },
+ description: 'Number of seconds between each workflow trigger',
+ },
+ {
+ displayName: 'Minutes Between Triggers',
+ name: 'minutesInterval',
+ type: 'number',
+ default: 5,
+ displayOptions: {
+ show: {
+ field: ['minutes'],
+ },
+ },
+ description: 'Number of minutes between each workflow trigger',
+ },
+ {
+ displayName: 'Hours Between Triggers',
+ name: 'hoursInterval',
+ type: 'number',
+ displayOptions: {
+ show: {
+ field: ['hours'],
+ },
+ },
+ default: 1,
+ description: 'Number of hours between each workflow trigger',
+ },
+ {
+ displayName: 'Days Between Triggers',
+ name: 'daysInterval',
+ type: 'number',
+ displayOptions: {
+ show: {
+ field: ['days'],
+ },
+ },
+ default: 1,
+ description: 'Number of days between each workflow trigger',
+ },
+ {
+ displayName: 'Weeks Between Triggers',
+ name: 'weeksInterval',
+ type: 'number',
+ displayOptions: {
+ show: {
+ field: ['weeks'],
+ },
+ },
+ default: 1,
+ description: 'Would run every week unless specified otherwise',
+ },
+ {
+ displayName: 'Months Between Triggers',
+ name: 'monthsInterval',
+ type: 'number',
+ displayOptions: {
+ show: {
+ field: ['months'],
+ },
+ },
+ default: 1,
+ description: 'Would run every month unless specified otherwise',
+ },
+ {
+ displayName: 'Trigger at Day of Month',
+ name: 'triggerAtDayOfMonth',
+ type: 'number',
+ displayOptions: {
+ show: {
+ field: ['months'],
+ },
+ },
+ typeOptions: {
+ minValue: 1,
+ maxValue: 31,
+ },
+ default: 1,
+ description: 'The day of the month to trigger (1-31)',
+ hint: 'If a month doesn’t have this day, the node won’t trigger',
+ },
+ {
+ displayName: 'Trigger on Weekdays',
+ name: 'triggerAtDay',
+ type: 'multiOptions',
+ displayOptions: {
+ show: {
+ field: ['weeks'],
+ },
+ },
+ typeOptions: {
+ maxValue: 7,
+ },
+ 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: 7,
+ },
+ ],
+ default: [7],
+ },
+ {
+ displayName: 'Trigger at Hour',
+ name: 'triggerAtHour',
+ type: 'options',
+ default: 0,
+ displayOptions: {
+ show: {
+ field: ['days', 'weeks', 'months'],
+ },
+ },
+ options: [
+ {
+ name: 'Midnight',
+ displayName: 'Midnight',
+ value: 0,
+ },
+ {
+ name: '1am',
+ displayName: '1am',
+ value: 1,
+ },
+ {
+ name: '2am',
+ displayName: '2am',
+ value: 2,
+ },
+ {
+ name: '3am',
+ displayName: '3am',
+ value: 3,
+ },
+ {
+ name: '4am',
+ displayName: '4am',
+ value: 4,
+ },
+ {
+ name: '5am',
+ displayName: '5am',
+ value: 5,
+ },
+ {
+ name: '6am',
+ displayName: '6am',
+ value: 6,
+ },
+ {
+ name: '7am',
+ displayName: '7am',
+ value: 7,
+ },
+ {
+ name: '8am',
+ displayName: '8am',
+ value: 8,
+ },
+ {
+ name: '9am',
+ displayName: '9am',
+ value: 9,
+ },
+ {
+ name: '10am',
+ displayName: '10am',
+ value: 10,
+ },
+ {
+ name: '11am',
+ displayName: '11am',
+ value: 11,
+ },
+ {
+ name: 'Noon',
+ displayName: 'Noon',
+ value: 12,
+ },
+ {
+ name: '1pm',
+ displayName: '1pm',
+ value: 13,
+ },
+ {
+ name: '2pm',
+ displayName: '2pm',
+ value: 14,
+ },
+ {
+ name: '3pm',
+ displayName: '3pm',
+ value: 15,
+ },
+ {
+ name: '4pm',
+ displayName: '4pm',
+ value: 16,
+ },
+ {
+ name: '5pm',
+ displayName: '5pm',
+ value: 17,
+ },
+ {
+ name: '6pm',
+ displayName: '6pm',
+ value: 18,
+ },
+ {
+ name: '7pm',
+ displayName: '7pm',
+ value: 19,
+ },
+ {
+ name: '8pm',
+ displayName: '8pm',
+ value: 20,
+ },
+ {
+ name: '9pm',
+ displayName: '9pm',
+ value: 21,
+ },
+ {
+ name: '10pm',
+ displayName: '10pm',
+ value: 22,
+ },
+ {
+ name: '11pm',
+ displayName: '11pm',
+ value: 23,
+ },
+ ],
+ description: 'The hour of the day to trigger',
+ },
+ {
+ displayName: 'Trigger at Minute',
+ name: 'triggerAtMinute',
+ type: 'number',
+ default: 0,
+ displayOptions: {
+ show: {
+ field: ['hours', 'days', 'weeks', 'months'],
+ },
+ },
+ typeOptions: {
+ minValue: 0,
+ maxValue: 59,
+ },
+ description: 'The minute past the hour to trigger (0-59)',
+ },
+ {
+ displayName:
+ 'You can find help generating your cron expression here',
+ name: 'notice',
+ type: 'notice',
+ displayOptions: {
+ show: {
+ field: ['cronExpression'],
+ },
+ },
+ default: '',
+ },
+ {
+ displayName: 'Expression',
+ name: 'expression',
+ type: 'string',
+ default: '',
+ placeholder: 'eg. 0 15 * 1 sun',
+ displayOptions: {
+ show: {
+ field: ['cronExpression'],
+ },
+ },
+ hint: 'Format: [Minute] [Hour] [Day of Month] [Month] [Day of Week]',
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ };
+
+ async trigger(this: ITriggerFunctions): Promise {
+ const rule = this.getNodeParameter('rule', []) as IDataObject;
+ const interval = rule.interval as IDataObject[];
+ const timezone = this.getTimezone();
+ const date = moment.tz(timezone).week();
+ const cronJobs: CronJob[] = [];
+ let intervalObj: NodeJS.Timeout;
+ const executeTrigger = () => {
+ const resultData = {
+ timestamp: moment.tz(timezone).toISOString(true),
+ 'Readable date': moment.tz(timezone).format('MMMM Do YYYY, h:mm:ss a'),
+ 'Readable time': moment.tz(timezone).format('h:mm:ss a'),
+ 'Day of week': moment.tz(timezone).format('dddd'),
+ Year: moment.tz(timezone).format('YYYY'),
+ Month: moment.tz(timezone).format('MMMM'),
+ 'Day of month': moment.tz(timezone).format('DD'),
+ Hour: moment.tz(timezone).format('HH'),
+ Minute: moment.tz(timezone).format('mm'),
+ Second: moment.tz(timezone).format('ss'),
+ Timezone: moment.tz(timezone).format('z Z'),
+ };
+ this.emit([this.helpers.returnJsonArray([resultData])]);
+ };
+ for (let i = 0; i < interval.length; i++) {
+ let intervalValue = 1000;
+ if (interval[i].field === 'cronExpression') {
+ const cronExpression = interval[i].expression as string;
+ try {
+ const cronJob = new CronJob(cronExpression, executeTrigger, undefined, true, timezone);
+ cronJobs.push(cronJob);
+ } catch (error) {
+ throw new NodeOperationError(this.getNode(), 'Invalid cron expression', {
+ description: 'More information on how to build them at http://www.cronmaker.com',
+ });
+ }
+ }
+
+ if (interval[i].field === 'seconds') {
+ const seconds = interval[i].secondsInterval as number;
+ intervalValue *= seconds;
+ intervalObj = setInterval(executeTrigger, intervalValue);
+ }
+
+ if (interval[i].field === 'minutes') {
+ const minutes = interval[i].minutesInterval as number;
+ intervalValue *= 60 * minutes;
+ intervalObj = setInterval(executeTrigger, intervalValue);
+ }
+
+ if (interval[i].field === 'hours') {
+ const hour = interval[i].triggerAtHour?.toString() as string;
+ const minute = interval[i].triggerAtMinute?.toString() as string;
+ const week = interval[i].triggerAtWeek as number;
+ const cronTimes: ICronExpression = [minute, hour, `*/${week * 7}`, '*', '*'];
+ const cronExpression = cronTimes.join(' ');
+ const cronJob = new CronJob(cronExpression, executeTrigger, undefined, true, timezone);
+ cronJobs.push(cronJob);
+ }
+
+ if (interval[i].field === 'days') {
+ const day = interval[i].daysInterval?.toString() as string;
+ const hour = interval[i].triggerAtHour?.toString() as string;
+ const minute = interval[i].triggerAtMinute?.toString() as string;
+ const cronTimes: ICronExpression = [minute, hour, `*/${day}`, '*', '*'];
+ const cronExpression: string = cronTimes.join(' ');
+ const cronJob = new CronJob(cronExpression, executeTrigger, undefined, true, timezone);
+ cronJobs.push(cronJob);
+ }
+
+ if (interval[i].field === 'weeks') {
+ const days = interval[i].triggerAtDay as IDataObject[];
+ const day = days.join(',') as string;
+ const hour = interval[i].triggerAtHour?.toString() as string;
+ const minute = interval[i].triggerAtMinute?.toString() as string;
+ const cronTimes: ICronExpression = [minute, hour, '*', '*', day];
+ const cronExpression: string = cronTimes.join(' ');
+ const cronJob = new CronJob(cronExpression, executeTrigger, undefined, true, timezone);
+ cronJobs.push(cronJob);
+ }
+
+ if (interval[i].field === 'months') {
+ const month = interval[i].monthsInterval?.toString() as string;
+ const day = interval[i].triggerAtDayOfMonth?.toString() as string;
+ const hour = interval[i].triggerAtHour?.toString() as string;
+ const minute = interval[i].triggerAtMinute?.toString() as string;
+ const cronTimes: ICronExpression = [minute, hour, day, `*/${month}`, '*'];
+ const cronExpression: string = cronTimes.join(' ');
+ const cronJob = new CronJob(cronExpression, executeTrigger, undefined, true, timezone);
+ cronJobs.push(cronJob);
+ }
+ }
+
+ async function closeFunction() {
+ for (const cronJob of cronJobs) {
+ cronJob.stop();
+ }
+ clearInterval(intervalObj);
+ }
+
+ async function manualTriggerFunction() {
+ executeTrigger();
+ }
+
+ return {
+ closeFunction,
+ manualTriggerFunction,
+ };
+ }
+}
diff --git a/packages/nodes-base/nodes/Schedule/schedule.svg b/packages/nodes-base/nodes/Schedule/schedule.svg
new file mode 100644
index 0000000000..b496e09d6b
--- /dev/null
+++ b/packages/nodes-base/nodes/Schedule/schedule.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/packages/nodes-base/package.json b/packages/nodes-base/package.json
index 9dfc6ac9ce..dc91d5d6c7 100644
--- a/packages/nodes-base/package.json
+++ b/packages/nodes-base/package.json
@@ -630,6 +630,7 @@
"dist/nodes/S3/S3.node.js",
"dist/nodes/Salesforce/Salesforce.node.js",
"dist/nodes/Salesmate/Salesmate.node.js",
+ "dist/nodes/Schedule/ScheduleTrigger.node.js",
"dist/nodes/SeaTable/SeaTable.node.js",
"dist/nodes/SeaTable/SeaTableTrigger.node.js",
"dist/nodes/SecurityScorecard/SecurityScorecard.node.js",