From 42758988f2fc2c484906e21fe79668963970627d Mon Sep 17 00:00:00 2001 From: ricardo Date: Fri, 6 Nov 2020 20:18:10 -0500 Subject: [PATCH 1/4] :sparkles: Airtable Trigger --- .../nodes/Airtable/AirtableTrigger.node.ts | 148 ++++++++++++++++++ .../nodes/Airtable/GenericFunctions.ts | 6 +- packages/nodes-base/package.json | 1 + 3 files changed, 152 insertions(+), 3 deletions(-) create mode 100644 packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts diff --git a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts new file mode 100644 index 0000000000..b0136649e0 --- /dev/null +++ b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts @@ -0,0 +1,148 @@ +import { + IPollFunctions, +} from 'n8n-core'; + +import { + IDataObject, + INodeExecutionData, + INodeType, + INodeTypeDescription, +} from 'n8n-workflow'; + +import { + apiRequestAllItems, +} from './GenericFunctions'; + +import * as moment from 'moment'; + +export class AirtableTrigger implements INodeType { + description: INodeTypeDescription = { + displayName: 'Airtable Trigger', + name: 'airtableTrigger', + icon: 'file:airtable.png', + group: ['trigger'], + version: 1, + description: 'Starts the workflow when Airtable events occur', + subtitle: '={{$parameter["event"]}}', + defaults: { + name: 'Airtable Trigger', + color: '#445599', + }, + credentials: [ + { + name: 'airtableApi', + required: true, + }, + ], + polling: true, + inputs: [], + outputs: ['main'], + properties: [ + { + displayName: 'Base ID', + name: 'baseId', + type: 'string', + default: '', + required: true, + }, + { + displayName: 'Table ID', + name: 'tableId', + type: 'string', + default: '', + required: true, + }, { + displayName: 'Trigger Field', + name: 'triggerField', + type: 'string', + default: '', + description: `A Created Time or Last Modified Time field that will be used to sort records.
+ If you do not have a Created Time or Last Modified Time field in your schema, please create one,
+ because without this field trigger will not work correctly.`, + required: true, + }, + { + displayName: 'Additional Fields', + name: 'additionalFields', + type: 'collection', + placeholder: 'Add Field', + default: {}, + options: [ + { + displayName: 'Fields', + name: 'fields', + type: 'string', + default: '', + description: `Fields to be included in the response.
+ Multiple ones can be set separated by comma. Example: name,id.
+ By default just the trigger field will be included.`, + }, + { + displayName: 'Formula', + name: 'formula', + type: 'string', + default: '', + description: '', + }, + { + displayName: 'View ID', + name: 'viewId', + type: 'string', + default: '', + description: '', + }, + ], + }, + ], + }; + + async poll(this: IPollFunctions): Promise { + + const webhookData = this.getWorkflowStaticData('node'); + + const qs: IDataObject = {}; + + const additionalFields = this.getNodeParameter('additionalFields') as IDataObject; + + const base = this.getNodeParameter('baseId') as string; + + const table = this.getNodeParameter('tableId') as string; + + const triggerField = this.getNodeParameter('triggerField') as string; + + const endpoint = `${base}/${table}`; + + const now = moment().utc().format(); + + const startDate = webhookData.lastTimeChecked as string || now; + + const endDate = now; + + qs['fields[]'] = [triggerField]; + + if (additionalFields.viewId) { + qs.view = additionalFields.viewId; + } + + if (additionalFields.fields) { + qs['fields[]'] = (additionalFields.fields as string).split(','); + } + + qs.filterByFormula = `IS_AFTER({${triggerField}}, DATETIME_PARSE("${startDate}", "YYYY-MM-DD HH:mm:ss"))`; + + if (additionalFields.formula) { + qs.filterByFormula = `AND(${qs.filterByFormula}, ${additionalFields.formula})`; + } + + const { records } = await apiRequestAllItems.call(this, 'GET', endpoint, {}, qs); + + webhookData.lastTimeChecked = endDate; + + if (Array.isArray(records) && records.length) { + + return [this.helpers.returnJsonArray(records)]; + } + + return null; + } +} diff --git a/packages/nodes-base/nodes/Airtable/GenericFunctions.ts b/packages/nodes-base/nodes/Airtable/GenericFunctions.ts index dd9c0aab91..e5d53567e9 100644 --- a/packages/nodes-base/nodes/Airtable/GenericFunctions.ts +++ b/packages/nodes-base/nodes/Airtable/GenericFunctions.ts @@ -5,7 +5,7 @@ import { } from 'n8n-core'; import { OptionsWithUri } from 'request'; -import { IDataObject } from 'n8n-workflow'; +import { IDataObject, IPollFunctions } from 'n8n-workflow'; /** @@ -17,7 +17,7 @@ import { IDataObject } from 'n8n-workflow'; * @param {object} body * @returns {Promise} */ -export async function apiRequest(this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions, method: string, endpoint: string, body: object, query?: IDataObject): Promise { // tslint:disable-line:no-any +export async function apiRequest(this: IHookFunctions | IExecuteFunctions | ILoadOptionsFunctions | IPollFunctions, method: string, endpoint: string, body: object, query?: IDataObject): Promise { // tslint:disable-line:no-any const credentials = this.getCredentials('airtableApi'); if (credentials === undefined) { @@ -77,7 +77,7 @@ export async function apiRequest(this: IHookFunctions | IExecuteFunctions | ILoa * @param {IDataObject} [query] * @returns {Promise} */ -export async function apiRequestAllItems(this: IHookFunctions | IExecuteFunctions, method: string, endpoint: string, body: IDataObject, query?: IDataObject): Promise { // tslint:disable-line:no-any +export async function apiRequestAllItems(this: IHookFunctions | IExecuteFunctions | IPollFunctions, method: string, endpoint: string, body: IDataObject, query?: IDataObject): Promise { // tslint:disable-line:no-any if (query === undefined) { query = {}; diff --git a/packages/nodes-base/package.json b/packages/nodes-base/package.json index 29a0b90bb7..5e7fd2b583 100644 --- a/packages/nodes-base/package.json +++ b/packages/nodes-base/package.json @@ -215,6 +215,7 @@ "dist/nodes/ActiveCampaign/ActiveCampaignTrigger.node.js", "dist/nodes/AgileCrm/AgileCrm.node.js", "dist/nodes/Airtable/Airtable.node.js", + "dist/nodes/Airtable/AirtableTrigger.node.js", "dist/nodes/AcuityScheduling/AcuitySchedulingTrigger.node.js", "dist/nodes/Amqp/Amqp.node.js", "dist/nodes/Amqp/AmqpTrigger.node.js", From 32cd54e705e6d8ed6d31fbba56888b820e6449c6 Mon Sep 17 00:00:00 2001 From: ricardo Date: Tue, 10 Nov 2020 16:40:44 -0500 Subject: [PATCH 2/4] :zap: Small improvements --- packages/nodes-base/nodes/Airtable/Airtable.node.ts | 4 ++-- packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts | 7 +++++-- 2 files changed, 7 insertions(+), 4 deletions(-) diff --git a/packages/nodes-base/nodes/Airtable/Airtable.node.ts b/packages/nodes-base/nodes/Airtable/Airtable.node.ts index 98d499f72d..68ce62d937 100644 --- a/packages/nodes-base/nodes/Airtable/Airtable.node.ts +++ b/packages/nodes-base/nodes/Airtable/Airtable.node.ts @@ -73,12 +73,12 @@ export class Airtable implements INodeType { // All // ---------------------------------- { - displayName: 'Application ID', + displayName: 'Base ID', name: 'application', type: 'string', default: '', required: true, - description: 'The ID of the application to access.', + description: 'The ID of the base to access.', }, { displayName: 'Table', diff --git a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts index b0136649e0..fe18dc6ada 100644 --- a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts +++ b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts @@ -44,14 +44,17 @@ export class AirtableTrigger implements INodeType { type: 'string', default: '', required: true, + description: 'The ID of this base.', }, { - displayName: 'Table ID', + displayName: 'Table', name: 'tableId', type: 'string', default: '', + description: 'The name of table to access.', required: true, - }, { + }, + { displayName: 'Trigger Field', name: 'triggerField', type: 'string', From fe97bf6619bdf312cc536a404eeb7b2e96c1d61e Mon Sep 17 00:00:00 2001 From: ricardo Date: Sat, 14 Nov 2020 20:21:41 -0500 Subject: [PATCH 3/4] :zap: Return last record when executing manually --- packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts index fe18dc6ada..ba2e9aabae 100644 --- a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts +++ b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts @@ -137,6 +137,11 @@ export class AirtableTrigger implements INodeType { qs.filterByFormula = `AND(${qs.filterByFormula}, ${additionalFields.formula})`; } + if (this.getMode() === 'manual') { + delete qs.filterByFormula; + qs.maxRecords = 1; + } + const { records } = await apiRequestAllItems.call(this, 'GET', endpoint, {}, qs); webhookData.lastTimeChecked = endDate; From 0e03ab7e79947369ab3e621b71ad52b9ae5a0efe Mon Sep 17 00:00:00 2001 From: ricardo Date: Thu, 19 Nov 2020 18:44:21 -0500 Subject: [PATCH 4/4] :zap: Return all fields by default --- packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts index ba2e9aabae..0340e5fcac 100644 --- a/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts +++ b/packages/nodes-base/nodes/Airtable/AirtableTrigger.node.ts @@ -121,8 +121,6 @@ export class AirtableTrigger implements INodeType { const endDate = now; - qs['fields[]'] = [triggerField]; - if (additionalFields.viewId) { qs.view = additionalFields.viewId; }