2021-03-08 18:12:59 +05:30
|
|
|
import { setHTMLFixture } from 'helpers/fixtures';
|
2021-11-11 11:23:49 +05:30
|
|
|
import { TEST_HOST } from 'helpers/test_constants';
|
2021-04-17 20:07:23 +05:30
|
|
|
import { TRACKING_CONTEXT_SCHEMA } from '~/experimentation/constants';
|
2021-11-11 11:23:49 +05:30
|
|
|
import { getExperimentData, getAllExperimentContexts } from '~/experimentation/utils';
|
2021-09-04 01:27:46 +05:30
|
|
|
import Tracking, { initUserTracking, initDefaultTrackers } from '~/tracking';
|
2021-11-11 11:23:49 +05:30
|
|
|
import { REFERRER_TTL, URLS_CACHE_STORAGE_KEY } from '~/tracking/constants';
|
2021-09-04 01:27:46 +05:30
|
|
|
import getStandardContext from '~/tracking/get_standard_context';
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
jest.mock('~/experimentation/utils', () => ({
|
|
|
|
getExperimentData: jest.fn(),
|
|
|
|
getAllExperimentContexts: jest.fn(),
|
|
|
|
}));
|
2019-10-12 21:52:04 +05:30
|
|
|
|
|
|
|
describe('Tracking', () => {
|
2021-09-04 01:27:46 +05:30
|
|
|
let standardContext;
|
2019-12-04 20:38:33 +05:30
|
|
|
let snowplowSpy;
|
2019-12-21 20:55:43 +05:30
|
|
|
let bindDocumentSpy;
|
2020-05-24 23:13:21 +05:30
|
|
|
let trackLoadEventsSpy;
|
2021-09-04 01:27:46 +05:30
|
|
|
let enableFormTracking;
|
2021-11-11 11:23:49 +05:30
|
|
|
let setAnonymousUrlsSpy;
|
2021-09-04 01:27:46 +05:30
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
window.gl = window.gl || {};
|
2021-11-11 11:23:49 +05:30
|
|
|
window.gl.snowplowUrls = {};
|
2021-09-04 01:27:46 +05:30
|
|
|
window.gl.snowplowStandardContext = {
|
|
|
|
schema: 'iglu:com.gitlab/gitlab_standard',
|
|
|
|
data: {
|
|
|
|
environment: 'testing',
|
|
|
|
source: 'unknown',
|
|
|
|
extra: {},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
standardContext = getStandardContext();
|
|
|
|
});
|
2019-12-04 20:38:33 +05:30
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
beforeEach(() => {
|
2021-04-17 20:07:23 +05:30
|
|
|
getExperimentData.mockReturnValue(undefined);
|
2021-11-11 11:23:49 +05:30
|
|
|
getAllExperimentContexts.mockReturnValue([]);
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
window.snowplow = window.snowplow || (() => {});
|
2019-12-04 20:38:33 +05:30
|
|
|
window.snowplowOptions = {
|
|
|
|
namespace: '_namespace_',
|
|
|
|
hostname: 'app.gitfoo.com',
|
|
|
|
cookieDomain: '.gitfoo.com',
|
|
|
|
};
|
|
|
|
snowplowSpy = jest.spyOn(window, 'snowplow');
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
describe('initUserTracking', () => {
|
|
|
|
it('calls through to get a new tracker with the expected options', () => {
|
|
|
|
initUserTracking();
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('newTracker', '_namespace_', 'app.gitfoo.com', {
|
|
|
|
namespace: '_namespace_',
|
|
|
|
hostname: 'app.gitfoo.com',
|
|
|
|
cookieDomain: '.gitfoo.com',
|
|
|
|
appId: '',
|
|
|
|
userFingerprint: false,
|
|
|
|
respectDoNotTrack: true,
|
|
|
|
forceSecureTracker: true,
|
|
|
|
eventMethod: 'post',
|
2021-01-03 14:25:43 +05:30
|
|
|
contexts: { webPage: true, performanceTiming: true },
|
2019-12-04 20:38:33 +05:30
|
|
|
formTracking: false,
|
|
|
|
linkClickTracking: false,
|
2021-01-29 00:20:46 +05:30
|
|
|
pageUnloadTimer: 10,
|
2021-09-04 01:27:46 +05:30
|
|
|
formTrackingConfig: {
|
|
|
|
fields: { allow: [] },
|
|
|
|
forms: { allow: [] },
|
|
|
|
},
|
2019-12-04 20:38:33 +05:30
|
|
|
});
|
|
|
|
});
|
2020-11-24 15:15:51 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
describe('initDefaultTrackers', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
bindDocumentSpy = jest.spyOn(Tracking, 'bindDocument').mockImplementation(() => null);
|
|
|
|
trackLoadEventsSpy = jest.spyOn(Tracking, 'trackLoadEvents').mockImplementation(() => null);
|
2021-09-04 01:27:46 +05:30
|
|
|
enableFormTracking = jest
|
|
|
|
.spyOn(Tracking, 'enableFormTracking')
|
|
|
|
.mockImplementation(() => null);
|
2021-11-11 11:23:49 +05:30
|
|
|
setAnonymousUrlsSpy = jest.spyOn(Tracking, 'setAnonymousUrls').mockImplementation(() => null);
|
2020-11-24 15:15:51 +05:30
|
|
|
});
|
2019-10-12 21:52:04 +05:30
|
|
|
|
2019-12-04 20:38:33 +05:30
|
|
|
it('should activate features based on what has been enabled', () => {
|
2020-11-24 15:15:51 +05:30
|
|
|
initDefaultTrackers();
|
2019-12-04 20:38:33 +05:30
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('enableActivityTracking', 30, 30);
|
2021-09-04 01:27:46 +05:30
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('trackPageView', null, [standardContext]);
|
2019-12-04 20:38:33 +05:30
|
|
|
expect(snowplowSpy).not.toHaveBeenCalledWith('enableFormTracking');
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalledWith('enableLinkClickTracking');
|
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
window.snowplowOptions = {
|
|
|
|
...window.snowplowOptions,
|
2019-12-04 20:38:33 +05:30
|
|
|
formTracking: true,
|
|
|
|
linkClickTracking: true,
|
2021-09-04 01:27:46 +05:30
|
|
|
formTrackingConfig: { forms: { whitelist: ['foo'] }, fields: { whitelist: ['bar'] } },
|
2020-05-24 23:13:21 +05:30
|
|
|
};
|
2019-12-04 20:38:33 +05:30
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
initDefaultTrackers();
|
2021-09-04 01:27:46 +05:30
|
|
|
expect(enableFormTracking).toHaveBeenCalledWith(window.snowplowOptions.formTrackingConfig);
|
2019-12-04 20:38:33 +05:30
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('enableLinkClickTracking');
|
|
|
|
});
|
2019-12-21 20:55:43 +05:30
|
|
|
|
|
|
|
it('binds the document event handling', () => {
|
2020-11-24 15:15:51 +05:30
|
|
|
initDefaultTrackers();
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(bindDocumentSpy).toHaveBeenCalled();
|
|
|
|
});
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
it('tracks page loaded events', () => {
|
2020-11-24 15:15:51 +05:30
|
|
|
initDefaultTrackers();
|
2020-05-24 23:13:21 +05:30
|
|
|
expect(trackLoadEventsSpy).toHaveBeenCalled();
|
|
|
|
});
|
2021-11-11 11:23:49 +05:30
|
|
|
|
|
|
|
it('calls the anonymized URLs method', () => {
|
|
|
|
initDefaultTrackers();
|
|
|
|
expect(setAnonymousUrlsSpy).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when there are experiment contexts', () => {
|
|
|
|
const experimentContexts = [
|
|
|
|
{
|
|
|
|
schema: TRACKING_CONTEXT_SCHEMA,
|
|
|
|
data: { experiment: 'experiment1', variant: 'control' },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
schema: TRACKING_CONTEXT_SCHEMA,
|
|
|
|
data: { experiment: 'experiment_two', variant: 'candidate' },
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
getAllExperimentContexts.mockReturnValue(experimentContexts);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('includes those contexts alongside the standard context', () => {
|
|
|
|
initDefaultTrackers();
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('trackPageView', null, [
|
|
|
|
standardContext,
|
|
|
|
...experimentContexts,
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
2019-12-04 20:38:33 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
describe('.event', () => {
|
|
|
|
afterEach(() => {
|
|
|
|
window.doNotTrack = undefined;
|
|
|
|
navigator.doNotTrack = undefined;
|
|
|
|
navigator.msDoNotTrack = undefined;
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('tracks to snowplow (our current tracking system)', () => {
|
|
|
|
Tracking.event('_category_', '_eventName_', { label: '_label_' });
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith(
|
|
|
|
'trackStructEvent',
|
|
|
|
'_category_',
|
|
|
|
'_eventName_',
|
|
|
|
'_label_',
|
|
|
|
undefined,
|
|
|
|
undefined,
|
2021-09-04 01:27:46 +05:30
|
|
|
[standardContext],
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('allows adding extra data to the default context', () => {
|
|
|
|
const extra = { foo: 'bar' };
|
|
|
|
|
|
|
|
Tracking.event('_category_', '_eventName_', { extra });
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith(
|
|
|
|
'trackStructEvent',
|
|
|
|
'_category_',
|
|
|
|
'_eventName_',
|
|
|
|
undefined,
|
|
|
|
undefined,
|
|
|
|
undefined,
|
|
|
|
[
|
|
|
|
{
|
|
|
|
...standardContext,
|
|
|
|
data: {
|
|
|
|
...standardContext.data,
|
|
|
|
extra,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
2019-12-21 20:55:43 +05:30
|
|
|
);
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('skips tracking if snowplow is unavailable', () => {
|
|
|
|
window.snowplow = false;
|
|
|
|
Tracking.event('_category_', '_eventName_');
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
2019-12-04 20:38:33 +05:30
|
|
|
|
|
|
|
it('skips tracking if the user does not want to be tracked (general spec)', () => {
|
|
|
|
window.doNotTrack = '1';
|
|
|
|
Tracking.event('_category_', '_eventName_');
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('skips tracking if the user does not want to be tracked (firefox legacy)', () => {
|
|
|
|
navigator.doNotTrack = 'yes';
|
|
|
|
Tracking.event('_category_', '_eventName_');
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('skips tracking if the user does not want to be tracked (IE legacy)', () => {
|
|
|
|
navigator.msDoNotTrack = '1';
|
|
|
|
Tracking.event('_category_', '_eventName_');
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
|
2021-06-08 01:23:25 +05:30
|
|
|
describe('.enableFormTracking', () => {
|
2021-09-04 01:27:46 +05:30
|
|
|
it('tells snowplow to enable form tracking, with only explicit contexts', () => {
|
|
|
|
const config = { forms: { allow: ['form-class1'] }, fields: { allow: ['input-class1'] } };
|
|
|
|
Tracking.enableFormTracking(config, ['_passed_context_', standardContext]);
|
2021-06-08 01:23:25 +05:30
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith(
|
|
|
|
'enableFormTracking',
|
|
|
|
{ forms: { whitelist: ['form-class1'] }, fields: { whitelist: ['input-class1'] } },
|
|
|
|
['_passed_context_'],
|
|
|
|
);
|
2021-06-08 01:23:25 +05:30
|
|
|
});
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
it('throws an error if no allow rules are provided', () => {
|
|
|
|
const expectedError = new Error('Unable to enable form event tracking without allow rules.');
|
2021-06-08 01:23:25 +05:30
|
|
|
|
|
|
|
expect(() => Tracking.enableFormTracking()).toThrow(expectedError);
|
2021-09-04 01:27:46 +05:30
|
|
|
expect(() => Tracking.enableFormTracking({ fields: { allow: true } })).toThrow(expectedError);
|
|
|
|
expect(() => Tracking.enableFormTracking({ fields: { allow: [] } })).not.toThrow(
|
2021-06-08 01:23:25 +05:30
|
|
|
expectedError,
|
|
|
|
);
|
|
|
|
});
|
2021-09-30 23:02:18 +05:30
|
|
|
|
|
|
|
it('does not add empty form whitelist rules', () => {
|
|
|
|
Tracking.enableFormTracking({ fields: { allow: ['input-class1'] } });
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith(
|
|
|
|
'enableFormTracking',
|
|
|
|
{ fields: { whitelist: ['input-class1'] } },
|
|
|
|
[],
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when `document.readyState` does not equal `complete`', () => {
|
|
|
|
const originalReadyState = document.readyState;
|
|
|
|
const setReadyState = (value) => {
|
|
|
|
Object.defineProperty(document, 'readyState', {
|
|
|
|
value,
|
|
|
|
configurable: true,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
const fireReadyStateChangeEvent = () => {
|
|
|
|
document.dispatchEvent(new Event('readystatechange'));
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
setReadyState('interactive');
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
setReadyState(originalReadyState);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not call `window.snowplow` until `readystatechange` is fired and `document.readyState` equals `complete`', () => {
|
|
|
|
Tracking.enableFormTracking({ fields: { allow: ['input-class1'] } });
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
|
|
|
|
fireReadyStateChangeEvent();
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
|
|
|
|
setReadyState('complete');
|
|
|
|
fireReadyStateChangeEvent();
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
2021-06-08 01:23:25 +05:30
|
|
|
});
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
describe('.flushPendingEvents', () => {
|
|
|
|
it('flushes any pending events', () => {
|
|
|
|
Tracking.initialized = false;
|
|
|
|
Tracking.event('_category_', '_eventName_', { label: '_label_' });
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
|
|
|
|
Tracking.flushPendingEvents();
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith(
|
|
|
|
'trackStructEvent',
|
|
|
|
'_category_',
|
|
|
|
'_eventName_',
|
|
|
|
'_label_',
|
|
|
|
undefined,
|
|
|
|
undefined,
|
2021-09-04 01:27:46 +05:30
|
|
|
[standardContext],
|
2021-04-17 20:07:23 +05:30
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
describe('.setAnonymousUrls', () => {
|
|
|
|
afterEach(() => {
|
|
|
|
window.gl.snowplowPseudonymizedPageUrl = '';
|
|
|
|
localStorage.removeItem(URLS_CACHE_STORAGE_KEY);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does nothing if URLs are not provided', () => {
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalled();
|
|
|
|
expect(localStorage.getItem(URLS_CACHE_STORAGE_KEY)).toBe(null);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('sets the page URL when provided and populates the cache', () => {
|
|
|
|
window.gl.snowplowPseudonymizedPageUrl = TEST_HOST;
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('setCustomUrl', TEST_HOST);
|
|
|
|
expect(JSON.parse(localStorage.getItem(URLS_CACHE_STORAGE_KEY))[0]).toStrictEqual({
|
|
|
|
url: TEST_HOST,
|
|
|
|
referrer: '',
|
|
|
|
originalUrl: window.location.href,
|
|
|
|
timestamp: Date.now(),
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('appends the hash/fragment to the pseudonymized URL', () => {
|
|
|
|
const hash = 'first-heading';
|
|
|
|
window.gl.snowplowPseudonymizedPageUrl = TEST_HOST;
|
|
|
|
window.location.hash = hash;
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('setCustomUrl', `${TEST_HOST}#${hash}`);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not set the referrer URL by default', () => {
|
|
|
|
window.gl.snowplowPseudonymizedPageUrl = TEST_HOST;
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalledWith('setReferrerUrl', expect.any(String));
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('with referrers cache', () => {
|
|
|
|
const testUrl = '/namespace:1/project:2/-/merge_requests/5';
|
|
|
|
const testOriginalUrl = '/my-namespace/my-project/-/merge_requests/';
|
|
|
|
const setUrlsCache = (data) =>
|
|
|
|
localStorage.setItem(URLS_CACHE_STORAGE_KEY, JSON.stringify(data));
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
window.gl.snowplowPseudonymizedPageUrl = TEST_HOST;
|
|
|
|
Object.defineProperty(document, 'referrer', { value: '', configurable: true });
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does nothing if a referrer can not be found', () => {
|
|
|
|
setUrlsCache([
|
|
|
|
{
|
|
|
|
url: testUrl,
|
|
|
|
originalUrl: TEST_HOST,
|
|
|
|
timestamp: Date.now(),
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalledWith('setReferrerUrl', expect.any(String));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('sets referrer URL from the page URL found in cache', () => {
|
|
|
|
Object.defineProperty(document, 'referrer', { value: testOriginalUrl });
|
|
|
|
setUrlsCache([
|
|
|
|
{
|
|
|
|
url: testUrl,
|
|
|
|
originalUrl: testOriginalUrl,
|
|
|
|
timestamp: Date.now(),
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).toHaveBeenCalledWith('setReferrerUrl', testUrl);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('ignores and removes old entries from the cache', () => {
|
|
|
|
const oldTimestamp = Date.now() - (REFERRER_TTL + 1);
|
|
|
|
Object.defineProperty(document, 'referrer', { value: testOriginalUrl });
|
|
|
|
setUrlsCache([
|
|
|
|
{
|
|
|
|
url: testUrl,
|
|
|
|
originalUrl: testOriginalUrl,
|
|
|
|
timestamp: oldTimestamp,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
|
|
|
|
Tracking.setAnonymousUrls();
|
|
|
|
|
|
|
|
expect(snowplowSpy).not.toHaveBeenCalledWith('setReferrerUrl', testUrl);
|
|
|
|
expect(localStorage.getItem(URLS_CACHE_STORAGE_KEY)).not.toContain(oldTimestamp);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
describe.each`
|
|
|
|
term
|
|
|
|
${'event'}
|
|
|
|
${'action'}
|
|
|
|
`('tracking interface events with data-track-$term', ({ term }) => {
|
2019-12-21 20:55:43 +05:30
|
|
|
let eventSpy;
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
beforeEach(() => {
|
|
|
|
eventSpy = jest.spyOn(Tracking, 'event');
|
2019-12-21 20:55:43 +05:30
|
|
|
Tracking.bindDocument('_category_'); // only happens once
|
2019-10-12 21:52:04 +05:30
|
|
|
setHTMLFixture(`
|
2021-09-04 01:27:46 +05:30
|
|
|
<input data-track-${term}="click_input1" data-track-label="_label_" value=0 />
|
|
|
|
<input data-track-${term}="click_input2" data-track-value=0 value=0/>
|
|
|
|
<input type="checkbox" data-track-${term}="toggle_checkbox" value=1 checked/>
|
2021-04-29 21:17:54 +05:30
|
|
|
<input class="dropdown" data-track-${term}="toggle_dropdown"/>
|
|
|
|
<div data-track-${term}="nested_event"><span class="nested"></span></div>
|
|
|
|
<input data-track-bogus="click_bogusinput" data-track-label="_label_" value="_value_"/>
|
|
|
|
<input data-track-${term}="click_input3" data-track-experiment="example" value="_value_"/>
|
2021-09-04 01:27:46 +05:30
|
|
|
<input data-track-${term}="event_with_extra" data-track-extra='{ "foo": "bar" }' />
|
|
|
|
<input data-track-${term}="event_with_invalid_extra" data-track-extra="invalid_json" />
|
2019-10-12 21:52:04 +05:30
|
|
|
`);
|
|
|
|
});
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
it(`binds to clicks on elements matching [data-track-${term}]`, () => {
|
|
|
|
document.querySelector(`[data-track-${term}="click_input1"]`).click();
|
2019-10-12 21:52:04 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'click_input1', {
|
|
|
|
label: '_label_',
|
2021-09-04 01:27:46 +05:30
|
|
|
value: '0',
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
it(`does not bind to clicks on elements without [data-track-${term}]`, () => {
|
|
|
|
document.querySelector('[data-track-bogus="click_bogusinput"]').click();
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
expect(eventSpy).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
it('allows value override with the data-track-value attribute', () => {
|
2021-04-29 21:17:54 +05:30
|
|
|
document.querySelector(`[data-track-${term}="click_input2"]`).click();
|
2019-10-12 21:52:04 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'click_input2', {
|
2021-09-04 01:27:46 +05:30
|
|
|
value: '0',
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('handles checkbox values correctly', () => {
|
2021-04-29 21:17:54 +05:30
|
|
|
const checkbox = document.querySelector(`[data-track-${term}="toggle_checkbox"]`);
|
2020-06-23 00:09:42 +05:30
|
|
|
|
|
|
|
checkbox.click(); // unchecking
|
2019-10-12 21:52:04 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'toggle_checkbox', {
|
2021-09-04 01:27:46 +05:30
|
|
|
value: 0,
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
checkbox.click(); // checking
|
2019-10-12 21:52:04 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'toggle_checkbox', {
|
2021-09-04 01:27:46 +05:30
|
|
|
value: '1',
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('handles bootstrap dropdowns', () => {
|
2021-04-29 21:17:54 +05:30
|
|
|
const dropdown = document.querySelector(`[data-track-${term}="toggle_dropdown"]`);
|
2020-06-23 00:09:42 +05:30
|
|
|
|
|
|
|
dropdown.dispatchEvent(new Event('show.bs.dropdown', { bubbles: true }));
|
2019-10-12 21:52:04 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'toggle_dropdown_show', {});
|
2019-10-12 21:52:04 +05:30
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
dropdown.dispatchEvent(new Event('hide.bs.dropdown', { bubbles: true }));
|
2019-12-21 20:55:43 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'toggle_dropdown_hide', {});
|
|
|
|
});
|
2019-10-12 21:52:04 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
it('handles nested elements inside an element with tracking', () => {
|
2020-06-23 00:09:42 +05:30
|
|
|
document.querySelector('span.nested').click();
|
2019-10-12 21:52:04 +05:30
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'nested_event', {});
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
it('includes experiment data if linked to an experiment', () => {
|
2021-04-17 20:07:23 +05:30
|
|
|
const mockExperimentData = {
|
|
|
|
variant: 'candidate',
|
2021-11-11 11:23:49 +05:30
|
|
|
experiment: 'example',
|
2021-04-17 20:07:23 +05:30
|
|
|
key: '2bff73f6bb8cc11156c50a8ba66b9b8b',
|
|
|
|
};
|
|
|
|
getExperimentData.mockReturnValue(mockExperimentData);
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
document.querySelector(`[data-track-${term}="click_input3"]`).click();
|
2021-04-17 20:07:23 +05:30
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'click_input3', {
|
|
|
|
value: '_value_',
|
|
|
|
context: { schema: TRACKING_CONTEXT_SCHEMA, data: mockExperimentData },
|
|
|
|
});
|
|
|
|
});
|
2021-09-04 01:27:46 +05:30
|
|
|
|
|
|
|
it('supports extra data as JSON', () => {
|
|
|
|
document.querySelector(`[data-track-${term}="event_with_extra"]`).click();
|
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'event_with_extra', {
|
|
|
|
extra: { foo: 'bar' },
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('ignores extra if provided JSON is invalid', () => {
|
|
|
|
document.querySelector(`[data-track-${term}="event_with_invalid_extra"]`).click();
|
|
|
|
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('_category_', 'event_with_invalid_extra', {});
|
|
|
|
});
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|
2020-01-01 13:55:28 +05:30
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
describe.each`
|
|
|
|
term
|
|
|
|
${'event'}
|
|
|
|
${'action'}
|
|
|
|
`('tracking page loaded events with -$term', ({ term }) => {
|
2020-05-24 23:13:21 +05:30
|
|
|
let eventSpy;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
eventSpy = jest.spyOn(Tracking, 'event');
|
|
|
|
setHTMLFixture(`
|
2021-10-27 15:23:28 +05:30
|
|
|
<div data-track-${term}="click_link" data-track-label="all_nested_links">
|
|
|
|
<input data-track-${term}="render" data-track-label="label1" value=1 data-track-property="_property_"/>
|
|
|
|
<span data-track-${term}="render" data-track-label="label2" data-track-value=1>
|
|
|
|
<a href="#" id="link">Something</a>
|
|
|
|
</span>
|
|
|
|
<input data-track-${term}="_render_bogus_" data-track-label="label3" value="_value_" data-track-property="_property_"/>
|
|
|
|
</div>
|
2020-05-24 23:13:21 +05:30
|
|
|
`);
|
|
|
|
Tracking.trackLoadEvents('_category_'); // only happens once
|
|
|
|
});
|
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
it(`sends tracking events when [data-track-${term}="render"] is on an element`, () => {
|
2020-05-24 23:13:21 +05:30
|
|
|
expect(eventSpy.mock.calls).toEqual([
|
|
|
|
[
|
|
|
|
'_category_',
|
|
|
|
'render',
|
|
|
|
{
|
|
|
|
label: 'label1',
|
2021-09-04 01:27:46 +05:30
|
|
|
value: '1',
|
2020-05-24 23:13:21 +05:30
|
|
|
property: '_property_',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'_category_',
|
|
|
|
'render',
|
|
|
|
{
|
|
|
|
label: 'label2',
|
2021-09-04 01:27:46 +05:30
|
|
|
value: '1',
|
2020-05-24 23:13:21 +05:30
|
|
|
},
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
});
|
2021-10-27 15:23:28 +05:30
|
|
|
|
|
|
|
describe.each`
|
|
|
|
event | actionSuffix
|
|
|
|
${'click'} | ${''}
|
|
|
|
${'show.bs.dropdown'} | ${'_show'}
|
|
|
|
${'hide.bs.dropdown'} | ${'_hide'}
|
|
|
|
`(`auto-tracking $event events on nested elements`, ({ event, actionSuffix }) => {
|
|
|
|
let link;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
link = document.querySelector('#link');
|
|
|
|
eventSpy.mockClear();
|
|
|
|
});
|
|
|
|
|
|
|
|
it(`avoids using ancestor [data-track-${term}="render"] tracking configurations`, () => {
|
|
|
|
link.dispatchEvent(new Event(event, { bubbles: true }));
|
|
|
|
|
|
|
|
expect(eventSpy).not.toHaveBeenCalledWith(
|
|
|
|
'_category_',
|
|
|
|
`render${actionSuffix}`,
|
|
|
|
expect.any(Object),
|
|
|
|
);
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith(
|
|
|
|
'_category_',
|
|
|
|
`click_link${actionSuffix}`,
|
|
|
|
expect.objectContaining({ label: 'all_nested_links' }),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2020-05-24 23:13:21 +05:30
|
|
|
});
|
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
describe('tracking mixin', () => {
|
|
|
|
describe('trackingOptions', () => {
|
2021-04-17 20:07:23 +05:30
|
|
|
it('returns the options defined on initialisation', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
const mixin = Tracking.mixin({ foo: 'bar' });
|
|
|
|
expect(mixin.computed.trackingOptions()).toEqual({ foo: 'bar' });
|
|
|
|
});
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it('lets local tracking value override and extend options', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
const mixin = Tracking.mixin({ foo: 'bar' });
|
2021-04-17 20:07:23 +05:30
|
|
|
// The value of this in the Vue lifecyle is different, but this serves the test's purposes
|
2020-01-01 13:55:28 +05:30
|
|
|
mixin.computed.tracking = { foo: 'baz', baz: 'bar' };
|
|
|
|
expect(mixin.computed.trackingOptions()).toEqual({ foo: 'baz', baz: 'bar' });
|
|
|
|
});
|
2021-04-29 21:17:54 +05:30
|
|
|
|
|
|
|
it('includes experiment data if linked to an experiment', () => {
|
|
|
|
const mockExperimentData = {
|
|
|
|
variant: 'candidate',
|
|
|
|
experiment: 'darkMode',
|
|
|
|
};
|
|
|
|
getExperimentData.mockReturnValue(mockExperimentData);
|
|
|
|
|
|
|
|
const mixin = Tracking.mixin({ foo: 'bar', experiment: 'darkMode' });
|
|
|
|
expect(mixin.computed.trackingOptions()).toEqual({
|
|
|
|
foo: 'bar',
|
|
|
|
context: {
|
|
|
|
schema: TRACKING_CONTEXT_SCHEMA,
|
|
|
|
data: mockExperimentData,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not include experiment data if experiment data does not exist', () => {
|
|
|
|
const mixin = Tracking.mixin({ foo: 'bar', experiment: 'lightMode' });
|
|
|
|
expect(mixin.computed.trackingOptions()).toEqual({
|
|
|
|
foo: 'bar',
|
|
|
|
});
|
|
|
|
});
|
2020-01-01 13:55:28 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
describe('trackingCategory', () => {
|
2021-04-17 20:07:23 +05:30
|
|
|
it('returns the category set in the component properties first', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
const mixin = Tracking.mixin({ category: 'foo' });
|
|
|
|
mixin.computed.tracking = {
|
|
|
|
category: 'bar',
|
|
|
|
};
|
|
|
|
expect(mixin.computed.trackingCategory()).toBe('bar');
|
|
|
|
});
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it('returns the category set in the options', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
const mixin = Tracking.mixin({ category: 'foo' });
|
|
|
|
expect(mixin.computed.trackingCategory()).toBe('foo');
|
|
|
|
});
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it('returns undefined if no category is selected', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
const mixin = Tracking.mixin();
|
|
|
|
expect(mixin.computed.trackingCategory()).toBe(undefined);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('track', () => {
|
|
|
|
let eventSpy;
|
|
|
|
let mixin;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
eventSpy = jest.spyOn(Tracking, 'event').mockReturnValue();
|
|
|
|
mixin = Tracking.mixin();
|
|
|
|
mixin = {
|
|
|
|
...mixin.computed,
|
|
|
|
...mixin.methods,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
it('calls the event method with no category or action defined', () => {
|
|
|
|
mixin.trackingCategory = mixin.trackingCategory();
|
|
|
|
mixin.trackingOptions = mixin.trackingOptions();
|
|
|
|
|
|
|
|
mixin.track();
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith(undefined, undefined, {});
|
|
|
|
});
|
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
it('calls the event method', () => {
|
|
|
|
mixin.trackingCategory = mixin.trackingCategory();
|
|
|
|
mixin.trackingOptions = mixin.trackingOptions();
|
|
|
|
|
|
|
|
mixin.track('foo');
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith(undefined, 'foo', {});
|
|
|
|
});
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it('gives precedence to data for category and options', () => {
|
2020-01-01 13:55:28 +05:30
|
|
|
mixin.trackingCategory = mixin.trackingCategory();
|
|
|
|
mixin.trackingOptions = mixin.trackingOptions();
|
|
|
|
const data = { category: 'foo', label: 'baz' };
|
|
|
|
mixin.track('foo', data);
|
|
|
|
expect(eventSpy).toHaveBeenCalledWith('foo', 'foo', data);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2019-10-12 21:52:04 +05:30
|
|
|
});
|