From fb1644dcf21796ec386102e840ecc605417d3e74 Mon Sep 17 00:00:00 2001 From: EXPLOSION Date: Fri, 26 Aug 2022 01:18:15 +0000 Subject: [PATCH 1/3] Add functionality to use navigator.sendBeacon --- src/lib/request.ts | 32 ++++++++++++++++++++------------ src/lib/tracker.spec.ts | 1 + 2 files changed, 21 insertions(+), 12 deletions(-) diff --git a/src/lib/request.ts b/src/lib/request.ts index e7c5f2a..ee64279 100644 --- a/src/lib/request.ts +++ b/src/lib/request.ts @@ -13,16 +13,20 @@ type EventPayload = { readonly p?: string; }; -// eslint-disable-next-line functional/no-mixed-type export type EventOptions = { /** * Callback called when the event is successfully sent. + * Does not work with `useSendBeacon = true`. */ readonly callback?: () => void; /** * Properties to be bound to the event. */ readonly props?: { readonly [propName: string]: string | number | boolean }; + /** + * Whether to use `Navigator#sendBeacon`. + */ + readonly useSendBeacon?: boolean; }; /** @@ -68,15 +72,19 @@ export function sendEvent( p: options && options.props ? JSON.stringify(options.props) : undefined, }; - const req = new XMLHttpRequest(); - req.open('POST', `${data.apiHost}/api/event`, true); - req.setRequestHeader('Content-Type', 'text/plain'); - req.send(JSON.stringify(payload)); - // eslint-disable-next-line functional/immutable-data - req.onreadystatechange = () => { - if (req.readyState !== 4) return; - if (options && options.callback) { - options.callback(); - } - }; + if (!options?.useSendBeacon) { + const req = new XMLHttpRequest(); + req.open('POST', `${data.apiHost}/api/event`, true); + req.setRequestHeader('Content-Type', 'text/plain'); + req.send(JSON.stringify(payload)); + // eslint-disable-next-line functional/immutable-data + req.onreadystatechange = () => { + if (req.readyState !== 4) return; + if (options && options.callback) { + options.callback(); + } + }; + } else { + navigator.sendBeacon(`${data.apiHost}/api/event`, JSON.stringify(payload)); + } } diff --git a/src/lib/tracker.spec.ts b/src/lib/tracker.spec.ts index 95dd922..88093eb 100644 --- a/src/lib/tracker.spec.ts +++ b/src/lib/tracker.spec.ts @@ -38,6 +38,7 @@ describe('tracker', () => { variation3: 1, variation4: true, }, + useSendBeacon: false, }); test('inits with default config', () => { From 67d3ecef7c4731c8aa37469bf9ea96ab2cabb80a Mon Sep 17 00:00:00 2001 From: EXPLOSION Date: Fri, 26 Aug 2022 02:02:11 +0000 Subject: [PATCH 2/3] Basic test --- src/lib/request.spec.ts | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/src/lib/request.spec.ts b/src/lib/request.spec.ts index 4016de0..5a4137f 100644 --- a/src/lib/request.spec.ts +++ b/src/lib/request.spec.ts @@ -19,6 +19,13 @@ let xhrMockClass: ReturnType; const xmr = jest.spyOn(window, 'XMLHttpRequest'); +Object.assign(navigator, { + sendBeacon() { + // just making node aware this function exists + }, +}); +const sendBeacon = jest.spyOn(navigator, 'sendBeacon'); + const defaultData: Required = { hashMode: false, trackLocalhost: false, @@ -78,6 +85,25 @@ describe('sendEvent', () => { expect(xhrMockClass.send).toHaveBeenCalledWith(JSON.stringify(payload)); }); + test('sends via Navigator#sendBeacon', () => { + expect(sendBeacon).not.toHaveBeenCalled(); + sendEvent('myEvent', defaultData, { useSendBeacon: true }); + expect(sendBeacon).toHaveBeenCalledTimes(1); + + const payload = { + n: 'myEvent', + u: defaultData.url, + d: defaultData.domain, + r: defaultData.referrer, + w: defaultData.deviceWidth, + h: 0, + }; + + expect(sendBeacon).toHaveBeenCalledWith( + `${defaultData.apiHost}/api/event`, + JSON.stringify(payload) + ); + }); test('hash mode', () => { expect(xmr).not.toHaveBeenCalled(); sendEvent('myEvent', { ...defaultData, hashMode: true }); From 61c32d1ce6123960166e5aaebe49d029d3f358de Mon Sep 17 00:00:00 2001 From: Joshua Gugun Siagian Date: Mon, 26 Jun 2023 16:41:05 +0800 Subject: [PATCH 3/3] use sendBeacon for Outbound link click tracking --- README.md | 3 ++- src/lib/request.spec.ts | 3 ++- src/lib/request.ts | 6 +----- src/lib/tracker.spec.ts | 3 ++- src/lib/tracker.ts | 28 ++++++++-------------------- 5 files changed, 15 insertions(+), 28 deletions(-) diff --git a/README.md b/README.md index 7638d86..24f1c66 100644 --- a/README.md +++ b/README.md @@ -60,6 +60,7 @@ const plausible = Plausible({ | hashMode | `bool` | Enables tracking based on URL hash changes. | `false` | | trackLocalhost | `bool` | Enables tracking on *localhost*. | `false` | | apiHost | `string` | Plausible's API host to use. Change this if you are self-hosting. | `'https://plausible.io'` | +| useSendBeacon | `bool` | Enables the use of `navigator.sendBeacon` method to send events. | `false` | The object returned from `Plausible()` contains the functions that you'll use to track your events. These functions are: @@ -211,7 +212,7 @@ You can also track all clicks to outbound links using `enableAutoOutboundTrackin For details on how to setup the tracking, visit the [docs](https://docs.plausible.io/outbound-link-click-tracking). -This function adds a `click` event listener to all `a` tags on the page and reports them to Plausible. It also creates a [MutationObserver](https://developer.mozilla.org/en-US/docs/Web/API/MutationObserver) that efficiently tracks node mutations, so dynamically-added links are also tracked. +This function adds a `click` event listener to all `a` tags on the page and reports them to Plausible. It also creates a [MutationObserver](https://developer.mozilla.org/en-US/docs/Web/API/MutationObserver) that efficiently tracks node mutations, so dynamically-added links are also tracked. Event is sent with `navigator.sendBeacon` method. ```ts import Plausible from 'plausible-tracker' diff --git a/src/lib/request.spec.ts b/src/lib/request.spec.ts index 5a4137f..c0255e2 100644 --- a/src/lib/request.spec.ts +++ b/src/lib/request.spec.ts @@ -29,6 +29,7 @@ const sendBeacon = jest.spyOn(navigator, 'sendBeacon'); const defaultData: Required = { hashMode: false, trackLocalhost: false, + useSendBeacon: false, url: 'https://my-app.com/my-url', domain: 'my-app.com', referrer: null, @@ -87,7 +88,7 @@ describe('sendEvent', () => { }); test('sends via Navigator#sendBeacon', () => { expect(sendBeacon).not.toHaveBeenCalled(); - sendEvent('myEvent', defaultData, { useSendBeacon: true }); + sendEvent('myEvent', { ...defaultData, useSendBeacon: true }); expect(sendBeacon).toHaveBeenCalledTimes(1); const payload = { diff --git a/src/lib/request.ts b/src/lib/request.ts index ee64279..2e91f97 100644 --- a/src/lib/request.ts +++ b/src/lib/request.ts @@ -23,10 +23,6 @@ export type EventOptions = { * Properties to be bound to the event. */ readonly props?: { readonly [propName: string]: string | number | boolean }; - /** - * Whether to use `Navigator#sendBeacon`. - */ - readonly useSendBeacon?: boolean; }; /** @@ -72,7 +68,7 @@ export function sendEvent( p: options && options.props ? JSON.stringify(options.props) : undefined, }; - if (!options?.useSendBeacon) { + if (!data?.useSendBeacon) { const req = new XMLHttpRequest(); req.open('POST', `${data.apiHost}/api/event`, true); req.setRequestHeader('Content-Type', 'text/plain'); diff --git a/src/lib/tracker.spec.ts b/src/lib/tracker.spec.ts index 88093eb..e250706 100644 --- a/src/lib/tracker.spec.ts +++ b/src/lib/tracker.spec.ts @@ -13,6 +13,7 @@ describe('tracker', () => { const getDefaultData: () => Required = () => ({ hashMode: false, trackLocalhost: false, + useSendBeacon: false, url: location.href, domain: location.hostname, referrer: document.referrer || null, @@ -23,6 +24,7 @@ describe('tracker', () => { const getCustomData: () => Required = () => ({ hashMode: true, trackLocalhost: true, + useSendBeacon: false, url: 'https://my-url.com', domain: 'my-domain.com', referrer: 'my-referrer', @@ -38,7 +40,6 @@ describe('tracker', () => { variation3: 1, variation4: true, }, - useSendBeacon: false, }); test('inits with default config', () => { diff --git a/src/lib/tracker.ts b/src/lib/tracker.ts index a0a6aa9..d05cb3c 100644 --- a/src/lib/tracker.ts +++ b/src/lib/tracker.ts @@ -23,6 +23,11 @@ export type PlausibleInitOptions = { * Defaults to `'https://plausible.io'` */ readonly apiHost?: string; + /** + * Whether to use `Navigator#sendBeacon`. + * Defaults to `false`. + */ + readonly useSendBeacon?: boolean; }; /** @@ -223,6 +228,7 @@ export default function Plausible( referrer: document.referrer || null, deviceWidth: window.innerWidth, apiHost: 'https://plausible.io', + useSendBeacon: false, ...defaults, }); @@ -276,26 +282,8 @@ export default function Plausible( attributeFilter: ['href'], } ) => { - function trackClick(this: HTMLAnchorElement, event: MouseEvent) { - trackEvent('Outbound Link: Click', { props: { url: this.href } }); - - /* istanbul ignore next */ - // eslint-disable-next-line @typescript-eslint/ban-ts-comment - // @ts-ignore - if ( - !( - typeof process !== 'undefined' && - process && - process.env.NODE_ENV === 'test' - ) - ) { - setTimeout(() => { - // eslint-disable-next-line functional/immutable-data - location.href = this.href; - }, 150); - } - - event.preventDefault(); + function trackClick(this: HTMLAnchorElement, _: MouseEvent) { + trackEvent('Outbound Link: Click', { props: { url: this.href } }, { useSendBeacon: true }); } // eslint-disable-next-line functional/prefer-readonly-type