feat(Emelia Node): Add Campaign > Duplicate functionality (#3000)

* feat(Emelia Node): Add campaign duplication feature

*  small ui fixes, added credential test, fixed nodelinter issues

*  Improvements

*  Updated wording for Number operations on IF-Node (#3065)

* fix(Google Tasks Node): Fix "Show Completed" option and hide title field where not needed (#2741)

* 🐛 Google Tasks: Fix showCompleted

*  Improvements

Co-authored-by: ricardo <ricardoespinoza105@gmail.com>

* feat(Mocean Node): Add "Delivery Report URL" option and credential tests (#3075)

* add dlr url column

add dlr url(delivery report URl) column. Allow user set the
endpoint
to receive the report

* update

update delivery report url description

*  fixed nodelinter issues, added credential test, replaced icon

*  Improvements

Co-authored-by: d3no <d3no520@gmail.com>
Co-authored-by: Michael Kret <michael.k@radency.com>

*  Normalize name

Co-authored-by: Michael Kret <michael.k@radency.com>
Co-authored-by: ricardo <ricardoespinoza105@gmail.com>
Co-authored-by: Jonathan Bennetts <jonathan.bennetts@gmail.com>
Co-authored-by: Tom <19203795+that-one-tom@users.noreply.github.com>
Co-authored-by: Ricardo Espinoza <ricardo@n8n.io>
Co-authored-by: d3no <d3no520@gmail.com>
Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
Charles Lecalier 2022-04-01 10:12:47 +02:00 committed by GitHub
parent c89d2b10f2
commit 0b08be1c0b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 247 additions and 51 deletions

View file

@ -9,6 +9,7 @@ export const campaignOperations: INodeProperties[] = [
type: 'options', type: 'options',
default: 'get', default: 'get',
description: 'Operation to perform', description: 'Operation to perform',
noDataExpression: true,
options: [ options: [
{ {
name: 'Add Contact', name: 'Add Contact',
@ -18,6 +19,10 @@ export const campaignOperations: INodeProperties[] = [
name: 'Create', name: 'Create',
value: 'create', value: 'create',
}, },
{
name: 'Duplicate',
value: 'duplicate',
},
{ {
name: 'Get', name: 'Get',
value: 'get', value: 'get',
@ -58,7 +63,7 @@ export const campaignFields: INodeProperties[] = [
}, },
default: [], default: [],
required: true, required: true,
description: 'The ID of the campaign to add the contact to.', description: 'The ID of the campaign to add the contact to',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -76,7 +81,7 @@ export const campaignFields: INodeProperties[] = [
type: 'string', type: 'string',
required: true, required: true,
default: '', default: '',
description: 'The email of the contact to add to the campaign.', description: 'The email of the contact to add to the campaign',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -113,7 +118,7 @@ export const campaignFields: INodeProperties[] = [
typeOptions: { typeOptions: {
multipleValues: true, multipleValues: true,
}, },
description: 'Filter by custom fields ', description: 'Filter by custom fields',
default: {}, default: {},
options: [ options: [
{ {
@ -125,14 +130,14 @@ export const campaignFields: INodeProperties[] = [
name: 'fieldName', name: 'fieldName',
type: 'string', type: 'string',
default: '', default: '',
description: 'The name of the field to add custom field to.', description: 'The name of the field to add custom field to',
}, },
{ {
displayName: 'Value', displayName: 'Value',
name: 'value', name: 'value',
type: 'string', type: 'string',
default: '', default: '',
description: 'The value to set on custom field.', description: 'The value to set on custom field',
}, },
], ],
}, },
@ -143,49 +148,49 @@ export const campaignFields: INodeProperties[] = [
name: 'firstName', name: 'firstName',
type: 'string', type: 'string',
default: '', default: '',
description: 'First name of the contact to add.', description: 'First name of the contact to add',
},
{
displayName: 'Last Contacted',
name: 'lastContacted',
type: 'dateTime',
default: '',
description: 'Last contacted date of the contact to add',
}, },
{ {
displayName: 'Last Name', displayName: 'Last Name',
name: 'lastName', name: 'lastName',
type: 'string', type: 'string',
default: '', default: '',
description: 'Last name of the contact to add.', description: 'Last name of the contact to add',
},
{
displayName: 'Last Contacted',
name: 'lastContacted',
type: 'string',
default: '',
description: 'Last contacted date of the contact to add.',
}, },
{ {
displayName: 'Last Open', displayName: 'Last Open',
name: 'lastOpen', name: 'lastOpen',
type: 'string', type: 'dateTime',
default: '', default: '',
description: 'Last opened date of the contact to add.', description: 'Last opened date of the contact to add',
}, },
{ {
displayName: 'Last Replied', displayName: 'Last Replied',
name: 'lastReplied', name: 'lastReplied',
type: 'string', type: 'dateTime',
default: '', default: '',
description: 'Last replied date of the contact to add.', description: 'Last replied date of the contact to add',
}, },
{ {
displayName: 'Mails Sent', displayName: 'Mails Sent',
name: 'mailsSent', name: 'mailsSent',
type: 'number', type: 'number',
default: 0, default: 0,
description: 'Number of emails sent to the contact to add.', description: 'Number of emails sent to the contact to add',
}, },
{ {
displayName: 'Phone Number', displayName: 'Phone Number',
name: 'phoneNumber', name: 'phoneNumber',
type: 'string', type: 'string',
default: '', default: '',
description: 'Phone number of the contact to add.', description: 'Phone number of the contact to add',
}, },
], ],
}, },
@ -199,7 +204,7 @@ export const campaignFields: INodeProperties[] = [
type: 'string', type: 'string',
required: true, required: true,
default: '', default: '',
description: 'The name of the campaign to create.', description: 'The name of the campaign to create',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -221,7 +226,7 @@ export const campaignFields: INodeProperties[] = [
type: 'string', type: 'string',
default: '', default: '',
required: true, required: true,
description: 'The ID of the campaign to retrieve.', description: 'The ID of the campaign to retrieve',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -242,7 +247,7 @@ export const campaignFields: INodeProperties[] = [
name: 'returnAll', name: 'returnAll',
type: 'boolean', type: 'boolean',
default: false, default: false,
description: 'Return all results.', description: 'Whether to return all results or only up to a given limit',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -259,7 +264,7 @@ export const campaignFields: INodeProperties[] = [
name: 'limit', name: 'limit',
type: 'number', type: 'number',
default: 100, default: 100,
description: 'The number of results to return.', description: 'Max number of results to return',
typeOptions: { typeOptions: {
minValue: 1, minValue: 1,
maxValue: 100, maxValue: 100,
@ -323,4 +328,93 @@ export const campaignFields: INodeProperties[] = [
}, },
}, },
// ----------------------------------
// campaign: duplicate
// ----------------------------------
{
displayName: 'Campaign ID',
name: 'campaignId',
type: 'options',
default: '',
required: true,
description: 'The ID of the campaign to duplicate',
typeOptions: {
loadOptionsMethod: 'getCampaigns',
},
displayOptions: {
show: {
resource: [
'campaign',
],
operation: [
'duplicate',
],
},
},
},
{
displayName: 'New Campaign Name',
name: 'campaignName',
type: 'string',
required: true,
default: '',
description: 'The name of the new campaign to create',
displayOptions: {
show: {
resource: [
'campaign',
],
operation: [
'duplicate',
],
},
},
},
{
displayName: 'Options',
name: 'options',
type: 'collection',
default: {},
placeholder: 'Add Field',
displayOptions: {
show: {
operation: [
'duplicate',
],
resource: [
'campaign',
],
},
},
options: [
{
displayName: 'Copy Contacts',
name: 'copyContacts',
type: 'boolean',
default: false,
description: 'Whether to copy all the contacts from the original campaign',
},
{
displayName: 'Copy Email Provider',
name: 'copyProvider',
type: 'boolean',
default: true,
description: 'Whether to set the same email provider than the original campaign',
},
{
displayName: 'Copy Email Sequence',
name: 'copyMails',
type: 'boolean',
default: true,
description: 'Whether to copy all the steps of the email sequence from the original campaign',
},
{
displayName: 'Copy Global Settings',
name: 'copySettings',
type: 'boolean',
default: true,
description: 'Whether to copy all the general settings from the original campaign',
},
],
},
]; ];

View file

@ -7,8 +7,9 @@ export const contactListOperations: INodeProperties[] = [
displayName: 'Operation', displayName: 'Operation',
name: 'operation', name: 'operation',
type: 'options', type: 'options',
default: 'get', default: 'getAll',
description: 'Operation to perform', description: 'Operation to perform',
noDataExpression: true,
options: [ options: [
{ {
name: 'Add', name: 'Add',
@ -42,7 +43,7 @@ export const contactListFields: INodeProperties[] = [
}, },
default: [], default: [],
required: true, required: true,
description: 'The ID of the contact list to add the contact to.', description: 'The ID of the contact list to add the contact to',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -60,7 +61,7 @@ export const contactListFields: INodeProperties[] = [
type: 'string', type: 'string',
required: true, required: true,
default: '', default: '',
description: 'The email of the contact to add to the contact list.', description: 'The email of the contact to add to the contact list',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -97,7 +98,7 @@ export const contactListFields: INodeProperties[] = [
typeOptions: { typeOptions: {
multipleValues: true, multipleValues: true,
}, },
description: 'Filter by custom fields ', description: 'Filter by custom fields',
default: {}, default: {},
options: [ options: [
{ {
@ -109,14 +110,14 @@ export const contactListFields: INodeProperties[] = [
name: 'fieldName', name: 'fieldName',
type: 'string', type: 'string',
default: '', default: '',
description: 'The name of the field to add custom field to.', description: 'The name of the field to add custom field to',
}, },
{ {
displayName: 'Value', displayName: 'Value',
name: 'value', name: 'value',
type: 'string', type: 'string',
default: '', default: '',
description: 'The value to set on custom field.', description: 'The value to set on custom field',
}, },
], ],
}, },
@ -127,49 +128,49 @@ export const contactListFields: INodeProperties[] = [
name: 'firstName', name: 'firstName',
type: 'string', type: 'string',
default: '', default: '',
description: 'First name of the contact to add.', description: 'First name of the contact to add',
},
{
displayName: 'Last Name',
name: 'lastName',
type: 'string',
default: '',
description: 'Last name of the contact to add.',
}, },
{ {
displayName: 'Last Contacted', displayName: 'Last Contacted',
name: 'lastContacted', name: 'lastContacted',
type: 'dateTime', type: 'dateTime',
default: '', default: '',
description: 'Last contacted date of the contact to add.', description: 'Last contacted date of the contact to add',
},
{
displayName: 'Last Name',
name: 'lastName',
type: 'string',
default: '',
description: 'Last name of the contact to add',
}, },
{ {
displayName: 'Last Open', displayName: 'Last Open',
name: 'lastOpen', name: 'lastOpen',
type: 'dateTime', type: 'dateTime',
default: '', default: '',
description: 'Last opened date of the contact to add.', description: 'Last opened date of the contact to add',
}, },
{ {
displayName: 'Last Replied', displayName: 'Last Replied',
name: 'lastReplied', name: 'lastReplied',
type: 'dateTime', type: 'dateTime',
default: '', default: '',
description: 'Last replied date of the contact to add.', description: 'Last replied date of the contact to add',
}, },
{ {
displayName: 'Mails Sent', displayName: 'Mails Sent',
name: 'mailsSent', name: 'mailsSent',
type: 'number', type: 'number',
default: 0, default: 0,
description: 'Number of emails sent to the contact to add.', description: 'Number of emails sent to the contact to add',
}, },
{ {
displayName: 'Phone Number', displayName: 'Phone Number',
name: 'phoneNumber', name: 'phoneNumber',
type: 'string', type: 'string',
default: '', default: '',
description: 'Phone number of the contact to add.', description: 'Phone number of the contact to add',
}, },
], ],
}, },
@ -182,7 +183,7 @@ export const contactListFields: INodeProperties[] = [
name: 'returnAll', name: 'returnAll',
type: 'boolean', type: 'boolean',
default: false, default: false,
description: 'Return all results.', description: 'Whether to return all results or only up to a given limit',
displayOptions: { displayOptions: {
show: { show: {
resource: [ resource: [
@ -199,7 +200,7 @@ export const contactListFields: INodeProperties[] = [
name: 'limit', name: 'limit',
type: 'number', type: 'number',
default: 100, default: 100,
description: 'The number of results to return.', description: 'Max number of results to return',
typeOptions: { typeOptions: {
minValue: 1, minValue: 1,
maxValue: 100, maxValue: 100,

View file

@ -7,10 +7,12 @@ import {
ILoadOptionsFunctions, ILoadOptionsFunctions,
INodeExecutionData, INodeExecutionData,
INodeType, INodeType,
INodeTypeDescription INodeTypeDescription,
JsonObject
} from 'n8n-workflow'; } from 'n8n-workflow';
import { import {
emeliaApiTest,
emeliaGraphqlRequest, emeliaGraphqlRequest,
loadResource, loadResource,
} from './GenericFunctions'; } from './GenericFunctions';
@ -36,7 +38,7 @@ export class Emelia implements INodeType {
icon: 'file:emelia.svg', icon: 'file:emelia.svg',
group: ['input'], group: ['input'],
version: 1, version: 1,
subtitle: '={{$parameter["resource"] + ": " + $parameter["operation"]}}', subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
description: 'Consume the Emelia API', description: 'Consume the Emelia API',
defaults: { defaults: {
name: 'Emelia', name: 'Emelia',
@ -47,6 +49,7 @@ export class Emelia implements INodeType {
{ {
name: 'emeliaApi', name: 'emeliaApi',
required: true, required: true,
testedBy: 'emeliaApiTest',
}, },
], ],
properties: [ properties: [
@ -54,6 +57,7 @@ export class Emelia implements INodeType {
displayName: 'Resource', displayName: 'Resource',
name: 'resource', name: 'resource',
type: 'options', type: 'options',
noDataExpression: true,
options: [ options: [
{ {
name: 'Campaign', name: 'Campaign',
@ -66,7 +70,7 @@ export class Emelia implements INodeType {
], ],
default: 'campaign', default: 'campaign',
required: true, required: true,
description: 'The resource to operate on.', description: 'The resource to operate on',
}, },
...campaignOperations, ...campaignOperations,
...campaignFields, ...campaignFields,
@ -76,6 +80,10 @@ export class Emelia implements INodeType {
}; };
methods = { methods = {
credentialTest: {
emeliaApiTest,
},
loadOptions: { loadOptions: {
async getCampaigns(this: ILoadOptionsFunctions) { async getCampaigns(this: ILoadOptionsFunctions) {
return loadResource.call(this, 'campaign'); return loadResource.call(this, 'campaign');
@ -290,6 +298,46 @@ export class Emelia implements INodeType {
returnData.push({ success: true }); returnData.push({ success: true });
} else if (operation === 'duplicate') {
// ----------------------------------
// campaign: duplicate
// ----------------------------------
const options = this.getNodeParameter('options', i) as IDataObject;
const variables = {
fromId: this.getNodeParameter('campaignId', i),
name: this.getNodeParameter('campaignName', i),
copySettings: true,
copyMails: true,
copyContacts: false,
copyProvider: true,
...options,
};
const { data: { duplicateCampaign } } = await emeliaGraphqlRequest.call(this, {
query: `
mutation duplicateCampaign(
$fromId: ID!
$name: String!
$copySettings: Boolean!
$copyMails: Boolean!
$copyContacts: Boolean!
$copyProvider: Boolean!
) {
duplicateCampaign(
fromId: $fromId
name: $name
copySettings: $copySettings
copyMails: $copyMails
copyContacts: $copyContacts
copyProvider: $copyProvider
)
}`,
operationName: 'duplicateCampaign',
variables,
});
returnData.push({ _id: duplicateCampaign });
} }
} else if (resource === 'contactList') { } else if (resource === 'contactList') {
@ -373,7 +421,7 @@ export class Emelia implements INodeType {
} catch (error) { } catch (error) {
if (this.continueOnFail()) { if (this.continueOnFail()) {
returnData.push({ error: error.message }); returnData.push({ error: (error as JsonObject).message });
continue; continue;
} }

View file

@ -10,6 +10,7 @@ import {
import { import {
emeliaApiRequest, emeliaApiRequest,
emeliaApiTest,
emeliaGraphqlRequest, emeliaGraphqlRequest,
} from './GenericFunctions'; } from './GenericFunctions';
@ -23,6 +24,7 @@ export class EmeliaTrigger implements INodeType {
displayName: 'Emelia Trigger', displayName: 'Emelia Trigger',
name: 'emeliaTrigger', name: 'emeliaTrigger',
icon: 'file:emelia.svg', icon: 'file:emelia.svg',
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
group: ['trigger'], group: ['trigger'],
version: 1, version: 1,
description: 'Handle Emelia campaign activity events via webhooks', description: 'Handle Emelia campaign activity events via webhooks',
@ -35,6 +37,7 @@ export class EmeliaTrigger implements INodeType {
{ {
name: 'emeliaApi', name: 'emeliaApi',
required: true, required: true,
testedBy: 'emeliaApiTest',
}, },
], ],
webhooks: [ webhooks: [
@ -93,6 +96,10 @@ export class EmeliaTrigger implements INodeType {
}; };
methods = { methods = {
credentialTest: {
emeliaApiTest,
},
loadOptions: { loadOptions: {
async getCampaigns(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> { async getCampaigns(this: ILoadOptionsFunctions): Promise<INodePropertyOptions[]> {
const responseData = await emeliaGraphqlRequest.call(this, { const responseData = await emeliaGraphqlRequest.call(this, {

View file

@ -4,8 +4,12 @@ import {
} from 'n8n-core'; } from 'n8n-core';
import { import {
ICredentialsDecrypted,
ICredentialTestFunctions,
IHookFunctions, IHookFunctions,
INodeCredentialTestResult,
INodePropertyOptions, INodePropertyOptions,
JsonObject,
NodeApiError, NodeApiError,
} from 'n8n-workflow'; } from 'n8n-workflow';
@ -51,7 +55,7 @@ export async function emeliaApiRequest(
try { try {
return await this.helpers.request!.call(this, options); return await this.helpers.request!.call(this, options);
} catch (error) { } catch (error) {
throw new NodeApiError(this.getNode(), error); throw new NodeApiError(this.getNode(), (error as JsonObject));
} }
} }
@ -91,5 +95,47 @@ export async function loadResource(
name: campaign.name, name: campaign.name,
value: campaign._id, value: campaign._id,
})); }));
}
export async function emeliaApiTest(this: ICredentialTestFunctions, credential: ICredentialsDecrypted): Promise<INodeCredentialTestResult> {
const credentials = credential.data;
const body = {
query: `
query all_campaigns {
all_campaigns {
_id
name
status
createdAt
stats {
mailsSent
}
}
}`,
operationName: 'all_campaigns',
};
const options = {
headers: {
Authorization: credentials?.apiKey,
},
method: 'POST',
body,
uri: `https://graphql.emelia.io/graphql`,
json: true,
};
try {
await this.helpers.request!(options);
} catch (error) {
return {
status: 'Error',
message: `Connection details not valid: ${(error as JsonObject).message}`,
};
}
return {
status: 'OK',
message: 'Authentication successful!',
};
} }