mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
⚡ Add user list resource to Iterable Node (#1252)
* Add user list resource
* ⚡ Now lists are loaded automatically
This commit is contained in:
parent
e3aff74f6b
commit
f36af0fcfd
|
@ -4,7 +4,9 @@ import {
|
||||||
|
|
||||||
import {
|
import {
|
||||||
IDataObject,
|
IDataObject,
|
||||||
|
ILoadOptionsFunctions,
|
||||||
INodeExecutionData,
|
INodeExecutionData,
|
||||||
|
INodePropertyOptions,
|
||||||
INodeType,
|
INodeType,
|
||||||
INodeTypeDescription,
|
INodeTypeDescription,
|
||||||
} from 'n8n-workflow';
|
} from 'n8n-workflow';
|
||||||
|
@ -23,6 +25,11 @@ import {
|
||||||
userOperations,
|
userOperations,
|
||||||
} from './UserDescription';
|
} from './UserDescription';
|
||||||
|
|
||||||
|
import {
|
||||||
|
userListFields,
|
||||||
|
userListOperations,
|
||||||
|
} from './UserListDescription';
|
||||||
|
|
||||||
import * as moment from 'moment-timezone';
|
import * as moment from 'moment-timezone';
|
||||||
|
|
||||||
export class Iterable implements INodeType {
|
export class Iterable implements INodeType {
|
||||||
|
@ -60,6 +67,10 @@ export class Iterable implements INodeType {
|
||||||
name: 'User',
|
name: 'User',
|
||||||
value: 'user',
|
value: 'user',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: 'User List',
|
||||||
|
value: 'userList',
|
||||||
|
},
|
||||||
],
|
],
|
||||||
default: 'user',
|
default: 'user',
|
||||||
description: 'The resource to operate on.',
|
description: 'The resource to operate on.',
|
||||||
|
@ -68,9 +79,28 @@ export class Iterable implements INodeType {
|
||||||
...eventFields,
|
...eventFields,
|
||||||
...userOperations,
|
...userOperations,
|
||||||
...userFields,
|
...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[][]> {
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||||
const items = this.getInputData();
|
const items = this.getInputData();
|
||||||
const returnData: IDataObject[] = [];
|
const returnData: IDataObject[] = [];
|
||||||
|
@ -181,7 +211,7 @@ export class Iterable implements INodeType {
|
||||||
|
|
||||||
if (by === 'email') {
|
if (by === 'email') {
|
||||||
const email = this.getNodeParameter('email', i) as string;
|
const email = this.getNodeParameter('email', i) as string;
|
||||||
endpoint = `/users/${email}`;
|
endpoint = `/users/${email}`;
|
||||||
} else {
|
} else {
|
||||||
const userId = this.getNodeParameter('userId', i) as string;
|
const userId = this.getNodeParameter('userId', i) as string;
|
||||||
endpoint = `/users/byUserId/${userId}`;
|
endpoint = `/users/byUserId/${userId}`;
|
||||||
|
@ -212,7 +242,7 @@ export class Iterable implements INodeType {
|
||||||
|
|
||||||
if (by === 'email') {
|
if (by === 'email') {
|
||||||
const email = this.getNodeParameter('email', i) as string;
|
const email = this.getNodeParameter('email', i) as string;
|
||||||
endpoint = `/users/getByEmail`;
|
endpoint = `/users/getByEmail`;
|
||||||
qs.email = email;
|
qs.email = email;
|
||||||
} else {
|
} else {
|
||||||
const userId = this.getNodeParameter('userId', i) as string;
|
const userId = this.getNodeParameter('userId', i) as string;
|
||||||
|
@ -234,6 +264,74 @@ export class Iterable implements INodeType {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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)];
|
return [this.helpers.returnJsonArray(returnData)];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import {
|
import {
|
||||||
INodeProperties,
|
INodeProperties,
|
||||||
} from 'n8n-workflow';
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
export const userOperations = [
|
export const userOperations = [
|
||||||
{
|
{
|
||||||
|
@ -38,9 +38,9 @@ export const userOperations = [
|
||||||
|
|
||||||
export const userFields = [
|
export const userFields = [
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- */
|
/* -------------------------------------------------------------------------- */
|
||||||
/* user:upsert */
|
/* user:upsert */
|
||||||
/* -------------------------------------------------------------------------- */
|
/* -------------------------------------------------------------------------- */
|
||||||
{
|
{
|
||||||
displayName: 'Identifier',
|
displayName: 'Identifier',
|
||||||
name: 'identifier',
|
name: 'identifier',
|
||||||
|
@ -167,9 +167,10 @@ export const userFields = [
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
/* -------------------------------------------------------------------------- */
|
|
||||||
/* user:delete */
|
/* -------------------------------------------------------------------------- */
|
||||||
/* -------------------------------------------------------------------------- */
|
/* user:delete */
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
{
|
{
|
||||||
displayName: 'By',
|
displayName: 'By',
|
||||||
name: 'by',
|
name: 'by',
|
||||||
|
@ -240,9 +241,10 @@ export const userFields = [
|
||||||
default: '',
|
default: '',
|
||||||
description: 'Email for a particular user',
|
description: 'Email for a particular user',
|
||||||
},
|
},
|
||||||
/* -------------------------------------------------------------------------- */
|
|
||||||
/* user:get */
|
/* -------------------------------------------------------------------------- */
|
||||||
/* -------------------------------------------------------------------------- */
|
/* user:get */
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
{
|
{
|
||||||
displayName: 'By',
|
displayName: 'By',
|
||||||
name: 'by',
|
name: 'by',
|
||||||
|
|
208
packages/nodes-base/nodes/Iterable/UserListDescription.ts
Normal file
208
packages/nodes-base/nodes/Iterable/UserListDescription.ts
Normal file
|
@ -0,0 +1,208 @@
|
||||||
|
import {
|
||||||
|
INodeProperties,
|
||||||
|
} from 'n8n-workflow';
|
||||||
|
|
||||||
|
export const userListOperations = [
|
||||||
|
{
|
||||||
|
displayName: 'Operation',
|
||||||
|
name: 'operation',
|
||||||
|
type: 'options',
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Add',
|
||||||
|
value: 'add',
|
||||||
|
description: 'Add user to list',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Remove',
|
||||||
|
value: 'remove',
|
||||||
|
description: 'Remove a user from a list',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default: 'add',
|
||||||
|
description: 'The operation to perform.',
|
||||||
|
},
|
||||||
|
] as INodeProperties[];
|
||||||
|
|
||||||
|
export const userListFields = [
|
||||||
|
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
|
/* userList:add */
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
|
{
|
||||||
|
displayName: 'List ID',
|
||||||
|
name: 'listId',
|
||||||
|
type: 'options',
|
||||||
|
typeOptions: {
|
||||||
|
loadOptionsMethod: 'getLists',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'add',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
description: 'Identifier to be used',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Identifier',
|
||||||
|
name: 'identifier',
|
||||||
|
type: 'options',
|
||||||
|
required: true,
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Email',
|
||||||
|
value: 'email',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'User ID',
|
||||||
|
value: 'userId',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'add',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
description: 'Identifier to be used',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Value',
|
||||||
|
name: 'value',
|
||||||
|
type: 'string',
|
||||||
|
required: true,
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'add',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
|
/* userList:remove */
|
||||||
|
/* -------------------------------------------------------------------------- */
|
||||||
|
{
|
||||||
|
displayName: 'List ID',
|
||||||
|
name: 'listId',
|
||||||
|
type: 'options',
|
||||||
|
typeOptions: {
|
||||||
|
loadOptionsMethod: 'getLists',
|
||||||
|
},
|
||||||
|
required: true,
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'remove',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
description: 'Identifier to be used',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Identifier',
|
||||||
|
name: 'identifier',
|
||||||
|
type: 'options',
|
||||||
|
required: true,
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
name: 'Email',
|
||||||
|
value: 'email',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'User ID',
|
||||||
|
value: 'userId',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'remove',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
description: 'Identifier to be used',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Value',
|
||||||
|
name: 'value',
|
||||||
|
type: 'string',
|
||||||
|
required: true,
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'remove',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Additional Fields',
|
||||||
|
name: 'additionalFields',
|
||||||
|
type: 'collection',
|
||||||
|
placeholder: 'Add Field',
|
||||||
|
default: {},
|
||||||
|
displayOptions: {
|
||||||
|
show: {
|
||||||
|
resource: [
|
||||||
|
'userList',
|
||||||
|
],
|
||||||
|
operation: [
|
||||||
|
'remove',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
displayName: 'Campaign ID',
|
||||||
|
name: 'campaignId',
|
||||||
|
type: 'number',
|
||||||
|
default: 0,
|
||||||
|
description: 'Attribute unsubscribe to a campaign',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
displayName: 'Channel Unsubscribe',
|
||||||
|
name: 'channelUnsubscribe',
|
||||||
|
type: 'boolean',
|
||||||
|
default: false,
|
||||||
|
description: `Unsubscribe email from list's associated channel - essentially a global unsubscribe`,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as INodeProperties[];
|
Loading…
Reference in a new issue