Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
const _ = require('lodash');
|
2019-11-05 12:14:54 +07:00
|
|
|
const template = require('./template');
|
2021-06-30 14:56:57 +01:00
|
|
|
const settingsCache = require('../../../shared/settings-cache');
|
2020-05-28 05:57:02 -05:00
|
|
|
const urlUtils = require('../../../shared/url-utils');
|
2020-05-26 17:46:45 +01:00
|
|
|
const moment = require('moment-timezone');
|
2022-04-06 13:36:12 +01:00
|
|
|
const api = require('../../api').endpoints;
|
|
|
|
const apiShared = require('../../api').shared;
|
2019-11-27 13:18:44 +05:30
|
|
|
const {URL} = require('url');
|
2020-04-17 10:22:53 +01:00
|
|
|
const mobiledocLib = require('../../lib/mobiledoc');
|
|
|
|
const htmlToText = require('html-to-text');
|
2021-05-14 11:57:29 +01:00
|
|
|
const {isUnsplashImage, isLocalContentImage} = require('@tryghost/kg-default-cards/lib/utils');
|
2021-07-29 15:39:04 +01:00
|
|
|
const {textColorForBackgroundColor, darkenToContrastThreshold} = require('@tryghost/color-utils');
|
2021-06-15 15:36:27 +01:00
|
|
|
const logging = require('@tryghost/logging');
|
2019-11-05 12:14:54 +07:00
|
|
|
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
const ALLOWED_REPLACEMENTS = ['first_name'];
|
|
|
|
|
2021-09-17 08:39:29 +01:00
|
|
|
// Format a full html document ready for email by inlining CSS, adjusting links,
|
|
|
|
// and performing any client-specific fixes
|
|
|
|
const formatHtmlForEmail = function formatHtmlForEmail(html) {
|
|
|
|
const juiceOptions = {inlinePseudoElements: true};
|
|
|
|
|
2021-10-21 10:27:56 +02:00
|
|
|
const juice = require('juice');
|
2021-09-17 08:39:29 +01:00
|
|
|
let juicedHtml = juice(html, juiceOptions);
|
|
|
|
|
|
|
|
// convert juiced HTML to a DOM-like interface for further manipulation
|
|
|
|
// happens after inlining of CSS so we can change element types without worrying about styling
|
2021-11-18 14:44:19 +01:00
|
|
|
|
|
|
|
const cheerio = require('cheerio');
|
2021-09-17 08:39:29 +01:00
|
|
|
const _cheerio = cheerio.load(juicedHtml);
|
|
|
|
|
|
|
|
// force all links to open in new tab
|
|
|
|
_cheerio('a').attr('target', '_blank');
|
|
|
|
// convert figure and figcaption to div so that Outlook applies margins
|
|
|
|
_cheerio('figure, figcaption').each((i, elem) => !!(elem.tagName = 'div'));
|
|
|
|
|
|
|
|
juicedHtml = _cheerio.html();
|
|
|
|
|
|
|
|
// Fix any unsupported chars in Outlook
|
|
|
|
juicedHtml = juicedHtml.replace(/'/g, ''');
|
|
|
|
|
|
|
|
return juicedHtml;
|
|
|
|
};
|
|
|
|
|
2019-11-05 12:14:54 +07:00
|
|
|
const getSite = () => {
|
2020-03-12 09:52:33 +05:30
|
|
|
const publicSettings = settingsCache.getPublic();
|
|
|
|
return Object.assign({}, publicSettings, {
|
|
|
|
url: urlUtils.urlFor('home', true),
|
|
|
|
iconUrl: publicSettings.icon ? urlUtils.urlFor('image', {image: publicSettings.icon}, true) : null
|
2019-11-05 12:14:54 +07:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2021-07-01 13:36:36 +02:00
|
|
|
const htmlToPlaintext = (html) => {
|
|
|
|
// same options as used in Post model for generating plaintext but without `wordwrap: 80`
|
|
|
|
// to avoid replacement strings being split across lines and for mail clients to handle
|
|
|
|
// word wrapping based on user preferences
|
|
|
|
return htmlToText.fromString(html, {
|
|
|
|
wordwrap: false,
|
|
|
|
ignoreImage: true,
|
|
|
|
hideLinkHrefIfSameAsText: true,
|
|
|
|
preserveNewlines: true,
|
|
|
|
returnDomByDefault: true,
|
|
|
|
uppercaseHeadings: false
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-11-26 21:37:04 +05:30
|
|
|
/**
|
|
|
|
* createUnsubscribeUrl
|
|
|
|
*
|
2019-11-27 10:58:21 +05:30
|
|
|
* Takes a member uuid and returns the url that should be used to unsubscribe
|
|
|
|
* In case of no member uuid, generates the preview unsubscribe url - `?preview=1`
|
2019-11-26 21:37:04 +05:30
|
|
|
*
|
2019-11-27 10:58:21 +05:30
|
|
|
* @param {string} uuid
|
2019-11-26 21:37:04 +05:30
|
|
|
*/
|
2019-11-27 10:58:21 +05:30
|
|
|
const createUnsubscribeUrl = (uuid) => {
|
2019-11-26 21:37:04 +05:30
|
|
|
const siteUrl = urlUtils.getSiteUrl();
|
|
|
|
const unsubscribeUrl = new URL(siteUrl);
|
|
|
|
unsubscribeUrl.pathname = `${unsubscribeUrl.pathname}/unsubscribe/`.replace('//', '/');
|
2019-11-27 10:58:21 +05:30
|
|
|
if (uuid) {
|
|
|
|
unsubscribeUrl.searchParams.set('uuid', uuid);
|
2019-11-26 21:37:04 +05:30
|
|
|
} else {
|
|
|
|
unsubscribeUrl.searchParams.set('preview', '1');
|
|
|
|
}
|
|
|
|
|
|
|
|
return unsubscribeUrl.href;
|
|
|
|
};
|
|
|
|
|
2022-04-08 12:27:43 +01:00
|
|
|
// NOTE: serialization is needed to make sure we do post transformations such as image URL transformation from relative to absolute
|
|
|
|
const serializePostModel = async (model) => {
|
2020-04-17 10:22:53 +01:00
|
|
|
// fetch mobiledoc rather than html and plaintext so we can render email-specific contents
|
|
|
|
const frame = {options: {context: {user: true}, formats: 'mobiledoc'}};
|
2019-11-26 21:37:04 +05:30
|
|
|
const docName = 'posts';
|
|
|
|
|
2022-04-06 13:36:12 +01:00
|
|
|
await apiShared
|
2019-11-26 21:37:04 +05:30
|
|
|
.serializers
|
|
|
|
.handle
|
2022-04-08 12:27:43 +01:00
|
|
|
.output(model, {docName: docName, method: 'read'}, api.serializers.output, frame);
|
2019-11-26 21:37:04 +05:30
|
|
|
|
|
|
|
return frame.response[docName][0];
|
|
|
|
};
|
|
|
|
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
// removes %% wrappers from unknown replacement strings in email content
|
|
|
|
const normalizeReplacementStrings = (email) => {
|
|
|
|
// we don't want to modify the email object in-place
|
|
|
|
const emailContent = _.pick(email, ['html', 'plaintext']);
|
|
|
|
|
2020-04-20 12:24:05 +01:00
|
|
|
const EMAIL_REPLACEMENT_REGEX = /%%(\{.*?\})%%/g;
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
const REPLACEMENT_STRING_REGEX = /\{(?<recipientProperty>\w*?)(?:,? *(?:"|")(?<fallback>.*?)(?:"|"))?\}/;
|
|
|
|
|
|
|
|
['html', 'plaintext'].forEach((format) => {
|
|
|
|
emailContent[format] = emailContent[format].replace(EMAIL_REPLACEMENT_REGEX, (replacementMatch, replacementStr) => {
|
|
|
|
const match = replacementStr.match(REPLACEMENT_STRING_REGEX);
|
|
|
|
|
|
|
|
if (match) {
|
|
|
|
const {recipientProperty} = match.groups;
|
|
|
|
|
|
|
|
if (ALLOWED_REPLACEMENTS.includes(recipientProperty)) {
|
|
|
|
// keeps wrapping %% for later replacement with real data
|
|
|
|
return replacementMatch;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// removes %% so output matches user supplied content
|
|
|
|
return replacementStr;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return emailContent;
|
|
|
|
};
|
|
|
|
|
2021-06-23 12:00:03 +04:00
|
|
|
/**
|
|
|
|
* Parses email content and extracts an array of replacements with desired fallbacks
|
|
|
|
*
|
|
|
|
* @param {Object} email
|
|
|
|
* @param {string} email.html
|
|
|
|
* @param {string} email.plaintext
|
|
|
|
*
|
|
|
|
* @returns {Object[]} replacements
|
|
|
|
*/
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
const parseReplacements = (email) => {
|
|
|
|
const EMAIL_REPLACEMENT_REGEX = /%%(\{.*?\})%%/g;
|
|
|
|
const REPLACEMENT_STRING_REGEX = /\{(?<recipientProperty>\w*?)(?:,? *(?:"|")(?<fallback>.*?)(?:"|"))?\}/;
|
2020-04-20 12:24:05 +01:00
|
|
|
|
|
|
|
const replacements = [];
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
|
2020-04-20 12:24:05 +01:00
|
|
|
['html', 'plaintext'].forEach((format) => {
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
let result;
|
|
|
|
while ((result = EMAIL_REPLACEMENT_REGEX.exec(email[format])) !== null) {
|
|
|
|
const [replacementMatch, replacementStr] = result;
|
2020-04-20 12:24:05 +01:00
|
|
|
const match = replacementStr.match(REPLACEMENT_STRING_REGEX);
|
|
|
|
|
|
|
|
if (match) {
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
const {recipientProperty, fallback} = match.groups;
|
2020-04-20 12:24:05 +01:00
|
|
|
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
if (ALLOWED_REPLACEMENTS.includes(recipientProperty)) {
|
2020-04-20 12:24:05 +01:00
|
|
|
const id = `replacement_${replacements.length + 1}`;
|
|
|
|
|
|
|
|
replacements.push({
|
|
|
|
format,
|
|
|
|
id,
|
|
|
|
match: replacementMatch,
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
recipientProperty: `member_${recipientProperty}`,
|
2020-04-20 12:24:05 +01:00
|
|
|
fallback
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
}
|
2020-04-20 12:24:05 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
return replacements;
|
|
|
|
};
|
|
|
|
|
2022-04-27 19:48:36 +02:00
|
|
|
const getTemplateSettings = async (newsletter) => {
|
2021-07-29 15:39:04 +01:00
|
|
|
const accentColor = settingsCache.get('accent_color');
|
|
|
|
const adjustedAccentColor = accentColor && darkenToContrastThreshold(accentColor, '#ffffff', 2).hex();
|
|
|
|
const adjustedAccentContrastColor = accentColor && textColorForBackgroundColor(adjustedAccentColor).hex();
|
|
|
|
|
2021-06-07 10:02:35 +01:00
|
|
|
const templateSettings = {
|
2022-04-26 13:01:27 +02:00
|
|
|
headerImage: newsletter.get('header_image'),
|
|
|
|
showHeaderIcon: newsletter.get('show_header_icon') && settingsCache.get('icon'),
|
|
|
|
showHeaderTitle: newsletter.get('show_header_title'),
|
|
|
|
showFeatureImage: newsletter.get('show_feature_image'),
|
|
|
|
titleFontCategory: newsletter.get('title_font_category'),
|
|
|
|
titleAlignment: newsletter.get('title_alignment'),
|
|
|
|
bodyFontCategory: newsletter.get('body_font_category'),
|
|
|
|
showBadge: newsletter.get('show_badge'),
|
|
|
|
footerContent: newsletter.get('footer_content'),
|
|
|
|
showHeaderName: newsletter.get('show_header_name'),
|
2021-07-29 15:39:04 +01:00
|
|
|
accentColor,
|
|
|
|
adjustedAccentColor,
|
|
|
|
adjustedAccentContrastColor
|
2021-06-07 10:02:35 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
if (templateSettings.headerImage) {
|
|
|
|
if (isUnsplashImage(templateSettings.headerImage)) {
|
|
|
|
// Unsplash images have a minimum size so assuming 1200px is safe
|
|
|
|
const unsplashUrl = new URL(templateSettings.headerImage);
|
2021-06-18 15:37:42 +01:00
|
|
|
unsplashUrl.searchParams.set('w', '1200');
|
2021-06-07 10:02:35 +01:00
|
|
|
|
|
|
|
templateSettings.headerImage = unsplashUrl.href;
|
|
|
|
templateSettings.headerImageWidth = 600;
|
|
|
|
} else {
|
|
|
|
const {imageSize} = require('../../lib/image');
|
|
|
|
try {
|
|
|
|
const size = await imageSize.getImageSizeFromUrl(templateSettings.headerImage);
|
|
|
|
|
|
|
|
if (size.width >= 600) {
|
|
|
|
// keep original image, just set a fixed width
|
|
|
|
templateSettings.headerImageWidth = 600;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isLocalContentImage(templateSettings.headerImage, urlUtils.getSiteUrl())) {
|
|
|
|
// we can safely request a 1200px image - Ghost will serve the original if it's smaller
|
|
|
|
templateSettings.headerImage = templateSettings.headerImage.replace(/\/content\/images\//, '/content/images/size/w1200/');
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
// log and proceed. Using original header image without fixed width isn't fatal.
|
|
|
|
logging.error(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return templateSettings;
|
|
|
|
};
|
|
|
|
|
2022-04-08 12:27:43 +01:00
|
|
|
const serialize = async (postModel, newsletter, options = {isBrowserPreview: false}) => {
|
|
|
|
const post = await serializePostModel(postModel);
|
2020-04-17 10:22:53 +01:00
|
|
|
|
2020-06-22 23:21:00 +12:00
|
|
|
const timezone = settingsCache.get('timezone');
|
2020-04-24 00:40:08 +01:00
|
|
|
const momentDate = post.published_at ? moment(post.published_at) : moment();
|
|
|
|
post.published_at = momentDate.tz(timezone).format('DD MMM YYYY');
|
|
|
|
|
2021-10-22 14:40:42 +01:00
|
|
|
post.authors = post.authors && post.authors.map(author => author.name).join(', ');
|
2019-11-06 18:32:11 +07:00
|
|
|
if (post.posts_meta) {
|
|
|
|
post.email_subject = post.posts_meta.email_subject;
|
|
|
|
}
|
2020-08-12 20:14:06 +01:00
|
|
|
|
|
|
|
// we use post.excerpt as a hidden piece of text that is picked up by some email
|
|
|
|
// clients as a "preview" when listing emails. Our current plaintext/excerpt
|
|
|
|
// generation outputs links as "Link [https://url/]" which isn't desired in the preview
|
2020-08-24 19:46:46 +12:00
|
|
|
if (!post.custom_excerpt && post.excerpt) {
|
2020-08-12 20:14:06 +01:00
|
|
|
post.excerpt = post.excerpt.replace(/\s\[http(.*?)\]/g, '');
|
|
|
|
}
|
|
|
|
|
2021-12-10 00:04:53 +05:30
|
|
|
post.html = mobiledocLib.mobiledocHtmlRenderer.render(
|
|
|
|
JSON.parse(post.mobiledoc), {target: 'email', postUrl: post.url}
|
|
|
|
);
|
2021-08-24 19:38:29 +01:00
|
|
|
|
|
|
|
// perform any email specific adjustments to the mobiledoc->HTML render output
|
2021-08-25 09:25:55 +01:00
|
|
|
// body wrapper is required so we can get proper top-level selections
|
2021-11-18 14:44:19 +01:00
|
|
|
const cheerio = require('cheerio');
|
2021-08-25 09:25:55 +01:00
|
|
|
let _cheerio = cheerio.load(`<body>${post.html}</body>`);
|
2021-08-24 19:38:29 +01:00
|
|
|
// remove leading/trailing HRs
|
2021-08-25 09:25:55 +01:00
|
|
|
_cheerio(`
|
|
|
|
body > hr:first-child,
|
|
|
|
body > hr:last-child,
|
|
|
|
body > div:first-child > hr:first-child,
|
|
|
|
body > div:last-child > hr:last-child
|
|
|
|
`).remove();
|
|
|
|
post.html = _cheerio('body').html();
|
2021-08-24 19:38:29 +01:00
|
|
|
|
2021-07-01 13:36:36 +02:00
|
|
|
post.plaintext = htmlToPlaintext(post.html);
|
2020-04-17 10:22:53 +01:00
|
|
|
|
2021-05-14 11:57:29 +01:00
|
|
|
// Outlook will render feature images at full-size breaking the layout.
|
|
|
|
// Content images fix this by rendering max 600px images - do the same for feature image here
|
2021-06-01 17:07:19 +01:00
|
|
|
if (post.feature_image) {
|
|
|
|
if (isUnsplashImage(post.feature_image)) {
|
|
|
|
// Unsplash images have a minimum size so assuming 1200px is safe
|
|
|
|
const unsplashUrl = new URL(post.feature_image);
|
2021-06-18 15:37:42 +01:00
|
|
|
unsplashUrl.searchParams.set('w', '1200');
|
2021-06-01 17:07:19 +01:00
|
|
|
|
|
|
|
post.feature_image = unsplashUrl.href;
|
|
|
|
post.feature_image_width = 600;
|
|
|
|
} else {
|
|
|
|
const {imageSize} = require('../../lib/image');
|
|
|
|
try {
|
|
|
|
const size = await imageSize.getImageSizeFromUrl(post.feature_image);
|
|
|
|
|
|
|
|
if (size.width >= 600) {
|
|
|
|
// keep original image, just set a fixed width
|
|
|
|
post.feature_image_width = 600;
|
|
|
|
}
|
2021-05-14 11:57:29 +01:00
|
|
|
|
2021-06-01 17:07:19 +01:00
|
|
|
if (isLocalContentImage(post.feature_image, urlUtils.getSiteUrl())) {
|
|
|
|
// we can safely request a 1200px image - Ghost will serve the original if it's smaller
|
|
|
|
post.feature_image = post.feature_image.replace(/\/content\/images\//, '/content/images/size/w1200/');
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
// log and proceed. Using original feature_image without fixed width isn't fatal.
|
|
|
|
logging.error(err);
|
2021-05-14 11:57:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-27 19:48:36 +02:00
|
|
|
const templateSettings = await getTemplateSettings(newsletter);
|
2021-06-07 10:02:35 +01:00
|
|
|
|
2021-09-02 13:11:11 +01:00
|
|
|
const render = template;
|
2021-07-29 15:25:09 +01:00
|
|
|
|
2022-04-27 19:48:36 +02:00
|
|
|
let htmlTemplate = render({post, site: getSite(), templateSettings, newsletter: newsletter.toJSON()});
|
2021-06-07 10:02:35 +01:00
|
|
|
|
2019-11-26 21:37:04 +05:30
|
|
|
if (options.isBrowserPreview) {
|
2021-06-18 15:37:42 +01:00
|
|
|
const previewUnsubscribeUrl = createUnsubscribeUrl(null);
|
2019-11-26 21:37:04 +05:30
|
|
|
htmlTemplate = htmlTemplate.replace('%recipient.unsubscribe_url%', previewUnsubscribeUrl);
|
|
|
|
}
|
2020-04-17 10:22:53 +01:00
|
|
|
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
// Clean up any unknown replacements strings to get our final content
|
|
|
|
const {html, plaintext} = normalizeReplacementStrings({
|
2021-09-17 08:39:29 +01:00
|
|
|
html: formatHtmlForEmail(htmlTemplate),
|
2019-11-05 15:04:48 +07:00
|
|
|
plaintext: post.plaintext
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
});
|
2020-04-20 12:24:05 +01:00
|
|
|
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
return {
|
|
|
|
subject: post.email_subject || post.title,
|
|
|
|
html,
|
|
|
|
plaintext
|
|
|
|
};
|
2019-11-05 12:14:54 +07:00
|
|
|
};
|
|
|
|
|
2021-07-01 13:36:36 +02:00
|
|
|
function renderEmailForSegment(email, memberSegment) {
|
2021-11-18 14:44:19 +01:00
|
|
|
const cheerio = require('cheerio');
|
|
|
|
|
2021-07-01 13:36:36 +02:00
|
|
|
const result = {...email};
|
|
|
|
const $ = cheerio.load(result.html);
|
|
|
|
|
|
|
|
$('[data-gh-segment]').get().forEach((node) => {
|
|
|
|
if (node.attribs['data-gh-segment'] !== memberSegment) { //TODO: replace with NQL interpretation
|
|
|
|
$(node).remove();
|
|
|
|
} else {
|
|
|
|
// Getting rid of the attribute for a cleaner html output
|
|
|
|
$(node).removeAttr('data-gh-segment');
|
|
|
|
}
|
|
|
|
});
|
2021-09-17 08:39:29 +01:00
|
|
|
|
|
|
|
result.html = formatHtmlForEmail($.html());
|
2021-07-01 13:36:36 +02:00
|
|
|
result.plaintext = htmlToPlaintext(result.html);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2019-11-05 12:14:54 +07:00
|
|
|
module.exports = {
|
2019-11-26 21:37:04 +05:30
|
|
|
serialize,
|
Refactor mega service to use stored email content and batch/recipient records
no issue
- store raw content in email record
- keep any replacement strings in the html/plaintext content so that it can be used when sending email rather than needing to re-serialize the post content which may have changed
- split post email serializer into separate serialization and replacement parsing functions
- serialization now returns any email content that is derived from the post content (subject/html/plaintext) rather than post content plus replacements
- `parseReplacements` has been split out so that it can be run against email content rather than a post, this allows mega and the email preview service to work with the stored email content
- move mailgun-specific functionality into the mailgun provider
- previously mailgun-specific behaviour was spread across the post email serializer, mega, and bulk-email service
- the per-batch `send` functionality was moved from the `bulk-email` service to the mailgun provider and updated to take email content, recipient info, and replacement info so that all mailgun-specific headers and replacement formatting can be handled in one place
- exposes the `BATCH_SIZE` constant because batch sizes are limited to what the provider allows
- `bulk-email` service split into three methods
- `send` responsible for taking email content and recipients, parsing replacement info from the email content and using that to collate a recipient data object, and finally coordinating the send via the mailgun provider. Usable directly for use-cases such as test emails
- `processEmail` takes an email ID, loads it and coordinates sending related batches concurrently
- `processEmailBatch` takes an email_batch ID, loads it along with associated email_recipient records and passes the data through to the `send` function, updating the batch status as it's processed
- `processEmail` and `processEmailBatch` take IDs rather than objects ready for future use by job-queues, it's best to keep job parameters as minimal as possible
- refactored `mega` service
- modified `getEmailData` to collate email content (from/reply-to/subject/html/plaintext) rather than being responsible for dealing with replacements and mailgun-specific replacement formats
- used for generating email content before storing in the email table, and when sending test emails
- from/reply-to calculation moved out of the post-email-serializer into mega and extracted into separate functions used by `getEmailData`
- `sendTestEmail` updated to generate `EmailRecipient`-like objects for each email address so that appropriate data can be supplied to the updated `bulk-email.send` method
- `sendEmailJob` updated to create `email_batches` and associated `email_recipients` records then hand over processing to the `bulk-email` service
- member row fetching extracted into a separate function and used by `createEmailBatches`
- moved updating of email status from `mega` to the `bulk-email` service, keeps concept of Successful/FailedBatch internal to the `bulk-email` service
2020-09-24 09:35:29 +01:00
|
|
|
createUnsubscribeUrl,
|
2021-07-01 13:36:36 +02:00
|
|
|
renderEmailForSegment,
|
2022-04-26 13:01:27 +02:00
|
|
|
parseReplacements,
|
|
|
|
// Export for tests
|
|
|
|
_getTemplateSettings: getTemplateSettings
|
2019-11-05 12:14:54 +07:00
|
|
|
};
|