238 lines
5.8 KiB
JavaScript
238 lines
5.8 KiB
JavaScript
Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
|
const utils = require('@sentry/utils');
|
|
const debugBuild = require('../common/debug-build.js');
|
|
const getCLS = require('./web-vitals/getCLS.js');
|
|
const getFID = require('./web-vitals/getFID.js');
|
|
const getINP = require('./web-vitals/getINP.js');
|
|
const getLCP = require('./web-vitals/getLCP.js');
|
|
const observe = require('./web-vitals/lib/observe.js');
|
|
const onTTFB = require('./web-vitals/onTTFB.js');
|
|
|
|
const handlers = {};
|
|
const instrumented = {};
|
|
|
|
let _previousCls;
|
|
let _previousFid;
|
|
let _previousLcp;
|
|
let _previousTtfb;
|
|
let _previousInp;
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a CLS metric is available.
|
|
* Returns a cleanup callback which can be called to remove the instrumentation handler.
|
|
*
|
|
* Pass `stopOnCallback = true` to stop listening for CLS when the cleanup callback is called.
|
|
* This will lead to the CLS being finalized and frozen.
|
|
*/
|
|
function addClsInstrumentationHandler(
|
|
callback,
|
|
stopOnCallback = false,
|
|
) {
|
|
return addMetricObserver('cls', callback, instrumentCls, _previousCls, stopOnCallback);
|
|
}
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a LCP metric is available.
|
|
* Returns a cleanup callback which can be called to remove the instrumentation handler.
|
|
*
|
|
* Pass `stopOnCallback = true` to stop listening for LCP when the cleanup callback is called.
|
|
* This will lead to the LCP being finalized and frozen.
|
|
*/
|
|
function addLcpInstrumentationHandler(
|
|
callback,
|
|
stopOnCallback = false,
|
|
) {
|
|
return addMetricObserver('lcp', callback, instrumentLcp, _previousLcp, stopOnCallback);
|
|
}
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a FID metric is available.
|
|
*/
|
|
function addTtfbInstrumentationHandler(callback) {
|
|
return addMetricObserver('ttfb', callback, instrumentTtfb, _previousTtfb);
|
|
}
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a FID metric is available.
|
|
* Returns a cleanup callback which can be called to remove the instrumentation handler.
|
|
*/
|
|
function addFidInstrumentationHandler(callback) {
|
|
return addMetricObserver('fid', callback, instrumentFid, _previousFid);
|
|
}
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a INP metric is available.
|
|
* Returns a cleanup callback which can be called to remove the instrumentation handler.
|
|
*/
|
|
function addInpInstrumentationHandler(
|
|
callback,
|
|
) {
|
|
return addMetricObserver('inp', callback, instrumentInp, _previousInp);
|
|
}
|
|
|
|
/**
|
|
* Add a callback that will be triggered when a performance observer is triggered,
|
|
* and receives the entries of the observer.
|
|
* Returns a cleanup callback which can be called to remove the instrumentation handler.
|
|
*/
|
|
function addPerformanceInstrumentationHandler(
|
|
type,
|
|
callback,
|
|
) {
|
|
addHandler(type, callback);
|
|
|
|
if (!instrumented[type]) {
|
|
instrumentPerformanceObserver(type);
|
|
instrumented[type] = true;
|
|
}
|
|
|
|
return getCleanupCallback(type, callback);
|
|
}
|
|
|
|
/** Trigger all handlers of a given type. */
|
|
function triggerHandlers(type, data) {
|
|
const typeHandlers = handlers[type];
|
|
|
|
if (!typeHandlers || !typeHandlers.length) {
|
|
return;
|
|
}
|
|
|
|
for (const handler of typeHandlers) {
|
|
try {
|
|
handler(data);
|
|
} catch (e) {
|
|
debugBuild.DEBUG_BUILD &&
|
|
utils.logger.error(
|
|
`Error while triggering instrumentation handler.\nType: ${type}\nName: ${utils.getFunctionName(handler)}\nError:`,
|
|
e,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
function instrumentCls() {
|
|
return getCLS.onCLS(
|
|
metric => {
|
|
triggerHandlers('cls', {
|
|
metric,
|
|
});
|
|
_previousCls = metric;
|
|
},
|
|
{ reportAllChanges: true },
|
|
);
|
|
}
|
|
|
|
function instrumentFid() {
|
|
return getFID.onFID(metric => {
|
|
triggerHandlers('fid', {
|
|
metric,
|
|
});
|
|
_previousFid = metric;
|
|
});
|
|
}
|
|
|
|
function instrumentLcp() {
|
|
return getLCP.onLCP(metric => {
|
|
triggerHandlers('lcp', {
|
|
metric,
|
|
});
|
|
_previousLcp = metric;
|
|
});
|
|
}
|
|
|
|
function instrumentTtfb() {
|
|
return onTTFB.onTTFB(metric => {
|
|
triggerHandlers('ttfb', {
|
|
metric,
|
|
});
|
|
_previousTtfb = metric;
|
|
});
|
|
}
|
|
|
|
function instrumentInp() {
|
|
return getINP.onINP(metric => {
|
|
triggerHandlers('inp', {
|
|
metric,
|
|
});
|
|
_previousInp = metric;
|
|
});
|
|
}
|
|
|
|
function addMetricObserver(
|
|
type,
|
|
callback,
|
|
instrumentFn,
|
|
previousValue,
|
|
stopOnCallback = false,
|
|
) {
|
|
addHandler(type, callback);
|
|
|
|
let stopListening;
|
|
|
|
if (!instrumented[type]) {
|
|
stopListening = instrumentFn();
|
|
instrumented[type] = true;
|
|
}
|
|
|
|
if (previousValue) {
|
|
callback({ metric: previousValue });
|
|
}
|
|
|
|
return getCleanupCallback(type, callback, stopOnCallback ? stopListening : undefined);
|
|
}
|
|
|
|
function instrumentPerformanceObserver(type) {
|
|
const options = {};
|
|
|
|
// Special per-type options we want to use
|
|
if (type === 'event') {
|
|
options.durationThreshold = 0;
|
|
}
|
|
|
|
observe.observe(
|
|
type,
|
|
entries => {
|
|
triggerHandlers(type, { entries });
|
|
},
|
|
options,
|
|
);
|
|
}
|
|
|
|
function addHandler(type, handler) {
|
|
handlers[type] = handlers[type] || [];
|
|
(handlers[type] ).push(handler);
|
|
}
|
|
|
|
// Get a callback which can be called to remove the instrumentation handler
|
|
function getCleanupCallback(
|
|
type,
|
|
callback,
|
|
stopListening,
|
|
) {
|
|
return () => {
|
|
if (stopListening) {
|
|
stopListening();
|
|
}
|
|
|
|
const typeHandlers = handlers[type];
|
|
|
|
if (!typeHandlers) {
|
|
return;
|
|
}
|
|
|
|
const index = typeHandlers.indexOf(callback);
|
|
if (index !== -1) {
|
|
typeHandlers.splice(index, 1);
|
|
}
|
|
};
|
|
}
|
|
|
|
exports.addClsInstrumentationHandler = addClsInstrumentationHandler;
|
|
exports.addFidInstrumentationHandler = addFidInstrumentationHandler;
|
|
exports.addInpInstrumentationHandler = addInpInstrumentationHandler;
|
|
exports.addLcpInstrumentationHandler = addLcpInstrumentationHandler;
|
|
exports.addPerformanceInstrumentationHandler = addPerformanceInstrumentationHandler;
|
|
exports.addTtfbInstrumentationHandler = addTtfbInstrumentationHandler;
|
|
//# sourceMappingURL=instrument.js.map
|