mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-13 05:47:31 -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>
616 lines
12 KiB
TypeScript
616 lines
12 KiB
TypeScript
import {
|
|
IExecuteFunctions,
|
|
} from 'n8n-core';
|
|
|
|
import {
|
|
IBinaryKeyData,
|
|
IDataObject,
|
|
ILoadOptionsFunctions,
|
|
INodeExecutionData,
|
|
INodePropertyOptions,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
pushbulletApiRequest,
|
|
pushbulletApiRequestAllItems,
|
|
} from './GenericFunctions';
|
|
|
|
import moment from 'moment-timezone';
|
|
|
|
export class Pushbullet implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Pushbullet',
|
|
name: 'pushbullet',
|
|
icon: 'file:pushbullet.svg',
|
|
group: ['input'],
|
|
version: 1,
|
|
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
|
|
description: 'Consume Pushbullet API',
|
|
defaults: {
|
|
name: 'Pushbullet',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
{
|
|
name: 'pushbulletOAuth2Api',
|
|
required: true,
|
|
},
|
|
],
|
|
properties: [
|
|
{
|
|
displayName: 'Resource',
|
|
name: 'resource',
|
|
type: 'options',
|
|
noDataExpression: true,
|
|
options: [
|
|
{
|
|
name: 'Push',
|
|
value: 'push',
|
|
},
|
|
],
|
|
default: 'push',
|
|
},
|
|
{
|
|
displayName: 'Operation',
|
|
name: 'operation',
|
|
type: 'options',
|
|
noDataExpression: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
},
|
|
},
|
|
options: [
|
|
{
|
|
name: 'Create',
|
|
value: 'create',
|
|
description: 'Create a push',
|
|
},
|
|
{
|
|
name: 'Delete',
|
|
value: 'delete',
|
|
description: 'Delete a push',
|
|
},
|
|
{
|
|
name: 'Get All',
|
|
value: 'getAll',
|
|
description: 'Get all pushes',
|
|
},
|
|
{
|
|
name: 'Update',
|
|
value: 'update',
|
|
description: 'Update a push',
|
|
},
|
|
],
|
|
default: 'create',
|
|
},
|
|
{
|
|
displayName: 'Type',
|
|
name: 'type',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'File',
|
|
value: 'file',
|
|
},
|
|
{
|
|
name: 'Link',
|
|
value: 'link',
|
|
},
|
|
{
|
|
name: 'Note',
|
|
value: 'note',
|
|
},
|
|
],
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
},
|
|
},
|
|
default: 'note',
|
|
},
|
|
{
|
|
displayName: 'Title',
|
|
name: 'title',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
type: [
|
|
'note',
|
|
'link',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
description: 'Title of the push',
|
|
},
|
|
{
|
|
displayName: 'Body',
|
|
name: 'body',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
type: [
|
|
'note',
|
|
'link',
|
|
'file',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
description: 'Body of the push',
|
|
},
|
|
{
|
|
displayName: 'URL',
|
|
name: 'url',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
type: [
|
|
'link',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
description: 'URL of the push',
|
|
},
|
|
{
|
|
displayName: 'Binary Property',
|
|
name: 'binaryPropertyName',
|
|
type: 'string',
|
|
default: 'data',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
type: [
|
|
'file',
|
|
],
|
|
},
|
|
},
|
|
placeholder: '',
|
|
description: 'Name of the binary property which contains the data for the file to be created',
|
|
},
|
|
{
|
|
displayName: 'Target',
|
|
name: 'target',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Channel Tag',
|
|
value: 'channel_tag',
|
|
description: 'Send the push to all subscribers to your channel that has this tag',
|
|
},
|
|
{
|
|
name: 'Default',
|
|
value: 'default',
|
|
description: 'Broadcast it to all of the user\'s devices',
|
|
},
|
|
{
|
|
name: 'Device ID',
|
|
value: 'device_iden',
|
|
description: 'Send the push to a specific device',
|
|
},
|
|
{
|
|
name: 'Email',
|
|
value: 'email',
|
|
description: 'Send the push to this email address',
|
|
},
|
|
],
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
},
|
|
},
|
|
default: 'default',
|
|
description: 'Define the medium that will be used to send the push',
|
|
},
|
|
{
|
|
displayName: 'Value',
|
|
name: 'value',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
},
|
|
hide: {
|
|
target: [
|
|
'default',
|
|
'device_iden',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
description: 'The value to be set depending on the target selected. For example, if the target selected is email then this field would take the email address of the person you are trying to send the push to.',
|
|
},
|
|
{
|
|
displayName: 'Value Name or ID',
|
|
name: 'value',
|
|
type: 'options',
|
|
typeOptions: {
|
|
loadOptionsMethod: 'getDevices',
|
|
},
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'create',
|
|
],
|
|
target: [
|
|
'device_iden',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
description: 'Choose from the list, or specify an ID using an <a href="https://docs.n8n.io/nodes/expressions.html#expressions">expression</a>',
|
|
},
|
|
{
|
|
displayName: 'Push ID',
|
|
name: 'pushId',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'delete',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
},
|
|
{
|
|
displayName: 'Return All',
|
|
name: 'returnAll',
|
|
type: 'boolean',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getAll',
|
|
],
|
|
resource: [
|
|
'push',
|
|
],
|
|
},
|
|
},
|
|
default: false,
|
|
description: 'Whether to return all results or only up to a given limit',
|
|
},
|
|
{
|
|
displayName: 'Limit',
|
|
name: 'limit',
|
|
type: 'number',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getAll',
|
|
],
|
|
resource: [
|
|
'push',
|
|
],
|
|
returnAll: [
|
|
false,
|
|
],
|
|
},
|
|
},
|
|
typeOptions: {
|
|
minValue: 1,
|
|
maxValue: 500,
|
|
},
|
|
default: 100,
|
|
description: 'Max number of results to return',
|
|
},
|
|
{
|
|
displayName: 'Filters',
|
|
name: 'filters',
|
|
type: 'collection',
|
|
placeholder: 'Add Filter',
|
|
default: {},
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'getAll',
|
|
],
|
|
},
|
|
},
|
|
options: [
|
|
{
|
|
displayName: 'Active',
|
|
name: 'active',
|
|
type: 'boolean',
|
|
default: false,
|
|
// eslint-disable-next-line n8n-nodes-base/node-param-description-boolean-without-whether
|
|
description: 'Don\'t return deleted pushes',
|
|
},
|
|
{
|
|
displayName: 'Modified After',
|
|
name: 'modified_after',
|
|
type: 'dateTime',
|
|
default: '',
|
|
description: 'Request pushes modified after this timestamp',
|
|
},
|
|
],
|
|
},
|
|
{
|
|
displayName: 'Push ID',
|
|
name: 'pushId',
|
|
type: 'string',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'update',
|
|
],
|
|
},
|
|
},
|
|
default: '',
|
|
},
|
|
{
|
|
displayName: 'Dismissed',
|
|
name: 'dismissed',
|
|
type: 'boolean',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'push',
|
|
],
|
|
operation: [
|
|
'update',
|
|
],
|
|
},
|
|
},
|
|
default: false,
|
|
description: 'Whether to mark a push as having been dismissed by the user, will cause any notifications for the push to be hidden if possible',
|
|
},
|
|
],
|
|
};
|
|
|
|
methods = {
|
|
loadOptions: {
|
|
async getDevices(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
|
|
const returnData: INodePropertyOptions[] = [];
|
|
const { devices } = await pushbulletApiRequest.call(this, 'GET', '/devices');
|
|
for (const device of devices) {
|
|
returnData.push({
|
|
name: device.nickname,
|
|
value: device.iden,
|
|
});
|
|
}
|
|
return returnData;
|
|
},
|
|
},
|
|
};
|
|
|
|
|
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
const items = this.getInputData();
|
|
const returnData: IDataObject[] = [];
|
|
const length = items.length;
|
|
const qs: IDataObject = {};
|
|
let responseData;
|
|
const resource = this.getNodeParameter('resource', 0) as string;
|
|
const operation = this.getNodeParameter('operation', 0) as string;
|
|
for (let i = 0; i < length; i++) {
|
|
try {
|
|
if (resource === 'push') {
|
|
if (operation === 'create') {
|
|
const type = this.getNodeParameter('type', i) as string;
|
|
|
|
const message = this.getNodeParameter('body', i) as string;
|
|
|
|
const target = this.getNodeParameter('target', i) as string;
|
|
|
|
const body: IDataObject = {
|
|
type,
|
|
body: message,
|
|
};
|
|
|
|
if (target !== 'default') {
|
|
const value = this.getNodeParameter('value', i) as string;
|
|
body[target as string] = value;
|
|
}
|
|
|
|
if (['note', 'link'].includes(type)) {
|
|
body.title = this.getNodeParameter('title', i) as string;
|
|
|
|
if (type === 'link') {
|
|
body.url = this.getNodeParameter('url', i) as string;
|
|
}
|
|
}
|
|
|
|
if (type === 'file') {
|
|
const binaryPropertyName = this.getNodeParameter('binaryPropertyName', 0) as string;
|
|
|
|
if (items[i].binary === undefined) {
|
|
throw new NodeOperationError(this.getNode(), 'No binary data exists on item!');
|
|
}
|
|
//@ts-ignore
|
|
if (items[i].binary[binaryPropertyName] === undefined) {
|
|
throw new NodeOperationError(this.getNode(), `No binary data property "${binaryPropertyName}" does not exists on item!`);
|
|
}
|
|
|
|
const binaryData = (items[i].binary as IBinaryKeyData)[binaryPropertyName];
|
|
const dataBuffer = await this.helpers.getBinaryDataBuffer(i, binaryPropertyName);
|
|
|
|
//create upload url
|
|
const {
|
|
upload_url: uploadUrl,
|
|
file_name,
|
|
file_type,
|
|
file_url,
|
|
} = await pushbulletApiRequest.call(
|
|
this,
|
|
'POST',
|
|
`/upload-request`,
|
|
{
|
|
file_name: binaryData.fileName,
|
|
file_type: binaryData.mimeType,
|
|
},
|
|
);
|
|
|
|
//upload the file
|
|
await pushbulletApiRequest.call(
|
|
this,
|
|
'POST',
|
|
'',
|
|
{},
|
|
{},
|
|
uploadUrl,
|
|
{
|
|
formData: {
|
|
file: {
|
|
value: dataBuffer,
|
|
options: {
|
|
filename: binaryData.fileName,
|
|
},
|
|
},
|
|
},
|
|
json: false,
|
|
},
|
|
);
|
|
|
|
body.file_name = file_name;
|
|
body.file_type = file_type;
|
|
body.file_url = file_url;
|
|
}
|
|
|
|
responseData = await pushbulletApiRequest.call(
|
|
this,
|
|
'POST',
|
|
`/pushes`,
|
|
body,
|
|
);
|
|
}
|
|
|
|
if (operation === 'getAll') {
|
|
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
|
|
|
const filters = this.getNodeParameter('filters', i) as IDataObject;
|
|
|
|
Object.assign(qs, filters);
|
|
|
|
if (qs.modified_after) {
|
|
qs.modified_after = moment(qs.modified_after as string).unix();
|
|
}
|
|
|
|
if (returnAll) {
|
|
responseData = await pushbulletApiRequestAllItems.call(this, 'pushes', 'GET', '/pushes', {}, qs);
|
|
|
|
} else {
|
|
qs.limit = this.getNodeParameter('limit', 0) as number;
|
|
|
|
responseData = await pushbulletApiRequest.call(this, 'GET', '/pushes', {}, qs);
|
|
|
|
responseData = responseData.pushes;
|
|
}
|
|
}
|
|
|
|
if (operation === 'delete') {
|
|
const pushId = this.getNodeParameter('pushId', i) as string;
|
|
|
|
responseData = await pushbulletApiRequest.call(
|
|
this,
|
|
'DELETE',
|
|
`/pushes/${pushId}`,
|
|
);
|
|
|
|
responseData = { success: true };
|
|
}
|
|
|
|
if (operation === 'update') {
|
|
const pushId = this.getNodeParameter('pushId', i) as string;
|
|
|
|
const dismissed = this.getNodeParameter('dismissed', i) as boolean;
|
|
|
|
responseData = await pushbulletApiRequest.call(
|
|
this,
|
|
'POST',
|
|
`/pushes/${pushId}`,
|
|
{
|
|
dismissed,
|
|
},
|
|
);
|
|
}
|
|
}
|
|
if (Array.isArray(responseData)) {
|
|
returnData.push.apply(returnData, responseData as IDataObject[]);
|
|
|
|
} else if (responseData !== undefined) {
|
|
returnData.push(responseData as IDataObject);
|
|
|
|
}
|
|
} catch (error) {
|
|
if (this.continueOnFail()) {
|
|
returnData.push({ error: error.message });
|
|
continue;
|
|
}
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
return [this.helpers.returnJsonArray(returnData)];
|
|
}
|
|
}
|