UNPKG

matrix-react-sdk

Version:
153 lines (144 loc) 18.3 kB
"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":[]}