Skip to content

Commit

Permalink
Merge branch 'master' into develop
Browse files Browse the repository at this point in the history
  • Loading branch information
emile-bex committed Nov 20, 2023
2 parents 09ad7a8 + 174dca6 commit 031ff6f
Showing 6 changed files with 93 additions and 412 deletions.
5 changes: 2 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "linkedout-backend",
"version": "2.17.1",
"version": "2.17.2",
"license": "ISC",
"engines": {
"node": "16.x"
@@ -59,7 +59,7 @@
"moment": "^2.29.4",
"moment-timezone": "^0.5.43",
"node-fetch": "^2.6.7",
"node-mailjet": "^3.3.9",
"node-mailjet": "^6.0.4",
"passport": "^0.6.0",
"passport-jwt": "^4.0.0",
"passport-local": "^1.0.0",
@@ -103,7 +103,6 @@
"@types/multer": "^1.4.7",
"@types/node": "^16.0.0",
"@types/node-fetch": "2.x",
"@types/node-mailjet": "^3.3.9",
"@types/passport-jwt": "^3.0.6",
"@types/passport-local": "^1.0.34",
"@types/puppeteer": "^5.4.6",
157 changes: 38 additions & 119 deletions src/external-services/mailjet/mailjet.service.ts
Original file line number Diff line number Diff line change
@@ -1,46 +1,35 @@
import { Injectable } from '@nestjs/common';
import { connect, Email } from 'node-mailjet';
import { Client, ContactSubscription, SendEmailV3_1 } from 'node-mailjet';

import {
CustomContactParams,
CustomMailParams,
MailjetContactList,
MailjetContactTag,
MailjetContactProperties,
MailjetContactTagNames,
MailjetCustomContact,
MailjetCustomResponse,
MailjetListActions,
MailjetOptions,
} from './mailjet.types';

import { createMail } from './mailjet.utils';
import SendParams = Email.SendParams;
import SendParamsRecipient = Email.SendParamsRecipient;

@Injectable()
export class MailjetService {
/*
private mailjetTransactional: Email.Client;
private mailjetNewsletter: Email.Client;
*/

/*
constructor() {
this.mailjetTransactional = this.getTransactionalInstance();
this.mailjetNewsletter = this.getNewsletterInstance();
}
*/

getTransactionalInstance() {
return connect(`${process.env.MAILJET_PUB}`, `${process.env.MAILJET_SEC}`);
}

getNewsletterInstance() {
return connect(
`${process.env.MAILJET_NEWSLETTER_PUB}`,
`${process.env.MAILJET_NEWSLETTER_SEC}`
);
private mailjetTransactional: Client;
private mailjetNewsletter: Client;

constructor() {
this.mailjetTransactional = new Client({
apiKey: `${process.env.MAILJET_PUB}`,
apiSecret: `${process.env.MAILJET_SEC}`,
});
this.mailjetNewsletter = new Client({
apiKey: `${process.env.MAILJET_NEWSLETTER_PUB}`,
apiSecret: `${process.env.MAILJET_NEWSLETTER_SEC}`,
});
}

async sendMail(params: CustomMailParams | CustomMailParams[]) {
const mailjetParams: SendParams = { Messages: [] };
const mailjetParams: SendEmailV3_1.Body = { Messages: [] };
if (Array.isArray(params)) {
mailjetParams.Messages = params.map((p) => {
return createMail(p);
@@ -49,109 +38,39 @@ export class MailjetService {
mailjetParams.Messages = [createMail(params)];
}

return await this.getTransactionalInstance()
.post('send', {
version: 'v3.1',
})
return this.mailjetTransactional
.post('send', MailjetOptions.MAILS)
.request(mailjetParams);
}

async findContact(
email: CustomContactParams['email']
): Promise<MailjetCustomContact> {
const contact: SendParamsRecipient = {
Email: email.toLowerCase(),
};

const {
body: { Data: data },
} = (await this.getNewsletterInstance()
.get('contact', { version: 'v3' })
.request(contact)) as MailjetCustomResponse;

return data[0];
}

async createContact(
email: CustomContactParams['email']
): Promise<MailjetCustomContact> {
const contact: SendParamsRecipient = {
Email: email,
};
const {
body: { Data: data },
} = (await this.getNewsletterInstance()
.post('contact', { version: 'v3' })
.request(contact)) as MailjetCustomResponse;

return data[0];
}

async updateContactTags(
id: string,
{ zone, status }: Pick<CustomContactParams, 'zone' | 'status'>
) {
let dataToUpdate: { Data: MailjetContactTag[] } = {
Data: [
{
Name: MailjetContactTagNames.NEWSLETTER,
Value: true,
},
],
async sendContact({ email, zone, status }: CustomContactParams) {
let contactProperties: Partial<MailjetContactProperties> = {
[MailjetContactTagNames.NEWSLETTER]: true,
};

if (zone) {
dataToUpdate = {
Data: [
...dataToUpdate.Data,
{
Name: MailjetContactTagNames.ZONE,
Value: zone,
},
],
contactProperties = {
...contactProperties,
[MailjetContactTagNames.ZONE]: zone,
};
}
if (status) {
dataToUpdate = {
Data: [
...dataToUpdate.Data,
{
Name: MailjetContactTagNames.STATUS,
Value: status,
},
],
contactProperties = {
...contactProperties,
[MailjetContactTagNames.STATUS]: status,
};
}

return await this.getNewsletterInstance()
.put('contactdata', { version: 'v3' })
.id(id)
.request(dataToUpdate);
}

async subscribeToNewsletterList(id: string) {
const dataToUpdate: { ContactsLists: MailjetContactList[] } = {
ContactsLists: [
{
Action: MailjetListActions.FORCE,
ListID: process.env.MAILJET_NEWSLETTER_LIST_ID,
},
],
const contact: ContactSubscription.PostContactsListManageContactBody = {
Properties: contactProperties,
Action: MailjetListActions.FORCE,
Email: email,
};

return await this.getNewsletterInstance()
.post('contact', { version: 'v3' })
.id(id)
.action('managecontactslists')
.request(dataToUpdate);
}

async sendContact({ email, zone, status }: CustomContactParams) {
let contact = await this.findContact(email);
if (!contact) {
contact = await this.createContact(email);
}
await this.subscribeToNewsletterList(contact.ID);
await this.updateContactTags(contact.ID, { zone, status });
return this.mailjetNewsletter
.post('contactslist', MailjetOptions.CONTACTS)
.id(parseInt(process.env.MAILJET_NEWSLETTER_LIST_ID))
.action('managecontact')
.request(contact);
}
}
44 changes: 17 additions & 27 deletions src/external-services/mailjet/mailjet.types.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,12 @@
import { BulkContactManagement } from 'node-mailjet';
import { RequestConstructorConfig } from 'node-mailjet/declarations/request/Request';
import { AdminZone } from 'src/utils/types';

export const MailjetOptions: { [K in string]: RequestConstructorConfig } = {
MAILS: { version: 'v3.1' },
CONTACTS: { version: 'v3' },
} as const;

export interface CustomMailParams {
toEmail:
| string
@@ -29,25 +36,21 @@ export const MailjetContactTagNames = {
STATUS: 'profil_linkedout',
} as const;

export type MailjetContactTagName =
typeof MailjetContactTagNames[keyof typeof MailjetContactTagNames];

export interface MailjetContactTag {
Name: MailjetContactTagName;
Value: string | boolean;
}

export const MailjetListActions = {
const MailjetListActionsValues = {
NO_FORCE: 'addnoforce',
FORCE: 'addforce',
} as const;

export type MailjetListAction =
typeof MailjetListActions[keyof typeof MailjetListActions];
// Hack because error when getting ManageContactsAction enum value
export const MailjetListActions = MailjetListActionsValues as Record<
keyof typeof MailjetListActionsValues,
BulkContactManagement.ManageContactsAction
>;

export interface MailjetContactList {
Action: MailjetListAction;
ListID: string;
export interface MailjetContactProperties {
[MailjetContactTagNames.NEWSLETTER]: boolean;
[MailjetContactTagNames.ZONE]: AdminZone;
[MailjetContactTagNames.STATUS]: ContactStatus;
}

export const MailjetTemplates = {
@@ -97,16 +100,3 @@ export const ContactStatuses = {

export type ContactStatus =
typeof ContactStatuses[keyof typeof ContactStatuses];

export interface MailjetError {
statusCode: number;
ErrorMessage: string;
}

export interface MailjetCustomContact {
ID: string;
}

export interface MailjetCustomResponse {
body: { Data: ReadonlyArray<MailjetCustomContact> };
}
5 changes: 2 additions & 3 deletions src/external-services/mailjet/mailjet.utils.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
import _ from 'lodash';
import { Email } from 'node-mailjet';
import { SendEmailV3_1 } from 'node-mailjet';
import { CustomMailParams, MailjetTemplates } from './mailjet.types';
import SendParams = Email.SendParams;

const useCampaigns = process.env.MAILJET_CAMPAIGNS_ACTIVATED === 'true';

@@ -14,7 +13,7 @@ export function createMail({
variables,
templateId,
}: CustomMailParams) {
const recipients: SendParams['Messages'][number] = {
const recipients: SendEmailV3_1.Body['Messages'][number] = {
To: [],
Cc: [],
From: { Email: '' },
6 changes: 0 additions & 6 deletions tests/mocks.types.ts
Original file line number Diff line number Diff line change
@@ -103,11 +103,5 @@ export const SalesforceMocks: ProviderMock<SalesforceService> = {

export const MailjetMock: ProviderMock<MailjetService> = {
sendMail: jest.fn(),
findContact: jest.fn(),
createContact: jest.fn(),
updateContactTags: jest.fn(),
subscribeToNewsletterList: jest.fn(),
sendContact: jest.fn(),
getNewsletterInstance: jest.fn(),
getTransactionalInstance: jest.fn(),
} as const;
Loading

0 comments on commit 031ff6f

Please sign in to comment.