🐛 Fixed ref attribute in email links (#15775)

fixes https://github.com/TryGhost/Team/issues/2025 
fixes https://github.com/TryGhost/Team/issues/2023

The `ref` attribute has changed in email links:
- We now use the site name when linking to external sites
- We blacklist facebook.com because it doesn't support ref attributes
- '-newsletter' is not repeated anymore if the newsletter name already ends with 'newsletter'
- We always sluggify the ref
- We no longer overwrite existing ref, utm_source or source parameters
This commit is contained in:
Simon Backx 2022-11-08 11:24:00 +01:00 committed by GitHub
parent 6214812ac0
commit 663b0dfeb2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 138 additions and 8 deletions

View File

@ -392,12 +392,17 @@ const PostEmailSerializer = {
if (!options.isBrowserPreview && !options.isTestEmail && settingsCache.get('email_track_clicks')) {
result.html = await linkReplacer.replace(result.html, async (url) => {
// Add newsletter source attribution
url = memberAttribution.service.addEmailSourceAttributionTracking(url, newsletter);
const isSite = urlUtils.isSiteUrl(url);
if (isSite) {
// Add newsletter name as ref to the URL
url = memberAttribution.service.addEmailSourceAttributionTracking(url, newsletter);
// Only add post attribution to our own site (because external sites could/should not process this information)
url = memberAttribution.service.addPostAttributionTracking(url, post);
} else {
// Add email source attribution without the newsletter name
url = memberAttribution.service.addEmailSourceAttributionTracking(url);
}
// Add link click tracking

View File

@ -40,7 +40,8 @@ class MemberAttributionServiceWrapper {
Integration: models.Integration
},
attributionBuilder: this.attributionBuilder,
getTrackingEnabled: () => !!settingsCache.get('members_track_sources')
getTrackingEnabled: () => !!settingsCache.get('members_track_sources'),
getSiteTitle: () => settingsCache.get('title')
});
}
}

View File

@ -1,4 +1,11 @@
const UrlHistory = require('./history');
const {slugify} = require('@tryghost/string');
const blacklistedReferrerDomains = [
// Facebook has some restrictions on the 'ref' attribute (max 15 chars + restricted character set) that breaks links if we add ?ref=longer-string
'facebook.com',
'www.facebook.com'
];
class MemberAttributionService {
/**
@ -9,17 +16,23 @@ class MemberAttributionService {
* @param {Object} deps.models.MemberCreatedEvent
* @param {Object} deps.models.SubscriptionCreatedEvent
* @param {() => boolean} deps.getTrackingEnabled
* @param {() => string} deps.getSiteTitle
*/
constructor({attributionBuilder, models, getTrackingEnabled}) {
constructor({attributionBuilder, models, getTrackingEnabled, getSiteTitle}) {
this.models = models;
this.attributionBuilder = attributionBuilder;
this._getTrackingEnabled = getTrackingEnabled;
this._getSiteTitle = getSiteTitle;
}
get isTrackingEnabled() {
return this._getTrackingEnabled();
}
get siteTitle() {
return this._getSiteTitle();
}
/**
*
* @param {Object} context instance of ghost framework context object
@ -85,14 +98,33 @@ class MemberAttributionService {
* Add some parameters to a URL so that the frontend script can detect this and add the required records
* in the URLHistory.
* @param {URL} url instance that will get updated
* @param {Object} newsletter The newsletter from which a link was clicked
* @param {Object} [useNewsletter] Use the newsletter name instead of the site name as referrer source
* @returns {URL}
*/
addEmailSourceAttributionTracking(url, newsletter) {
addEmailSourceAttributionTracking(url, useNewsletter) {
// Create a deep copy
url = new URL(url);
// We slugify the name here so that we don't use the default slugs in fixtures
url.searchParams.append('ref', newsletter.get('name') + '-newsletter');
if (url.searchParams.has('ref') || url.searchParams.has('utm_source') || url.searchParams.has('source')) {
// Don't overwrite + keep existing source attribution
return url;
}
// Check blacklist domains
const referrerDomain = url.hostname;
if (blacklistedReferrerDomains.includes(referrerDomain)) {
return url;
}
if (useNewsletter) {
const name = slugify(useNewsletter.get('name'));
// If newsletter name ends with newsletter, don't add it again
const ref = name.endsWith('newsletter') ? name : `${name}-newsletter`;
url.searchParams.append('ref', ref);
} else {
url.searchParams.append('ref', slugify(this.siteTitle));
}
return url;
}
@ -107,6 +139,11 @@ class MemberAttributionService {
// Create a deep copy
url = new URL(url);
if (url.searchParams.has('attribution_id') || url.searchParams.has('attribution_type')) {
// Don't overwrite
return url;
}
// Post attribution
url.searchParams.append('attribution_id', post.id);
url.searchParams.append('attribution_type', 'post');

View File

@ -22,6 +22,7 @@
"dependencies": {
"@tryghost/domain-events": "0.0.0",
"@tryghost/member-events": "0.0.0",
"@tryghost/referrers": "0.0.0"
"@tryghost/referrers": "0.0.0",
"@tryghost/string": "0.2.1"
}
}

View File

@ -10,6 +10,92 @@ describe('MemberAttributionService', function () {
});
});
describe('addEmailSourceAttributionTracking', function () {
it('uses sluggified sitename for external urls', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/');
const updatedUrl = await service.addEmailSourceAttributionTracking(url);
should(updatedUrl.toString()).equal('https://example.com/?ref=hello-world');
});
it('uses sluggified newsletter name for internal urls', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/');
const newsletterName = 'used newsletter name';
const newsletter = {
get: (t) => {
if (t === 'name') {
return newsletterName;
}
}
};
const updatedUrl = await service.addEmailSourceAttributionTracking(url, newsletter);
should(updatedUrl.toString()).equal('https://example.com/?ref=used-newsletter-name-newsletter');
});
it('does not repeat newsletter at the end of the newsletter name', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/');
const newsletterName = 'Weekly newsletter';
const newsletter = {
get: (t) => {
if (t === 'name') {
return newsletterName;
}
}
};
const updatedUrl = await service.addEmailSourceAttributionTracking(url, newsletter);
should(updatedUrl.toString()).equal('https://example.com/?ref=weekly-newsletter');
});
it('does not add ref to blacklisted domains', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://facebook.com/');
const updatedUrl = await service.addEmailSourceAttributionTracking(url);
should(updatedUrl.toString()).equal('https://facebook.com/');
});
it('does not add ref if utm_source is present', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/?utm_source=hello');
const updatedUrl = await service.addEmailSourceAttributionTracking(url);
should(updatedUrl.toString()).equal('https://example.com/?utm_source=hello');
});
it('does not add ref if ref is present', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/?ref=hello');
const updatedUrl = await service.addEmailSourceAttributionTracking(url);
should(updatedUrl.toString()).equal('https://example.com/?ref=hello');
});
it('does not add ref if source is present', async function () {
const service = new MemberAttributionService({
getSiteTitle: () => 'Hello world'
});
const url = new URL('https://example.com/?source=hello');
const updatedUrl = await service.addEmailSourceAttributionTracking(url);
should(updatedUrl.toString()).equal('https://example.com/?source=hello');
});
});
describe('getAttributionFromContext', function () {
it('returns null if no context is provided', async function () {
const service = new MemberAttributionService({