mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-06 22:40:14 -05:00
Added email card and replacement handling to member emails
no issue - adjusted mega's post serializer to get full email contents - fetch `mobiledoc` from the API rather than the pre-rendered `html` and `plaintext` - re-generate `html` using the mobiledoc renderer with an "email" target so that the email-only card content is included - re-generate `plaintext` from the newly generated email html - added replacement handling to mega's `getEmailData` function - find all of our `%%{replacement "fallback"}%%` instances in the html template and push them into a replacements array with the respective property on the member instance and desired fallback - transform the replacement for Mailgun compatibility. Mailgun uses `%recipient.variable_name%` for its template variables so we need to replace our custom replacement string with the compatible version. Our replacements system allows for the same replacement (`{subscriber_name}`) to be used multiple times and have different fallbacks, Mailgun doesn't support fallbacks so for each replacement we also need an indexed `variable_name` part so that we can put our fallbacks in the correct place - perform the same Mailgun template transformation for the plaintext version except we re-use the replacements array to avoid bloating the API request to Mailgun with duplicate template variables for every recipient - swapped `reduce` for a plain loop for easier readability
This commit is contained in:
parent
cdaa1b5dbb
commit
a801352c7f
2 changed files with 88 additions and 12 deletions
|
@ -8,19 +8,76 @@ const models = require('../../models');
|
||||||
const postEmailSerializer = require('./post-email-serializer');
|
const postEmailSerializer = require('./post-email-serializer');
|
||||||
const config = require('../../config');
|
const config = require('../../config');
|
||||||
|
|
||||||
const getEmailData = async (postModel, recipients = []) => {
|
const getEmailData = async (postModel, members = []) => {
|
||||||
const emailTmpl = await postEmailSerializer.serialize(postModel);
|
const emailTmpl = await postEmailSerializer.serialize(postModel);
|
||||||
emailTmpl.from = membersService.config.getEmailFromAddress();
|
emailTmpl.from = membersService.config.getEmailFromAddress();
|
||||||
|
|
||||||
const emails = recipients.map(recipient => recipient.email);
|
const EMAIL_REPLACEMENT_REGEX = /%%(\{.*?\})%%/g;
|
||||||
const emailData = recipients.reduce((emailData, recipient) => {
|
// the " is necessary here because `juice` will convert "->" for email compatibility
|
||||||
return Object.assign({
|
const REPLACEMENT_STRING_REGEX = /\{(?<memberProp>\w*?)(?:,? *(?:"|")(?<fallback>.*?)(?:"|"))?\}/;
|
||||||
[recipient.email]: {
|
const ALLOWED_REPLACEMENTS = ['subscriber_firstname'];
|
||||||
unique_id: recipient.uuid,
|
|
||||||
unsubscribe_url: postEmailSerializer.createUnsubscribeUrl(recipient.uuid)
|
// extract replacements with fallbacks. We have to handle replacements here because
|
||||||
|
// it's the only place we have access to both member data and specified fallbacks
|
||||||
|
const replacements = [];
|
||||||
|
emailTmpl.html = emailTmpl.html.replace(EMAIL_REPLACEMENT_REGEX, (replacementMatch, replacementStr) => {
|
||||||
|
const match = replacementStr.match(REPLACEMENT_STRING_REGEX);
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
const {memberProp, fallback} = match.groups;
|
||||||
|
|
||||||
|
if (ALLOWED_REPLACEMENTS.includes(memberProp)) {
|
||||||
|
const varName = `replacement_${replacements.length}`;
|
||||||
|
|
||||||
|
replacements.push({
|
||||||
|
varName,
|
||||||
|
memberProp,
|
||||||
|
fallback
|
||||||
|
});
|
||||||
|
return `%recipient.${varName}%`;
|
||||||
}
|
}
|
||||||
}, emailData);
|
}
|
||||||
}, {});
|
|
||||||
|
// output the user-entered replacement string for unknown or invalid replacements
|
||||||
|
// so that it's obvious there's an error in test emails
|
||||||
|
return replacementStr;
|
||||||
|
});
|
||||||
|
|
||||||
|
// plaintext will have the same replacements so no need to add them to the list and
|
||||||
|
// bloat the template variables object but we still need replacements for mailgun template syntax
|
||||||
|
let count = 0;
|
||||||
|
emailTmpl.plaintext = emailTmpl.plaintext.replace(EMAIL_REPLACEMENT_REGEX, (match, replacementStr) => {
|
||||||
|
const {groups: {memberProp}} = replacementStr.match(REPLACEMENT_STRING_REGEX);
|
||||||
|
if (ALLOWED_REPLACEMENTS.includes(memberProp)) {
|
||||||
|
const varName = `replacement_${count}`;
|
||||||
|
count++;
|
||||||
|
return `%recipient.${varName}`;
|
||||||
|
}
|
||||||
|
return replacementStr;
|
||||||
|
});
|
||||||
|
|
||||||
|
const emails = [];
|
||||||
|
const emailData = {};
|
||||||
|
members.forEach((member) => {
|
||||||
|
emails.push(member.email);
|
||||||
|
|
||||||
|
// firstname is a computed property only used here for now
|
||||||
|
// TODO: move into model computed property or output serializer?
|
||||||
|
member.firstname = (member.name || '').split(' ')[0];
|
||||||
|
|
||||||
|
// add static data to mailgun template variables
|
||||||
|
const data = {
|
||||||
|
unique_id: member.uuid,
|
||||||
|
unsubscribe_url: postEmailSerializer.createUnsubscribeUrl(member.uuid)
|
||||||
|
};
|
||||||
|
|
||||||
|
// add replacement data/requested fallback to mailgun template variables
|
||||||
|
replacements.forEach(({varName, memberProp, fallback}) => {
|
||||||
|
data[varName] = member[memberProp] || fallback || '';
|
||||||
|
});
|
||||||
|
|
||||||
|
emailData[member.email] = data;
|
||||||
|
});
|
||||||
|
|
||||||
return {emailTmpl, emails, emailData};
|
return {emailTmpl, emails, emailData};
|
||||||
};
|
};
|
||||||
|
@ -233,7 +290,9 @@ async function pendingEmailHandler(emailModel, options) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const statusChangedHandler = (emailModel, options) => {
|
const statusChangedHandler = (emailModel, options) => {
|
||||||
const emailRetried = emailModel.wasChanged() && (emailModel.get('status') === 'pending') && (emailModel.previous('status') === 'failed');
|
const emailRetried = emailModel.wasChanged()
|
||||||
|
&& emailModel.get('status') === 'pending'
|
||||||
|
&& emailModel.previous('status') === 'failed';
|
||||||
|
|
||||||
if (emailRetried) {
|
if (emailRetried) {
|
||||||
pendingEmailHandler(emailModel, options);
|
pendingEmailHandler(emailModel, options);
|
||||||
|
|
|
@ -6,6 +6,8 @@ const moment = require('moment');
|
||||||
const cheerio = require('cheerio');
|
const cheerio = require('cheerio');
|
||||||
const api = require('../../api');
|
const api = require('../../api');
|
||||||
const {URL} = require('url');
|
const {URL} = require('url');
|
||||||
|
const mobiledocLib = require('../../lib/mobiledoc');
|
||||||
|
const htmlToText = require('html-to-text');
|
||||||
|
|
||||||
const getSite = () => {
|
const getSite = () => {
|
||||||
const publicSettings = settingsCache.getPublic();
|
const publicSettings = settingsCache.getPublic();
|
||||||
|
@ -39,7 +41,8 @@ const createUnsubscribeUrl = (uuid) => {
|
||||||
// NOTE: serialization is needed to make sure we are using current API and do post transformations
|
// NOTE: serialization is needed to make sure we are using current API and do post transformations
|
||||||
// such as image URL transformation from relative to absolute
|
// such as image URL transformation from relative to absolute
|
||||||
const serializePostModel = async (model) => {
|
const serializePostModel = async (model) => {
|
||||||
const frame = {options: {context: {user: true}, formats: 'html, plaintext'}};
|
// fetch mobiledoc rather than html and plaintext so we can render email-specific contents
|
||||||
|
const frame = {options: {context: {user: true}, formats: 'mobiledoc'}};
|
||||||
const apiVersion = model.get('api_version') || 'v3';
|
const apiVersion = model.get('api_version') || 'v3';
|
||||||
const docName = 'posts';
|
const docName = 'posts';
|
||||||
|
|
||||||
|
@ -53,18 +56,32 @@ const serializePostModel = async (model) => {
|
||||||
|
|
||||||
const serialize = async (postModel, options = {isBrowserPreview: false}) => {
|
const serialize = async (postModel, options = {isBrowserPreview: false}) => {
|
||||||
const post = await serializePostModel(postModel);
|
const post = await serializePostModel(postModel);
|
||||||
|
|
||||||
post.published_at = post.published_at ? moment(post.published_at).format('DD MMM YYYY') : moment().format('DD MMM YYYY');
|
post.published_at = post.published_at ? moment(post.published_at).format('DD MMM YYYY') : moment().format('DD MMM YYYY');
|
||||||
post.authors = post.authors && post.authors.map(author => author.name).join(',');
|
post.authors = post.authors && post.authors.map(author => author.name).join(',');
|
||||||
post.html = post.html || '';
|
|
||||||
if (post.posts_meta) {
|
if (post.posts_meta) {
|
||||||
post.email_subject = post.posts_meta.email_subject;
|
post.email_subject = post.posts_meta.email_subject;
|
||||||
}
|
}
|
||||||
|
post.html = mobiledocLib.mobiledocHtmlRenderer.render(JSON.parse(post.mobiledoc), {target: 'email'});
|
||||||
|
// 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
|
||||||
|
post.plaintext = htmlToText.fromString(post.html, {
|
||||||
|
ignoreImage: true,
|
||||||
|
hideLinkHrefIfSameAsText: true,
|
||||||
|
preserveNewlines: true,
|
||||||
|
returnDomByDefault: true,
|
||||||
|
uppercaseHeadings: false
|
||||||
|
});
|
||||||
|
|
||||||
let htmlTemplate = template({post, site: getSite()});
|
let htmlTemplate = template({post, site: getSite()});
|
||||||
if (options.isBrowserPreview) {
|
if (options.isBrowserPreview) {
|
||||||
const previewUnsubscribeUrl = createUnsubscribeUrl();
|
const previewUnsubscribeUrl = createUnsubscribeUrl();
|
||||||
htmlTemplate = htmlTemplate.replace('%recipient.unsubscribe_url%', previewUnsubscribeUrl);
|
htmlTemplate = htmlTemplate.replace('%recipient.unsubscribe_url%', previewUnsubscribeUrl);
|
||||||
}
|
}
|
||||||
|
|
||||||
let juicedHtml = juice(htmlTemplate);
|
let juicedHtml = juice(htmlTemplate);
|
||||||
|
|
||||||
// Force all links to open in new tab
|
// Force all links to open in new tab
|
||||||
let _cheerio = cheerio.load(juicedHtml);
|
let _cheerio = cheerio.load(juicedHtml);
|
||||||
_cheerio('a').attr('target','_blank');
|
_cheerio('a').attr('target','_blank');
|
||||||
|
|
Loading…
Reference in a new issue