mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-16 01:24:05 -08:00
0448feec56
* ⚡ Initial setup
* 👕 Update `.eslintignore`
* 👕 Autofix node-param-default-missing (#3173)
* 🔥 Remove duplicate key
* 👕 Add exceptions
* 📦 Update package-lock.json
* 👕 Apply `node-class-description-inputs-wrong-trigger-node` (#3176)
* 👕 Apply `node-class-description-inputs-wrong-regular-node` (#3177)
* 👕 Apply `node-class-description-outputs-wrong` (#3178)
* 👕 Apply `node-execute-block-double-assertion-for-items` (#3179)
* 👕 Apply `node-param-default-wrong-for-collection` (#3180)
* 👕 Apply node-param-default-wrong-for-boolean (#3181)
* Autofixed default missing
* Autofixed booleans, worked well
* ⚡ Fix params
* ⏪ Undo exempted autofixes
* 📦 Update package-lock.json
* 👕 Apply node-class-description-missing-subtitle (#3182)
* ⚡ Fix missing comma
* 👕 Apply `node-param-default-wrong-for-fixed-collection` (#3184)
* 👕 Add exception for `node-class-description-missing-subtitle`
* 👕 Apply `node-param-default-wrong-for-multi-options` (#3185)
* 👕 Apply `node-param-collection-type-unsorted-items` (#3186)
* Missing coma
* 👕 Apply `node-param-default-wrong-for-simplify` (#3187)
* 👕 Apply `node-param-description-comma-separated-hyphen` (#3190)
* 👕 Apply `node-param-description-empty-string` (#3189)
* 👕 Apply `node-param-description-excess-inner-whitespace` (#3191)
* Rule looks good
* Add whitespace rule in eslint config
* :zao: fix
* 👕 Apply `node-param-description-identical-to-display-name` (#3193)
* 👕 Apply `node-param-description-missing-for-ignore-ssl-issues` (#3195)
* ⏪ Revert ":zao: fix"
This reverts commit ef8a76f3df
.
* 👕 Apply `node-param-description-missing-for-simplify` (#3196)
* 👕 Apply `node-param-description-missing-final-period` (#3194)
* Rule working as intended
* Add rule to eslint
* 👕 Apply node-param-description-missing-for-return-all (#3197)
* ⚡ Restore `lintfix` command
Co-authored-by: agobrech <45268029+agobrech@users.noreply.github.com>
Co-authored-by: Omar Ajoue <krynble@gmail.com>
Co-authored-by: agobrech <ael.gobrecht@gmail.com>
Co-authored-by: Michael Kret <michael.k@radency.com>
337 lines
8.6 KiB
TypeScript
337 lines
8.6 KiB
TypeScript
import {
|
|
IExecuteFunctions,
|
|
} from 'n8n-core';
|
|
|
|
import {
|
|
IDataObject,
|
|
ILoadOptionsFunctions,
|
|
INodeExecutionData,
|
|
INodePropertyOptions,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
NodeApiError,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
iterableApiRequest,
|
|
} from './GenericFunctions';
|
|
|
|
import {
|
|
eventFields,
|
|
eventOperations,
|
|
} from './EventDescription';
|
|
|
|
import {
|
|
userFields,
|
|
userOperations,
|
|
} from './UserDescription';
|
|
|
|
import {
|
|
userListFields,
|
|
userListOperations,
|
|
} from './UserListDescription';
|
|
|
|
import moment from 'moment-timezone';
|
|
|
|
export class Iterable implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Iterable',
|
|
name: 'iterable',
|
|
icon: 'file:iterable.png',
|
|
group: ['input'],
|
|
version: 1,
|
|
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
|
|
description: 'Consume Iterable API',
|
|
defaults: {
|
|
name: 'Iterable',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
{
|
|
name: 'iterableApi',
|
|
required: true,
|
|
},
|
|
],
|
|
properties: [
|
|
{
|
|
displayName: 'Resource',
|
|
name: 'resource',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Event',
|
|
value: 'event',
|
|
},
|
|
{
|
|
name: 'User',
|
|
value: 'user',
|
|
},
|
|
{
|
|
name: 'User List',
|
|
value: 'userList',
|
|
},
|
|
],
|
|
default: 'user',
|
|
description: 'The resource to operate on.',
|
|
},
|
|
...eventOperations,
|
|
...eventFields,
|
|
...userOperations,
|
|
...userFields,
|
|
...userListOperations,
|
|
...userListFields,
|
|
],
|
|
};
|
|
|
|
methods = {
|
|
loadOptions: {
|
|
// Get all the lists available channels
|
|
async getLists(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
|
|
const { lists } = await iterableApiRequest.call(this, 'GET', '/lists');
|
|
const returnData: INodePropertyOptions[] = [];
|
|
for (const list of lists) {
|
|
returnData.push({
|
|
name: list.name,
|
|
value: list.id,
|
|
});
|
|
}
|
|
return returnData;
|
|
},
|
|
},
|
|
};
|
|
|
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
const items = this.getInputData();
|
|
const returnData: IDataObject[] = [];
|
|
const length = items.length;
|
|
const timezone = this.getTimezone();
|
|
const qs: IDataObject = {};
|
|
let responseData;
|
|
const resource = this.getNodeParameter('resource', 0) as string;
|
|
const operation = this.getNodeParameter('operation', 0) as string;
|
|
|
|
if (resource === 'event') {
|
|
if (operation === 'track') {
|
|
// https://api.iterable.com/api/docs#events_trackBulk
|
|
const events = [];
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const name = this.getNodeParameter('name', i) as string;
|
|
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
|
|
if (!additionalFields.email && !additionalFields.id) {
|
|
throw new NodeOperationError(this.getNode(), 'Either email or userId must be passed in to identify the user. Please add one of both via "Additional Fields". If both are passed in, email takes precedence.');
|
|
}
|
|
|
|
const body: IDataObject = {
|
|
eventName: name,
|
|
};
|
|
|
|
Object.assign(body, additionalFields);
|
|
|
|
if (body.dataFieldsUi) {
|
|
const dataFields = (body.dataFieldsUi as IDataObject).dataFieldValues as IDataObject[];
|
|
const data: IDataObject = {};
|
|
for (const dataField of dataFields) {
|
|
data[dataField.key as string] = dataField.value;
|
|
}
|
|
body.dataFields = data;
|
|
delete body.dataFieldsUi;
|
|
}
|
|
|
|
if (body.createdAt) {
|
|
body.createdAt = moment.tz(body.createdAt, timezone).unix();
|
|
}
|
|
|
|
events.push(body);
|
|
}
|
|
|
|
responseData = await iterableApiRequest.call(this, 'POST', '/events/trackBulk', { events });
|
|
|
|
returnData.push(responseData);
|
|
}
|
|
}
|
|
|
|
if (resource === 'user') {
|
|
if (operation === 'upsert') {
|
|
// https://api.iterable.com/api/docs#users_updateUser
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const identifier = this.getNodeParameter('identifier', i) as string;
|
|
|
|
const value = this.getNodeParameter('value', i) as string;
|
|
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
|
|
const body: IDataObject = {};
|
|
|
|
if (identifier === 'email') {
|
|
body.email = value;
|
|
} else {
|
|
body.preferUserId = this.getNodeParameter('preferUserId', i) as boolean;
|
|
body.userId = value;
|
|
}
|
|
|
|
Object.assign(body, additionalFields);
|
|
|
|
if (body.dataFieldsUi) {
|
|
const dataFields = (body.dataFieldsUi as IDataObject).dataFieldValues as IDataObject[];
|
|
const data: IDataObject = {};
|
|
for (const dataField of dataFields) {
|
|
data[dataField.key as string] = dataField.value;
|
|
}
|
|
body.dataFields = data;
|
|
delete body.dataFieldsUi;
|
|
}
|
|
|
|
responseData = await iterableApiRequest.call(this, 'POST', '/users/update', body);
|
|
|
|
if (this.continueOnFail() === false) {
|
|
if (responseData.code !== 'Success') {
|
|
throw new NodeOperationError(this.getNode(),
|
|
`Iterable error response [400]: ${responseData.msg}`,
|
|
);
|
|
}
|
|
}
|
|
|
|
returnData.push(responseData);
|
|
}
|
|
}
|
|
|
|
if (operation === 'delete') {
|
|
// https://api.iterable.com/api/docs#users_delete
|
|
// https://api.iterable.com/api/docs#users_delete_0
|
|
for (let i = 0; i < length; i++) {
|
|
const by = this.getNodeParameter('by', i) as string;
|
|
|
|
let endpoint;
|
|
|
|
if (by === 'email') {
|
|
const email = this.getNodeParameter('email', i) as string;
|
|
endpoint = `/users/${email}`;
|
|
} else {
|
|
const userId = this.getNodeParameter('userId', i) as string;
|
|
endpoint = `/users/byUserId/${userId}`;
|
|
}
|
|
|
|
responseData = await iterableApiRequest.call(this, 'DELETE', endpoint);
|
|
|
|
if (this.continueOnFail() === false) {
|
|
if (responseData.code !== 'Success') {
|
|
throw new NodeApiError(this.getNode(), responseData);
|
|
}
|
|
}
|
|
|
|
returnData.push(responseData);
|
|
}
|
|
}
|
|
|
|
if (operation === 'get') {
|
|
// https://api.iterable.com/api/docs#users_getUser
|
|
// https://api.iterable.com/api/docs#users_getUserById
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const by = this.getNodeParameter('by', i) as string;
|
|
|
|
let endpoint;
|
|
|
|
if (by === 'email') {
|
|
const email = this.getNodeParameter('email', i) as string;
|
|
endpoint = `/users/getByEmail`;
|
|
qs.email = email;
|
|
} else {
|
|
const userId = this.getNodeParameter('userId', i) as string;
|
|
endpoint = `/users/byUserId/${userId}`;
|
|
}
|
|
|
|
responseData = await iterableApiRequest.call(this, 'GET', endpoint, {}, qs);
|
|
|
|
if (this.continueOnFail() === false) {
|
|
if (Object.keys(responseData).length === 0) {
|
|
throw new NodeApiError(this.getNode(), responseData,
|
|
{ message: `User not found`, httpCode: '404' },
|
|
);
|
|
}
|
|
}
|
|
|
|
responseData = responseData.user || {};
|
|
returnData.push(responseData);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (resource === 'userList') {
|
|
if (operation === 'add') {
|
|
//https://api.iterable.com/api/docs#lists_subscribe
|
|
const listId = this.getNodeParameter('listId', 0) as string;
|
|
|
|
const identifier = this.getNodeParameter('identifier', 0) as string;
|
|
|
|
const body: IDataObject = {
|
|
listId: parseInt(listId, 10),
|
|
subscribers: [],
|
|
};
|
|
|
|
const subscribers: IDataObject[] = [];
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const value = this.getNodeParameter('value', i) as string;
|
|
|
|
if (identifier === 'email') {
|
|
subscribers.push({ email: value });
|
|
} else {
|
|
subscribers.push({ userId: value });
|
|
}
|
|
}
|
|
|
|
body.subscribers = subscribers;
|
|
|
|
responseData = await iterableApiRequest.call(this, 'POST', '/lists/subscribe', body);
|
|
|
|
returnData.push(responseData);
|
|
}
|
|
|
|
if (operation === 'remove') {
|
|
//https://api.iterable.com/api/docs#lists_unsubscribe
|
|
const listId = this.getNodeParameter('listId', 0) as string;
|
|
|
|
const identifier = this.getNodeParameter('identifier', 0) as string;
|
|
|
|
const additionalFields = this.getNodeParameter('additionalFields', 0) as IDataObject;
|
|
|
|
const body: IDataObject = {
|
|
listId: parseInt(listId, 10),
|
|
subscribers: [],
|
|
campaignId: additionalFields.campaignId as number,
|
|
channelUnsubscribe: additionalFields.channelUnsubscribe as boolean,
|
|
};
|
|
|
|
const subscribers: IDataObject[] = [];
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
const value = this.getNodeParameter('value', i) as string;
|
|
|
|
if (identifier === 'email') {
|
|
subscribers.push({ email: value });
|
|
} else {
|
|
subscribers.push({ userId: value });
|
|
}
|
|
}
|
|
|
|
body.subscribers = subscribers;
|
|
|
|
responseData = await iterableApiRequest.call(this, 'POST', '/lists/unsubscribe', body);
|
|
|
|
returnData.push(responseData);
|
|
}
|
|
}
|
|
return [this.helpers.returnJsonArray(returnData)];
|
|
}
|
|
}
|