n8n/packages/nodes-base/nodes/Chargebee/Chargebee.node.ts

637 lines
14 KiB
TypeScript
Raw Normal View History

2019-06-23 03:35:23 -07:00
import { IExecuteFunctions } from 'n8n-core';
import {
IDataObject,
INodeTypeDescription,
INodeExecutionData,
INodeType,
NodeParameterValue,
} from 'n8n-workflow';
import * as requestPromise from 'request-promise-native';
interface CustomProperty {
name: string;
value: string;
}
interface FilterValue {
operation: string;
value: NodeParameterValue;
}
interface FilterValues {
[key: string]: FilterValue[];
}
export class Chargebee implements INodeType {
description: INodeTypeDescription = {
displayName: 'Chargebee',
name: 'chargebee',
icon: 'file:chargebee.png',
group: ['input'],
version: 1,
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
2019-06-23 03:35:23 -07:00
description: 'Retrieve data from Chargebee API',
defaults: {
name: 'Chargebee',
color: '#22BB11',
},
inputs: ['main'],
outputs: ['main'],
credentials: [
{
name: 'chargebeeApi',
required: true,
}
],
properties: [
{
displayName: 'Resource',
name: 'resource',
2019-06-23 03:35:23 -07:00
type: 'options',
options: [
{
name: 'Customer',
value: 'customer',
2019-06-23 03:35:23 -07:00
},
{
name: 'Invoice',
value: 'invoice',
2019-06-23 03:35:23 -07:00
},
{
name: 'Subscription',
value: 'subscription',
2019-06-23 03:35:23 -07:00
},
],
default: 'invoice',
description: 'The resource to operate on.',
2019-06-23 03:35:23 -07:00
},
2019-06-23 03:35:23 -07:00
// ----------------------------------
// customer
2019-06-23 03:35:23 -07:00
// ----------------------------------
{
displayName: 'Operation',
name: 'operation',
type: 'options',
2019-06-23 03:35:23 -07:00
displayOptions: {
show: {
resource: [
'customer',
2019-06-23 03:35:23 -07:00
],
},
},
options: [
{
name: 'Create',
value: 'create',
description: 'Create a customer',
2019-06-23 03:35:23 -07:00
},
],
default: 'create',
description: 'The operation to perform.',
2019-06-23 03:35:23 -07:00
},
// ----------------------------------
// customer:create
2019-06-23 03:35:23 -07:00
// ----------------------------------
{
displayName: 'Properties',
name: 'properties',
type: 'collection',
displayOptions: {
show: {
operation: [
'create'
],
resource: [
'customer',
2019-06-23 03:35:23 -07:00
],
},
},
default: {},
description: 'Properties to set on the new user',
placeholder: 'Add Property',
options: [
{
displayName: 'User Id',
name: 'id',
type: 'string',
default: '',
description: 'Id for the new customer. If not given, this will be auto-generated.',
},
{
displayName: 'First Name',
name: 'first_name',
type: 'string',
default: '',
description: 'The first name of the customer.',
},
{
displayName: 'Last Name',
name: 'last_name',
type: 'string',
default: '',
description: 'The last name of the customer.',
},
{
displayName: 'Email',
name: 'email',
type: 'string',
default: '',
description: 'The email address of the customer.',
},
{
displayName: 'Phone',
name: 'phone',
type: 'string',
default: '',
description: 'The phone number of the customer.',
},
{
displayName: 'Company',
name: 'company',
type: 'string',
default: '',
description: 'The company of the customer.',
},
{
displayName: 'Custom Properties',
name: 'customProperties',
placeholder: 'Add Custom Property',
description: 'Adds a custom property to set also values which have not been predefined.',
type: 'fixedCollection',
typeOptions: {
multipleValues: true,
},
default: {},
options: [
{
name: 'property',
displayName: 'Property',
values: [
{
displayName: 'Property Name',
name: 'name',
type: 'string',
default: '',
description: 'Name of the property to set.',
},
{
displayName: 'Property Value',
name: 'value',
type: 'string',
default: '',
description: 'Value of the property to set.',
},
]
},
],
}
],
},
// ----------------------------------
// invoice
2019-06-23 03:35:23 -07:00
// ----------------------------------
{
displayName: 'Operation',
name: 'operation',
default: 'list',
description: 'The operation to perform.',
type: 'options',
2019-06-23 03:35:23 -07:00
displayOptions: {
show: {
resource: [
'invoice',
2019-06-23 03:35:23 -07:00
],
},
},
options: [
{
name: 'List',
value: 'list',
description: 'Returns the invoices',
},
{
name: 'PDF Invoice URL',
value: 'pdfUrl',
description: 'Gets PDF invoice URL and adds it as property "pdfUrl"',
},
],
2019-06-23 03:35:23 -07:00
},
// ----------------------------------
// invoice:list
2019-06-23 03:35:23 -07:00
// ----------------------------------
{
displayName: 'Max results',
name: 'maxResults',
type: 'number',
typeOptions: {
minValue: 1,
maxValue: 100,
},
default: 10,
displayOptions: {
show: {
operation: [
'list'
],
resource: [
'invoice'
2019-06-23 03:35:23 -07:00
],
},
},
description: 'Max. amount of results to return(< 100).',
},
{
displayName: 'Filters',
name: 'filters',
placeholder: 'Add Filter',
description: 'Filter for invoices.',
type: 'fixedCollection',
typeOptions: {
multipleValues: true,
},
default: {},
displayOptions: {
show: {
operation: [
'list'
],
resource: [
'invoice'
2019-06-23 03:35:23 -07:00
],
},
},
options: [
{
name: 'date',
displayName: 'Invoice Date',
values: [
{
displayName: 'Operation',
name: 'operation',
type: 'options',
options: [
{
name: 'Is',
value: 'is'
},
{
name: 'Is Not',
value: 'is_not'
},
{
name: 'After',
value: 'after'
},
{
name: 'Before',
value: 'before'
},
],
default: 'after',
description: 'Operation to decide where the the data should be mapped to.',
},
{
displayName: 'Date',
name: 'value',
type: 'dateTime',
default: '',
description: 'Query date.',
},
]
},
{
name: 'total',
displayName: 'Invoice Amount',
values: [
{
displayName: 'Operation',
name: 'operation',
type: 'options',
options: [
{
name: 'Is',
value: 'is'
},
{
name: 'Is Not',
value: 'is_not'
},
{
name: 'Greater than',
value: 'gt'
},
{
name: 'Greater equal than',
value: 'gte'
},
{
name: 'Less than',
value: 'lt'
},
{
name: 'Less equal than',
value: 'lte'
},
],
default: 'gt',
description: 'Operation to decide where the the data should be mapped to.',
},
{
displayName: 'Amount',
name: 'value',
type: 'number',
typeOptions: {
numberPrecision: 2,
},
default: 0,
description: 'Query amount.',
},
]
},
],
},
// ----------------------------------
// invoice:pdfUrl
// ----------------------------------
{
displayName: 'Invoice Id',
name: 'invoiceId',
description: 'The id of the invoice to get.',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
operation: [
'pdfUrl'
],
resource: [
'invoice'
],
},
},
},
// ----------------------------------
// subscription
// ----------------------------------
{
displayName: 'Operation',
name: 'operation',
type: 'options',
displayOptions: {
show: {
resource: [
'subscription',
],
},
},
options: [
{
name: 'Cancel',
value: 'cancel',
description: 'Cancel a subscription',
},
{
name: 'Delete',
value: 'delete',
description: 'Delete a subscription',
},
],
default: 'delete',
description: 'The operation to perform.',
},
// ----------------------------------
// subscription:cancel
// ----------------------------------
{
displayName: 'Subscription Id',
name: 'subscriptionId',
description: 'The id of the subscription to cancel.',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
operation: [
'cancel'
],
resource: [
'subscription',
],
},
},
},
{
displayName: 'Schedule end of Term',
name: 'endOfTerm',
type: 'boolean',
default: false,
displayOptions: {
show: {
operation: [
'cancel'
],
resource: [
'subscription',
],
},
},
description: 'If set it will not cancel it directly in will instead schedule the cancelation for the end of the term..',
},
// ----------------------------------
// subscription:delete
// ----------------------------------
{
displayName: 'Subscription Id',
name: 'subscriptionId',
description: 'The id of the subscription to delete.',
type: 'string',
default: '',
required: true,
displayOptions: {
show: {
operation: [
'delete'
],
resource: [
'subscription',
],
},
},
},
2019-06-23 03:35:23 -07:00
],
};
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
const items = this.getInputData();
const returnData: IDataObject[] = [];
let item: INodeExecutionData;
const credentials = this.getCredentials('chargebeeApi');
if (credentials === undefined) {
throw new Error('No credentials got returned!');
}
const baseUrl = `https://${credentials.accountName}.chargebee.com/api/v2`;
// For Post
let body: IDataObject;
// For Query string
let qs: IDataObject;
for (let i = 0; i < items.length; i++) {
item = items[i];
const resource = this.getNodeParameter('resource', i) as string;
2019-06-23 03:35:23 -07:00
const operation = this.getNodeParameter('operation', i) as string;
let requestMethod = 'GET';
let endpoint = '';
body = {};
qs = {};
if (resource === 'customer') {
if (operation === 'create') {
// ----------------------------------
// create
// ----------------------------------
requestMethod = 'POST';
const properties = this.getNodeParameter('properties', i, {}) as IDataObject;
for (const key of Object.keys(properties)) {
if (key === 'customProperties' && (properties.customProperties as IDataObject).property !== undefined) {
for (const customProperty of (properties.customProperties as IDataObject)!.property! as CustomProperty[]) {
qs[customProperty.name] = customProperty.value;
}
} else {
qs[key] = properties[key];
}
}
endpoint = `customers`;
} else {
throw new Error(`The operation "${operation}" is not known!`);
}
2019-06-23 03:35:23 -07:00
} else if (resource === 'invoice') {
if (operation === 'list') {
// ----------------------------------
// list
// ----------------------------------
2019-06-23 03:35:23 -07:00
endpoint = 'invoices';
// TODO: Make also sorting configurable
qs['sort_by[desc]'] = 'date';
2019-06-23 03:35:23 -07:00
qs.limit = this.getNodeParameter('maxResults', i, {});
2019-06-23 03:35:23 -07:00
const setFilters: FilterValues = this.getNodeParameter('filters', i, {}) as unknown as FilterValues;
2019-06-23 03:35:23 -07:00
let filter: FilterValue;
let value: NodeParameterValue;
for (const filterProperty of Object.keys(setFilters)) {
for (filter of setFilters[filterProperty]) {
value = filter.value;
if (filterProperty === 'date') {
value = Math.floor(new Date(value as string).getTime() / 1000);
}
qs[`${filterProperty}[${filter.operation}]`] = value;
2019-06-23 03:35:23 -07:00
}
}
} else if (operation === 'pdfUrl') {
// ----------------------------------
// pdfUrl
// ----------------------------------
requestMethod = 'POST';
const invoiceId = this.getNodeParameter('invoiceId', i) as string;
endpoint = `invoices/${invoiceId.trim()}/pdf`;
} else {
throw new Error(`The operation "${operation}" is not known!`);
2019-06-23 03:35:23 -07:00
}
} else if (resource === 'subscription') {
if (operation === 'cancel') {
// ----------------------------------
// cancel
// ----------------------------------
2019-06-23 03:35:23 -07:00
requestMethod = 'POST';
2019-06-23 03:35:23 -07:00
const subscriptionId = this.getNodeParameter('subscriptionId', i, {}) as string;
body.end_of_term = this.getNodeParameter('endOfTerm', i, {}) as boolean;
2019-06-23 03:35:23 -07:00
endpoint = `subscriptions/${subscriptionId.trim()}/cancel`;
} else if (operation === 'delete') {
// ----------------------------------
// delete
// ----------------------------------
2019-06-23 03:35:23 -07:00
requestMethod = 'POST';
2019-06-23 03:35:23 -07:00
const subscriptionId = this.getNodeParameter('subscriptionId', i, {}) as string;
2019-06-23 03:35:23 -07:00
endpoint = `subscriptions/${subscriptionId.trim()}/delete`;
} else {
throw new Error(`The operation "${operation}" is not known!`);
2019-06-23 03:35:23 -07:00
}
} else {
throw new Error(`The resource "${resource}" is not known!`);
2019-06-23 03:35:23 -07:00
}
const options = {
method: requestMethod,
body,
qs,
uri: `${baseUrl}/${endpoint}`,
auth: {
user: credentials.apiKey as string,
pass: '',
},
json: true
};
const responseData = await requestPromise(options);
if (resource === 'invoice' && operation === 'list') {
2019-06-23 03:35:23 -07:00
responseData.list.forEach((data: IDataObject) => {
returnData.push(data.invoice as IDataObject);
});
} else if (resource === 'invoice' && operation === 'pdfUrl') {
const data: IDataObject = {};
Object.assign(data, items[i].json);
data.pdfUrl = responseData.download.download_url;
returnData.push(data);
2019-06-23 03:35:23 -07:00
} else {
returnData.push(responseData);
}
}
return [this.helpers.returnJsonArray(returnData)];
}
}