matrix-react-sdk
Version:
SDK for matrix.org using React
153 lines (144 loc) • 18.3 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "PerformanceEntryNames", {
enumerable: true,
get: function () {
return _entryNames.PerformanceEntryNames;
}
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _logger = require("matrix-js-sdk/src/logger");
var _entryNames = require("./entry-names");
/*
Copyright 2024 New Vector Ltd.
Copyright 2021 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
class PerformanceMonitor {
constructor() {
(0, _defineProperty2.default)(this, "START_PREFIX", "start:");
(0, _defineProperty2.default)(this, "STOP_PREFIX", "stop:");
(0, _defineProperty2.default)(this, "listeners", []);
(0, _defineProperty2.default)(this, "entries", []);
}
static get instance() {
if (!PerformanceMonitor._instance) {
PerformanceMonitor._instance = new PerformanceMonitor();
}
return PerformanceMonitor._instance;
}
/**
* Starts a performance recording
* @param name Name of the recording
* @param id Specify an identifier appended to the measurement name
* @returns {void}
*/
start(name, id) {
if (!this.supportsPerformanceApi()) {
return;
}
const key = this.buildKey(name, id);
if (performance.getEntriesByName(this.START_PREFIX + key).length > 0) {
_logger.logger.warn(`Recording already started for: ${name}`);
return;
}
performance.mark(this.START_PREFIX + key);
}
/**
* Stops a performance recording and stores delta duration
* with the start marker
* @param name Name of the recording
* @param id Specify an identifier appended to the measurement name
* @returns The measurement
*/
stop(name, id) {
if (!this.supportsPerformanceApi()) {
return;
}
const key = this.buildKey(name, id);
if (performance.getEntriesByName(this.START_PREFIX + key).length === 0) {
_logger.logger.warn(`No recording started for: ${name}`);
return;
}
performance.mark(this.STOP_PREFIX + key);
performance.measure(key, this.START_PREFIX + key, this.STOP_PREFIX + key);
this.clear(name, id);
const measurement = performance.getEntriesByName(key).pop();
// Keeping a reference to all PerformanceEntry created
// by this abstraction for historical events collection
// when adding a data callback
this.entries.push(measurement);
this.listeners.forEach(listener => {
if (this.shouldEmit(listener, measurement)) {
listener.callback([measurement]);
}
});
return measurement;
}
clear(name, id) {
if (!this.supportsPerformanceApi()) {
return;
}
const key = this.buildKey(name, id);
performance.clearMarks(this.START_PREFIX + key);
performance.clearMarks(this.STOP_PREFIX + key);
}
getEntries({
name,
type
} = {}) {
return this.entries.filter(entry => {
const satisfiesName = !name || entry.name === name;
const satisfiedType = !type || entry.entryType === type;
return satisfiesName && satisfiedType;
});
}
addPerformanceDataCallback(listener, buffer = false) {
this.listeners.push(listener);
if (buffer) {
const toEmit = this.entries.filter(entry => this.shouldEmit(listener, entry));
if (toEmit.length > 0) {
listener.callback(toEmit);
}
}
}
removePerformanceDataCallback(callback) {
if (!callback) {
this.listeners = [];
} else {
this.listeners.splice(this.listeners.findIndex(listener => listener.callback === callback), 1);
}
}
/**
* Tor browser does not support the Performance API
* @returns {boolean} true if the Performance API is supported
*/
supportsPerformanceApi() {
return performance !== undefined && performance.mark !== undefined;
}
shouldEmit(listener, entry) {
return !listener.entryNames || listener.entryNames.includes(entry.name);
}
/**
* Internal utility to ensure consistent name for the recording
* @param name Name of the recording
* @param id Specify an identifier appended to the measurement name
* @returns {string} a compound of the name and identifier if present
*/
buildKey(name, id) {
const suffix = id ? `:${id}` : "";
return `${name}${suffix}`;
}
}
// Convenience exports
exports.default = PerformanceMonitor;
(0, _defineProperty2.default)(PerformanceMonitor, "_instance", void 0);
// Exposing those to the window object to bridge them from tests
window.mxPerformanceMonitor = PerformanceMonitor.instance;
window.mxPerformanceEntryNames = _entryNames.PerformanceEntryNames;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_logger","require","_entryNames","PerformanceMonitor","constructor","_defineProperty2","default","instance","_instance","start","name","id","supportsPerformanceApi","key","buildKey","performance","getEntriesByName","START_PREFIX","length","logger","warn","mark","stop","STOP_PREFIX","measure","clear","measurement","pop","entries","push","listeners","forEach","listener","shouldEmit","callback","clearMarks","getEntries","type","filter","entry","satisfiesName","satisfiedType","entryType","addPerformanceDataCallback","buffer","toEmit","removePerformanceDataCallback","splice","findIndex","undefined","entryNames","includes","suffix","exports","window","mxPerformanceMonitor","mxPerformanceEntryNames","PerformanceEntryNames"],"sources":["../../src/performance/index.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport { PerformanceEntryNames } from \"./entry-names\";\n\ninterface GetEntriesOptions {\n    name?: string;\n    type?: string;\n}\n\ntype PerformanceCallbackFunction = (entry: PerformanceEntry[]) => void;\n\ninterface PerformanceDataListener {\n    entryNames?: string[];\n    callback: PerformanceCallbackFunction;\n}\n\nexport default class PerformanceMonitor {\n    private static _instance: PerformanceMonitor;\n\n    private START_PREFIX = \"start:\";\n    private STOP_PREFIX = \"stop:\";\n\n    private listeners: PerformanceDataListener[] = [];\n    private entries: PerformanceEntry[] = [];\n\n    public static get instance(): PerformanceMonitor {\n        if (!PerformanceMonitor._instance) {\n            PerformanceMonitor._instance = new PerformanceMonitor();\n        }\n        return PerformanceMonitor._instance;\n    }\n\n    /**\n     * Starts a performance recording\n     * @param name Name of the recording\n     * @param id Specify an identifier appended to the measurement name\n     * @returns {void}\n     */\n    public start(name: string, id?: string): void {\n        if (!this.supportsPerformanceApi()) {\n            return;\n        }\n        const key = this.buildKey(name, id);\n\n        if (performance.getEntriesByName(this.START_PREFIX + key).length > 0) {\n            logger.warn(`Recording already started for: ${name}`);\n            return;\n        }\n\n        performance.mark(this.START_PREFIX + key);\n    }\n\n    /**\n     * Stops a performance recording and stores delta duration\n     * with the start marker\n     * @param name Name of the recording\n     * @param id Specify an identifier appended to the measurement name\n     * @returns The measurement\n     */\n    public stop(name: string, id?: string): PerformanceEntry | undefined {\n        if (!this.supportsPerformanceApi()) {\n            return;\n        }\n        const key = this.buildKey(name, id);\n        if (performance.getEntriesByName(this.START_PREFIX + key).length === 0) {\n            logger.warn(`No recording started for: ${name}`);\n            return;\n        }\n\n        performance.mark(this.STOP_PREFIX + key);\n        performance.measure(key, this.START_PREFIX + key, this.STOP_PREFIX + key);\n\n        this.clear(name, id);\n\n        const measurement = performance.getEntriesByName(key).pop()!;\n\n        // Keeping a reference to all PerformanceEntry created\n        // by this abstraction for historical events collection\n        // when adding a data callback\n        this.entries.push(measurement);\n\n        this.listeners.forEach((listener) => {\n            if (this.shouldEmit(listener, measurement)) {\n                listener.callback([measurement]);\n            }\n        });\n\n        return measurement;\n    }\n\n    public clear(name: string, id?: string): void {\n        if (!this.supportsPerformanceApi()) {\n            return;\n        }\n        const key = this.buildKey(name, id);\n        performance.clearMarks(this.START_PREFIX + key);\n        performance.clearMarks(this.STOP_PREFIX + key);\n    }\n\n    public getEntries({ name, type }: GetEntriesOptions = {}): PerformanceEntry[] {\n        return this.entries.filter((entry) => {\n            const satisfiesName = !name || entry.name === name;\n            const satisfiedType = !type || entry.entryType === type;\n            return satisfiesName && satisfiedType;\n        });\n    }\n\n    public addPerformanceDataCallback(listener: PerformanceDataListener, buffer = false): void {\n        this.listeners.push(listener);\n        if (buffer) {\n            const toEmit = this.entries.filter((entry) => this.shouldEmit(listener, entry));\n            if (toEmit.length > 0) {\n                listener.callback(toEmit);\n            }\n        }\n    }\n\n    public removePerformanceDataCallback(callback?: PerformanceCallbackFunction): void {\n        if (!callback) {\n            this.listeners = [];\n        } else {\n            this.listeners.splice(\n                this.listeners.findIndex((listener) => listener.callback === callback),\n                1,\n            );\n        }\n    }\n\n    /**\n     * Tor browser does not support the Performance API\n     * @returns {boolean} true if the Performance API is supported\n     */\n    private supportsPerformanceApi(): boolean {\n        return performance !== undefined && performance.mark !== undefined;\n    }\n\n    private shouldEmit(listener: PerformanceDataListener, entry: PerformanceEntry): boolean {\n        return !listener.entryNames || listener.entryNames.includes(entry.name);\n    }\n\n    /**\n     * Internal utility to ensure consistent name for the recording\n     * @param name Name of the recording\n     * @param id Specify an identifier appended to the measurement name\n     * @returns {string} a compound of the name and identifier if present\n     */\n    private buildKey(name: string, id?: string): string {\n        const suffix = id ? `:${id}` : \"\";\n        return `${name}${suffix}`;\n    }\n}\n\n// Convenience exports\nexport { PerformanceEntryNames };\n\n// Exposing those to the window object to bridge them from tests\nwindow.mxPerformanceMonitor = PerformanceMonitor.instance;\nwindow.mxPerformanceEntryNames = PerformanceEntryNames;\n"],"mappings":"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","ignoreList":[]}