mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-14 00:24:07 -08:00
88dea330b9
* ⚡ Update `lintfix` script * ⚡ Run baseline `lintfix` * 🔥 Remove unneeded exceptions (#3538) * 🔥 Remove exceptions for `node-param-default-wrong-for-simplify` * 🔥 Remove exceptions for `node-param-placeholder-miscased-id` * ⚡ Update version * 👕 Apply `node-param-placeholder-missing` (#3542) * 👕 Apply `filesystem-wrong-cred-filename` (#3543) * 👕 Apply `node-param-description-missing-from-dynamic-options` (#3545) Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * 👕 Apply `node-class-description-empty-string` (#3546) * 👕 Apply `node-class-description-icon-not-svg` (#3548) * 👕 Apply `filesystem-wrong-node-filename` (#3549) Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * 👕 Expand lintings to credentials (#3550) * 👕 Apply `node-param-multi-options-type-unsorted-items` (#3552) * ⚡ fix * ⚡ Minor fixes Co-authored-by: Michael Kret <michael.k@radency.com> * 👕 Apply `node-param-description-wrong-for-dynamic-multi-options` (#3541) * ⚡ Add new lint rule, node-param-description-wrong-for-dynamic-multi-options * ⚡ Fix with updated linting rules * ⚡ Minor fixes Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * 👕 Apply `node-param-description-boolean-without-whether` (#3553) * ⚡ fix * Update packages/nodes-base/nodes/Clockify/ProjectDescription.ts Co-authored-by: Iván Ovejero <ivov.src@gmail.com> * 👕 Apply node-param-display-name-wrong-for-dynamic-multi-options (#3537) * 👕 Add exceptions * 👕 Add exception * ✏️ Alphabetize rules * ⚡ Restore `lintfix` command Co-authored-by: agobrech <45268029+agobrech@users.noreply.github.com> Co-authored-by: Omar Ajoue <krynble@gmail.com> Co-authored-by: Michael Kret <michael.k@radency.com> Co-authored-by: brianinoa <54530642+brianinoa@users.noreply.github.com> Co-authored-by: Michael Kret <88898367+michael-radency@users.noreply.github.com>
215 lines
5.5 KiB
TypeScript
215 lines
5.5 KiB
TypeScript
|
|
import {
|
|
IDataObject,
|
|
ILoadOptionsFunctions,
|
|
INodeExecutionData,
|
|
INodePropertyOptions,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
IPollFunctions,
|
|
NodeApiError,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
googleApiRequest,
|
|
googleApiRequestAllItems,
|
|
} from './GenericFunctions';
|
|
|
|
import moment from 'moment';
|
|
|
|
export class GoogleCalendarTrigger implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Google Calendar Trigger',
|
|
name: 'googleCalendarTrigger',
|
|
icon: 'file:googleCalendar.svg',
|
|
group: ['trigger'],
|
|
version: 1,
|
|
subtitle: '={{$parameter["triggerOn"]}}',
|
|
description: 'Starts the workflow when Google Calendar events occur',
|
|
defaults: {
|
|
name: 'Google Calendar Trigger',
|
|
},
|
|
inputs: [],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
{
|
|
name: 'googleCalendarOAuth2Api',
|
|
required: true,
|
|
},
|
|
],
|
|
polling: true,
|
|
properties: [
|
|
{
|
|
displayName: 'Calendar Name or ID',
|
|
name: 'calendarId',
|
|
type: 'options',
|
|
description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/nodes/expressions.html#expressions">expression</a>',
|
|
required: true,
|
|
typeOptions: {
|
|
loadOptionsMethod: 'getCalendars',
|
|
},
|
|
default: '',
|
|
},
|
|
{
|
|
displayName: 'Trigger On',
|
|
name: 'triggerOn',
|
|
type: 'options',
|
|
required: true,
|
|
default: '',
|
|
options: [
|
|
{
|
|
name: 'Event Created',
|
|
value: 'eventCreated',
|
|
},
|
|
{
|
|
name: 'Event Ended',
|
|
value: 'eventEnded',
|
|
},
|
|
{
|
|
name: 'Event Started',
|
|
value: 'eventStarted',
|
|
},
|
|
{
|
|
name: 'Event Updated',
|
|
value: 'eventUpdated',
|
|
},
|
|
],
|
|
},
|
|
{
|
|
displayName: 'Options',
|
|
name: 'options',
|
|
type: 'collection',
|
|
placeholder: 'Add Option',
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Match Term',
|
|
name: 'matchTerm',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Free text search terms to filter events that match these terms in any field, except for extended properties',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
};
|
|
|
|
methods = {
|
|
loadOptions: {
|
|
// Get all the calendars to display them to user so that he can
|
|
// select them easily
|
|
async getCalendars(
|
|
this: ILoadOptionsFunctions,
|
|
): Promise<INodePropertyOptions[]> {
|
|
const returnData: INodePropertyOptions[] = [];
|
|
const calendars = await googleApiRequestAllItems.call(
|
|
this,
|
|
'items',
|
|
'GET',
|
|
'/calendar/v3/users/me/calendarList',
|
|
);
|
|
for (const calendar of calendars) {
|
|
returnData.push({
|
|
name: calendar.summary,
|
|
value: calendar.id,
|
|
});
|
|
}
|
|
return returnData;
|
|
},
|
|
},
|
|
};
|
|
|
|
async poll(this: IPollFunctions): Promise<INodeExecutionData[][] | null> {
|
|
const poolTimes = this.getNodeParameter('pollTimes.item', []) as IDataObject[];
|
|
const triggerOn = this.getNodeParameter('triggerOn', '') as string;
|
|
const calendarId = this.getNodeParameter('calendarId') as string;
|
|
const webhookData = this.getWorkflowStaticData('node');
|
|
const matchTerm = this.getNodeParameter('options.matchTerm', '') as string;
|
|
|
|
if (poolTimes.length === 0) {
|
|
throw new NodeOperationError(
|
|
this.getNode(),
|
|
'Please set a poll time',
|
|
);
|
|
}
|
|
|
|
if (triggerOn === '') {
|
|
throw new NodeOperationError(
|
|
this.getNode(),
|
|
'Please select an event',
|
|
);
|
|
}
|
|
|
|
if (calendarId === '') {
|
|
throw new NodeOperationError(
|
|
this.getNode(),
|
|
'Please select a calendar',
|
|
);
|
|
}
|
|
|
|
const now = moment().utc().format();
|
|
|
|
const startDate = webhookData.lastTimeChecked as string || now;
|
|
|
|
const endDate = now;
|
|
|
|
const qs: IDataObject = {
|
|
showDeleted: false,
|
|
};
|
|
|
|
if (matchTerm !== '') {
|
|
qs.q = matchTerm;
|
|
}
|
|
|
|
let events;
|
|
|
|
if (triggerOn === 'eventCreated' || triggerOn === 'eventUpdated') {
|
|
Object.assign(qs, {
|
|
updatedMin: startDate,
|
|
orderBy: 'updated',
|
|
});
|
|
} else if (triggerOn === 'eventStarted' || triggerOn === 'eventEnded') {
|
|
Object.assign(qs, {
|
|
singleEvents: true,
|
|
timeMin: moment(startDate).startOf('second').utc().format(),
|
|
timeMax: moment(endDate).endOf('second').utc().format(),
|
|
orderBy: 'startTime',
|
|
});
|
|
}
|
|
|
|
if (this.getMode() === 'manual') {
|
|
delete qs.updatedMin;
|
|
delete qs.timeMin;
|
|
delete qs.timeMax;
|
|
|
|
qs.maxResults = 1;
|
|
events = await googleApiRequest.call(this, 'GET', `/calendar/v3/calendars/${calendarId}/events`, {}, qs);
|
|
events = events.items;
|
|
} else {
|
|
events = await googleApiRequestAllItems.call(this, 'items', 'GET', `/calendar/v3/calendars/${calendarId}/events`, {}, qs);
|
|
if (triggerOn === 'eventCreated') {
|
|
events = events.filter((event: { created: string }) => moment(event.created).isBetween(startDate, endDate));
|
|
} else if (triggerOn === 'eventUpdated') {
|
|
events = events.filter((event: { created: string, updated: string }) => !moment(moment(event.created).format('YYYY-MM-DDTHH:mm:ss')).isSame(moment(event.updated).format('YYYY-MM-DDTHH:mm:ss')));
|
|
} else if (triggerOn === 'eventStarted') {
|
|
events = events.filter((event: { start: { dateTime: string } }) => moment(event.start.dateTime).isBetween(startDate, endDate, null, '[]'));
|
|
} else if (triggerOn === 'eventEnded') {
|
|
events = events.filter((event: { end: { dateTime: string } }) => moment(event.end.dateTime).isBetween(startDate, endDate, null, '[]'));
|
|
}
|
|
}
|
|
|
|
webhookData.lastTimeChecked = endDate;
|
|
|
|
if (Array.isArray(events) && events.length) {
|
|
return [this.helpers.returnJsonArray(events)];
|
|
}
|
|
|
|
if (this.getMode() === 'manual') {
|
|
throw new NodeApiError(this.getNode(), { message: 'No data with the current filter could be found' });
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|