251 lines
6.9 KiB
JavaScript
251 lines
6.9 KiB
JavaScript
import { omitBy, isUndefined } from 'lodash';
|
|
import { TRACKING_CONTEXT_SCHEMA } from '~/experimentation/constants';
|
|
import { getExperimentData } from '~/experimentation/utils';
|
|
import getStandardContext from './get_standard_context';
|
|
|
|
const DEFAULT_SNOWPLOW_OPTIONS = {
|
|
namespace: 'gl',
|
|
hostname: window.location.hostname,
|
|
cookieDomain: window.location.hostname,
|
|
appId: '',
|
|
userFingerprint: false,
|
|
respectDoNotTrack: true,
|
|
forceSecureTracker: true,
|
|
eventMethod: 'post',
|
|
contexts: { webPage: true, performanceTiming: true },
|
|
formTracking: false,
|
|
linkClickTracking: false,
|
|
pageUnloadTimer: 10,
|
|
formTrackingConfig: {
|
|
forms: { allow: [] },
|
|
fields: { allow: [] },
|
|
},
|
|
};
|
|
|
|
const addExperimentContext = (opts) => {
|
|
const { experiment, ...options } = opts;
|
|
if (experiment) {
|
|
const data = getExperimentData(experiment);
|
|
if (data) {
|
|
const context = { schema: TRACKING_CONTEXT_SCHEMA, data };
|
|
return { ...options, context };
|
|
}
|
|
}
|
|
return options;
|
|
};
|
|
|
|
const createEventPayload = (el, { suffix = '' } = {}) => {
|
|
const {
|
|
trackAction,
|
|
trackEvent,
|
|
trackValue,
|
|
trackExtra,
|
|
trackExperiment,
|
|
trackContext,
|
|
trackLabel,
|
|
trackProperty,
|
|
} = el?.dataset || {};
|
|
|
|
const action = (trackAction || trackEvent) + (suffix || '');
|
|
let value = trackValue || el.value || undefined;
|
|
if (el.type === 'checkbox' && !el.checked) value = 0;
|
|
|
|
let extra = trackExtra;
|
|
|
|
if (extra !== undefined) {
|
|
try {
|
|
extra = JSON.parse(extra);
|
|
} catch (e) {
|
|
extra = undefined;
|
|
}
|
|
}
|
|
|
|
const context = addExperimentContext({
|
|
experiment: trackExperiment,
|
|
context: trackContext,
|
|
});
|
|
|
|
const data = {
|
|
label: trackLabel,
|
|
property: trackProperty,
|
|
value,
|
|
extra,
|
|
...context,
|
|
};
|
|
|
|
return {
|
|
action,
|
|
data: omitBy(data, isUndefined),
|
|
};
|
|
};
|
|
|
|
const eventHandler = (e, func, opts = {}) => {
|
|
const el = e.target.closest('[data-track-event], [data-track-action]');
|
|
|
|
if (!el) return;
|
|
|
|
const { action, data } = createEventPayload(el, opts);
|
|
func(opts.category, action, data);
|
|
};
|
|
|
|
const eventHandlers = (category, func) => {
|
|
const handler = (opts) => (e) => eventHandler(e, func, { ...{ category }, ...opts });
|
|
const handlers = [];
|
|
handlers.push({ name: 'click', func: handler() });
|
|
handlers.push({ name: 'show.bs.dropdown', func: handler({ suffix: '_show' }) });
|
|
handlers.push({ name: 'hide.bs.dropdown', func: handler({ suffix: '_hide' }) });
|
|
return handlers;
|
|
};
|
|
|
|
const dispatchEvent = (category = document.body.dataset.page, action = 'generic', data = {}) => {
|
|
// eslint-disable-next-line @gitlab/require-i18n-strings
|
|
if (!category) throw new Error('Tracking: no category provided for tracking.');
|
|
|
|
const { label, property, value, extra = {} } = data;
|
|
|
|
const standardContext = getStandardContext({ extra });
|
|
const contexts = [standardContext];
|
|
|
|
if (data.context) {
|
|
contexts.push(data.context);
|
|
}
|
|
|
|
return window.snowplow('trackStructEvent', category, action, label, property, value, contexts);
|
|
};
|
|
|
|
export default class Tracking {
|
|
static queuedEvents = [];
|
|
static initialized = false;
|
|
|
|
static trackable() {
|
|
return !['1', 'yes'].includes(
|
|
window.doNotTrack || navigator.doNotTrack || navigator.msDoNotTrack,
|
|
);
|
|
}
|
|
|
|
static flushPendingEvents() {
|
|
this.initialized = true;
|
|
|
|
while (this.queuedEvents.length) {
|
|
dispatchEvent(...this.queuedEvents.shift());
|
|
}
|
|
}
|
|
|
|
static enabled() {
|
|
return typeof window.snowplow === 'function' && this.trackable();
|
|
}
|
|
|
|
static event(...eventData) {
|
|
if (!this.enabled()) return false;
|
|
|
|
if (!this.initialized) {
|
|
this.queuedEvents.push(eventData);
|
|
return false;
|
|
}
|
|
|
|
return dispatchEvent(...eventData);
|
|
}
|
|
|
|
static bindDocument(category = document.body.dataset.page, parent = document) {
|
|
if (!this.enabled() || parent.trackingBound) return [];
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
parent.trackingBound = true;
|
|
|
|
const handlers = eventHandlers(category, (...args) => this.event(...args));
|
|
handlers.forEach((event) => parent.addEventListener(event.name, event.func));
|
|
return handlers;
|
|
}
|
|
|
|
static trackLoadEvents(category = document.body.dataset.page, parent = document) {
|
|
if (!this.enabled()) return [];
|
|
|
|
const loadEvents = parent.querySelectorAll(
|
|
'[data-track-action="render"], [data-track-event="render"]',
|
|
);
|
|
|
|
loadEvents.forEach((element) => {
|
|
const { action, data } = createEventPayload(element);
|
|
this.event(category, action, data);
|
|
});
|
|
|
|
return loadEvents;
|
|
}
|
|
|
|
static enableFormTracking(config, contexts = []) {
|
|
if (!this.enabled()) return;
|
|
|
|
if (!Array.isArray(config?.forms?.allow) && !Array.isArray(config?.fields?.allow)) {
|
|
// eslint-disable-next-line @gitlab/require-i18n-strings
|
|
throw new Error('Unable to enable form event tracking without allow rules.');
|
|
}
|
|
|
|
// Ignore default/standard schema
|
|
const standardContext = getStandardContext();
|
|
const userProvidedContexts = contexts.filter(
|
|
(context) => context.schema !== standardContext.schema,
|
|
);
|
|
|
|
const mappedConfig = {
|
|
forms: { whitelist: config.forms?.allow || [] },
|
|
fields: { whitelist: config.fields?.allow || [] },
|
|
};
|
|
|
|
const enabler = () => window.snowplow('enableFormTracking', mappedConfig, userProvidedContexts);
|
|
|
|
if (document.readyState !== 'loading') enabler();
|
|
else document.addEventListener('DOMContentLoaded', enabler);
|
|
}
|
|
|
|
static mixin(opts = {}) {
|
|
return {
|
|
computed: {
|
|
trackingCategory() {
|
|
const localCategory = this.tracking ? this.tracking.category : null;
|
|
return localCategory || opts.category;
|
|
},
|
|
trackingOptions() {
|
|
const options = addExperimentContext(opts);
|
|
return { ...options, ...this.tracking };
|
|
},
|
|
},
|
|
methods: {
|
|
track(action, data = {}) {
|
|
const category = data.category || this.trackingCategory;
|
|
const options = {
|
|
...this.trackingOptions,
|
|
...data,
|
|
};
|
|
Tracking.event(category, action, options);
|
|
},
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
export function initUserTracking() {
|
|
if (!Tracking.enabled()) return;
|
|
|
|
const opts = { ...DEFAULT_SNOWPLOW_OPTIONS, ...window.snowplowOptions };
|
|
window.snowplow('newTracker', opts.namespace, opts.hostname, opts);
|
|
|
|
document.dispatchEvent(new Event('SnowplowInitialized'));
|
|
Tracking.flushPendingEvents();
|
|
}
|
|
|
|
export function initDefaultTrackers() {
|
|
if (!Tracking.enabled()) return;
|
|
|
|
const opts = { ...DEFAULT_SNOWPLOW_OPTIONS, ...window.snowplowOptions };
|
|
|
|
window.snowplow('enableActivityTracking', 30, 30);
|
|
// must be after enableActivityTracking
|
|
const standardContext = getStandardContext();
|
|
window.snowplow('trackPageView', null, [standardContext]);
|
|
|
|
if (window.snowplowOptions.formTracking) Tracking.enableFormTracking(opts.formTrackingConfig);
|
|
if (window.snowplowOptions.linkClickTracking) window.snowplow('enableLinkClickTracking');
|
|
|
|
Tracking.bindDocument();
|
|
Tracking.trackLoadEvents();
|
|
}
|