2019-11-15 12:44:46 -08:00
|
|
|
import * as moment from 'moment';
|
|
|
|
|
2019-11-12 12:48:45 -08:00
|
|
|
import {
|
|
|
|
IExecuteSingleFunctions,
|
|
|
|
} from 'n8n-core';
|
|
|
|
import {
|
|
|
|
IDataObject,
|
2019-11-15 12:44:46 -08:00
|
|
|
ILoadOptionsFunctions,
|
2019-11-12 12:48:45 -08:00
|
|
|
INodeTypeDescription,
|
|
|
|
INodeExecutionData,
|
|
|
|
INodeType,
|
|
|
|
INodePropertyOptions,
|
|
|
|
} from 'n8n-workflow';
|
|
|
|
import {
|
|
|
|
mailchimpApiRequest,
|
2019-11-15 12:44:46 -08:00
|
|
|
validateJSON,
|
2019-11-12 12:48:45 -08:00
|
|
|
} from './GenericFunctions';
|
2019-11-14 15:44:07 -08:00
|
|
|
|
|
|
|
enum Status {
|
|
|
|
subscribe = 'subscribe',
|
|
|
|
unsubscribed = 'unsubscribe',
|
|
|
|
cleaned = 'cleaned',
|
|
|
|
pending = 'pending',
|
2019-11-15 12:44:46 -08:00
|
|
|
transactional = 'transactional',
|
2019-11-14 15:44:07 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
interface ILocation {
|
|
|
|
latitude?: number;
|
|
|
|
longitude?: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ICreateMemberBody {
|
|
|
|
listId: string;
|
|
|
|
email_address: string;
|
|
|
|
email_type?: string;
|
|
|
|
status?: Status;
|
|
|
|
language?: string;
|
|
|
|
vip?: boolean;
|
|
|
|
location?: ILocation;
|
|
|
|
ips_signup?: string;
|
|
|
|
timestamp_signup?: string;
|
|
|
|
ip_opt?: string;
|
|
|
|
timestamp_opt?: string;
|
|
|
|
tags?: string[];
|
|
|
|
merge_fields?: IDataObject;
|
|
|
|
}
|
2019-11-12 12:48:45 -08:00
|
|
|
|
|
|
|
export class Mailchimp implements INodeType {
|
|
|
|
|
|
|
|
description: INodeTypeDescription = {
|
|
|
|
displayName: 'Mailchimp',
|
|
|
|
name: 'mailchimp',
|
|
|
|
icon: 'file:mailchimp.png',
|
|
|
|
group: ['output'],
|
|
|
|
version: 1,
|
|
|
|
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
|
|
|
|
description: 'Consume Mailchimp API',
|
|
|
|
defaults: {
|
|
|
|
name: 'Mailchimp',
|
|
|
|
color: '#c02428',
|
|
|
|
},
|
|
|
|
inputs: ['main'],
|
|
|
|
outputs: ['main'],
|
|
|
|
credentials: [
|
|
|
|
{
|
|
|
|
name: 'mailchimpApi',
|
|
|
|
required: true,
|
|
|
|
}
|
|
|
|
],
|
|
|
|
properties: [
|
|
|
|
{
|
|
|
|
displayName: 'Resource',
|
|
|
|
name: 'resource',
|
|
|
|
type: 'options',
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'Member',
|
|
|
|
value: 'member',
|
|
|
|
description: 'Add member to list',
|
|
|
|
},
|
|
|
|
],
|
2019-11-15 12:44:46 -08:00
|
|
|
default: 'member',
|
2019-11-12 12:48:45 -08:00
|
|
|
required: true,
|
|
|
|
description: 'Resource to consume.',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Operation',
|
|
|
|
name: 'operation',
|
|
|
|
type: 'options',
|
|
|
|
required: true,
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource: [
|
|
|
|
'member',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'Create',
|
|
|
|
value: 'create',
|
2019-11-14 15:44:07 -08:00
|
|
|
description: 'Create a new member on list',
|
2019-11-12 12:48:45 -08:00
|
|
|
},
|
|
|
|
],
|
2019-11-15 12:44:46 -08:00
|
|
|
default: 'create',
|
2019-11-12 12:48:45 -08:00
|
|
|
description: 'The operation to perform.',
|
|
|
|
},
|
2019-11-14 15:44:07 -08:00
|
|
|
{
|
|
|
|
displayName: 'List',
|
|
|
|
name: 'list',
|
|
|
|
type: 'options',
|
|
|
|
typeOptions: {
|
|
|
|
loadOptionsMethod: 'getLists',
|
|
|
|
},
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource: [
|
2019-11-15 12:44:46 -08:00
|
|
|
'member',
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
options: [],
|
|
|
|
required: true,
|
2019-11-15 12:44:46 -08:00
|
|
|
description: 'List of lists',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Email',
|
|
|
|
name: 'email',
|
|
|
|
type: 'string',
|
|
|
|
required: true,
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource: [
|
|
|
|
'member',
|
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
description: 'Email address for a subscriber.',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Status',
|
|
|
|
name: 'status',
|
|
|
|
type: 'options',
|
|
|
|
required: true,
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource: [
|
|
|
|
'member',
|
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'Subscribed',
|
|
|
|
value: 'subscribed',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Unsubscribed',
|
|
|
|
value: 'unsubscribed',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Cleaned',
|
|
|
|
value: 'cleaned',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Pending',
|
|
|
|
value: 'pending',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Transactional',
|
|
|
|
value: 'transactional',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
default: '',
|
|
|
|
description: `Subscriber's current status.`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'JSON Parameters',
|
|
|
|
name: 'jsonParameters',
|
|
|
|
type: 'boolean',
|
|
|
|
default: false,
|
|
|
|
description: '',
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
|
|
|
'member'
|
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Options',
|
|
|
|
name: 'options',
|
|
|
|
type: 'collection',
|
|
|
|
placeholder: 'Add Option',
|
|
|
|
default: {},
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
2019-11-15 12:44:46 -08:00
|
|
|
'member',
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
displayName: 'Email Type',
|
|
|
|
name: 'emailType',
|
|
|
|
type: 'options',
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'Email',
|
|
|
|
value: 'email',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Text',
|
|
|
|
value: 'text',
|
|
|
|
description: '',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
default: '',
|
|
|
|
description: 'Type of email this member asked to get',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Signup IP',
|
|
|
|
name: 'ipSignup',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
description: 'IP address the subscriber signed up from.',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Opt-in IP',
|
|
|
|
name: 'ipOptIn',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
description: 'The IP address the subscriber used to confirm their opt-in status.',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Signup Timestamp',
|
|
|
|
name: 'timestampSignup',
|
|
|
|
type: 'dateTime',
|
|
|
|
default: '',
|
|
|
|
description: 'The date and time the subscriber signed up for the list in ISO 8601 format.',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Language',
|
|
|
|
name: 'language',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
description: `If set/detected, the subscriber's language.`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Vip',
|
|
|
|
name: 'vip',
|
|
|
|
type: 'boolean',
|
|
|
|
default: false,
|
|
|
|
description: `Vip status for subscribers`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Opt-in Timestamp',
|
|
|
|
name: 'timestampOpt',
|
|
|
|
type: 'dateTime',
|
|
|
|
default: '',
|
|
|
|
description: `The date and time the subscribe confirmed their opt-in status in ISO 8601 format.`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Tags',
|
|
|
|
name: 'tags',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
description: `The tags that are associated with a member separeted by ,.`,
|
|
|
|
},
|
|
|
|
]
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Location',
|
|
|
|
name: 'locationFieldsUi',
|
|
|
|
type: 'fixedCollection',
|
|
|
|
placeholder: 'Add Location',
|
2019-11-15 12:44:46 -08:00
|
|
|
default: {},
|
2019-11-14 15:44:07 -08:00
|
|
|
description: `Subscriber location information.n`,
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
2019-11-15 12:44:46 -08:00
|
|
|
'member',
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
jsonParameters: [
|
|
|
|
false,
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'locationFieldsValues',
|
|
|
|
displayName: 'Location',
|
|
|
|
values: [
|
|
|
|
{
|
|
|
|
displayName: 'Latitude',
|
|
|
|
name: 'latitude',
|
|
|
|
type: 'string',
|
|
|
|
required: true,
|
|
|
|
description: 'The location latitude.',
|
2019-11-15 12:44:46 -08:00
|
|
|
default: '',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Longitude',
|
|
|
|
name: 'longitude',
|
|
|
|
type: 'string',
|
|
|
|
required: true,
|
|
|
|
description: 'The location longitude.',
|
2019-11-15 12:44:46 -08:00
|
|
|
default: '',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Merge Fields',
|
|
|
|
name: 'mergeFieldsUi',
|
|
|
|
placeholder: 'Add Merge Fields',
|
|
|
|
type: 'fixedCollection',
|
2019-11-15 12:44:46 -08:00
|
|
|
default: {},
|
2019-11-14 15:44:07 -08:00
|
|
|
typeOptions: {
|
|
|
|
multipleValues: true,
|
|
|
|
},
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
|
|
|
'member'
|
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
jsonParameters: [
|
|
|
|
false,
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
description: 'An individual merge var and value for a member.',
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
name: 'mergeFieldsValues',
|
|
|
|
displayName: 'Field',
|
|
|
|
typeOptions: {
|
2019-11-15 12:44:46 -08:00
|
|
|
multipleValueButtonText: 'Add Field',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
values: [
|
|
|
|
{
|
2019-11-15 12:44:46 -08:00
|
|
|
displayName: 'Field Name',
|
2019-11-14 15:44:07 -08:00
|
|
|
name: 'name',
|
|
|
|
type: 'string',
|
|
|
|
required: true,
|
2019-11-14 15:47:12 -08:00
|
|
|
description: 'Merge Field name',
|
2019-11-15 12:44:46 -08:00
|
|
|
default: '',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
{
|
2019-11-15 12:44:46 -08:00
|
|
|
displayName: 'Field Value',
|
2019-11-14 15:44:07 -08:00
|
|
|
name: 'value',
|
|
|
|
required: true,
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
2019-11-14 15:47:12 -08:00
|
|
|
description: 'Merge field value.',
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Merge Fields',
|
|
|
|
name: 'mergeFieldsJson',
|
|
|
|
type: 'json',
|
|
|
|
typeOptions: {
|
2019-11-15 12:44:46 -08:00
|
|
|
alwaysOpenEditWindow: true,
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
description: '',
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
2019-11-15 12:44:46 -08:00
|
|
|
'member',
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
jsonParameters: [
|
2019-11-15 12:44:46 -08:00
|
|
|
true,
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Location',
|
|
|
|
name: 'locationJson',
|
|
|
|
type: 'json',
|
|
|
|
typeOptions: {
|
2019-11-15 12:44:46 -08:00
|
|
|
alwaysOpenEditWindow: true,
|
2019-11-14 15:44:07 -08:00
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
description: '',
|
|
|
|
displayOptions: {
|
|
|
|
show: {
|
|
|
|
resource:[
|
2019-11-15 12:44:46 -08:00
|
|
|
'member',
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
operation: [
|
|
|
|
'create',
|
|
|
|
],
|
|
|
|
jsonParameters: [
|
2019-11-15 12:44:46 -08:00
|
|
|
true,
|
2019-11-14 15:44:07 -08:00
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2019-11-12 12:48:45 -08:00
|
|
|
]
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
methods = {
|
|
|
|
loadOptions: {
|
2019-11-14 15:44:07 -08:00
|
|
|
|
|
|
|
// Get all the available lists to display them to user so that he can
|
2019-11-12 12:48:45 -08:00
|
|
|
// select them easily
|
|
|
|
async getLists(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
|
|
|
|
const returnData: INodePropertyOptions[] = [];
|
|
|
|
let lists, response;
|
|
|
|
try {
|
|
|
|
response = await mailchimpApiRequest.call(this, '/lists', 'GET');
|
|
|
|
lists = response.lists;
|
|
|
|
} catch (err) {
|
|
|
|
throw new Error(`Mailchimp Error: ${err}`);
|
|
|
|
}
|
|
|
|
for (const list of lists) {
|
|
|
|
const listName = list.name;
|
|
|
|
const listId = list.id;
|
|
|
|
|
|
|
|
returnData.push({
|
|
|
|
name: listName,
|
|
|
|
value: listId,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return returnData;
|
|
|
|
},
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
async executeSingle(this: IExecuteSingleFunctions): Promise<INodeExecutionData> {
|
2019-11-14 15:44:07 -08:00
|
|
|
let response = {};
|
2019-11-12 12:48:45 -08:00
|
|
|
const resource = this.getNodeParameter('resource') as string;
|
2019-11-14 15:44:07 -08:00
|
|
|
const operation = this.getNodeParameter('operation') as string;
|
|
|
|
|
|
|
|
if (resource === 'member') {
|
|
|
|
if (operation === 'create') {
|
2019-11-12 12:48:45 -08:00
|
|
|
|
2019-11-14 15:44:07 -08:00
|
|
|
const listId = this.getNodeParameter('list') as string;
|
|
|
|
const email = this.getNodeParameter('email') as string;
|
|
|
|
const status = this.getNodeParameter('status') as Status;
|
|
|
|
const options = this.getNodeParameter('options') as IDataObject;
|
|
|
|
const jsonActive = this.getNodeParameter('jsonParameters') as IDataObject;
|
|
|
|
|
|
|
|
const body: ICreateMemberBody = {
|
|
|
|
listId,
|
|
|
|
email_address: email,
|
|
|
|
status
|
|
|
|
};
|
|
|
|
if (options.emailType) {
|
|
|
|
body.email_type = options.emailType as string;
|
|
|
|
}
|
|
|
|
if (options.languaje) {
|
|
|
|
body.language = options.language as string;
|
|
|
|
}
|
|
|
|
if (options.vip) {
|
|
|
|
body.vip = options.vip as boolean;
|
|
|
|
}
|
|
|
|
if (options.ipSignup) {
|
|
|
|
body.ips_signup = options.ipSignup as string;
|
|
|
|
}
|
|
|
|
if (options.ipOptIn) {
|
|
|
|
body.ip_opt = options.ipOptIn as string;
|
|
|
|
}
|
|
|
|
if (options.timestampOpt) {
|
|
|
|
body.timestamp_opt = moment(options.timestampOpt as string).format('YYYY-MM-DD HH:MM:SS') as string;
|
|
|
|
}
|
|
|
|
if (options.timestampSignup) {
|
|
|
|
body.timestamp_signup = moment(options.timestampSignup as string).format('YYYY-MM-DD HH:MM:SS') as string;
|
|
|
|
}
|
|
|
|
if (options.tags) {
|
|
|
|
// @ts-ignore
|
|
|
|
body.tags = options.tags.split(',') as string[];
|
|
|
|
}
|
|
|
|
if (!jsonActive) {
|
|
|
|
const locationValues = (this.getNodeParameter('locationFieldsUi') as IDataObject).locationFieldsValues as IDataObject;
|
|
|
|
if (locationValues) {
|
|
|
|
const location: ILocation = {};
|
|
|
|
for (const key of Object.keys(locationValues)) {
|
|
|
|
if (key === 'latitude') {
|
|
|
|
location.latitude = parseInt(locationValues[key] as string, 10) as number;
|
|
|
|
} else if (key === 'longitude') {
|
|
|
|
location.longitude = parseInt(locationValues[key] as string, 10) as number;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
body.location = location;
|
|
|
|
}
|
|
|
|
const mergeFieldsValues = (this.getNodeParameter('mergeFieldsUi') as IDataObject).mergeFieldsValues as IDataObject[];
|
|
|
|
if (mergeFieldsValues) {
|
|
|
|
const mergeFields = {};
|
|
|
|
for (let i = 0; i < mergeFieldsValues.length; i++) {
|
|
|
|
// @ts-ignore
|
|
|
|
mergeFields[mergeFieldsValues[i].name] = mergeFieldsValues[i].value;
|
|
|
|
}
|
|
|
|
body.merge_fields = mergeFields;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const locationJson = validateJSON(this.getNodeParameter('locationJson') as string);
|
|
|
|
const mergeFieldsJson = validateJSON(this.getNodeParameter('mergeFieldsJson') as string);
|
|
|
|
if (locationJson) {
|
|
|
|
body.location = locationJson;
|
|
|
|
}
|
|
|
|
if (mergeFieldsJson) {
|
|
|
|
body.merge_fields = mergeFieldsJson;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
response = await mailchimpApiRequest.call(this, `/lists/${listId}/members`, 'POST', body);
|
|
|
|
} catch (err) {
|
|
|
|
throw new Error(`Mailchimp Error: ${JSON.stringify(err)}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-11-12 12:48:45 -08:00
|
|
|
return {
|
2019-11-15 12:44:46 -08:00
|
|
|
json: response,
|
2019-11-12 12:48:45 -08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|