UNPKG

@twilio/voice-sdk

Version:
484 lines 39.7 kB
"use strict"; var __extends = (this && this.__extends) || (function () { var extendStatics = function (d, b) { extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; return extendStatics(d, b); }; return function (d, b) { if (typeof b !== "function" && b !== null) throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); extendStatics(d, b); function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; })(); var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) { if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) { if (ar || !(i in from)) { if (!ar) ar = Array.prototype.slice.call(from, 0, i); ar[i] = from[i]; } } return to.concat(ar || Array.prototype.slice.call(from)); }; Object.defineProperty(exports, "__esModule", { value: true }); var events_1 = require("events"); var errors_1 = require("./errors"); var mos_1 = require("./rtc/mos"); var stats_1 = require("./rtc/stats"); var util_1 = require("./util"); // How many samples we use when testing metric thresholds var SAMPLE_COUNT_METRICS = 5; // How many samples that need to cross the threshold to // raise or clear a warning. var SAMPLE_COUNT_CLEAR = 0; var SAMPLE_COUNT_RAISE = 3; var SAMPLE_INTERVAL = 1000; var WARNING_TIMEOUT = 5 * 1000; var DEFAULT_THRESHOLDS = { audioInputLevel: { minStandardDeviation: 327.67, sampleCount: 10 }, audioOutputLevel: { minStandardDeviation: 327.67, sampleCount: 10 }, bytesReceived: { clearCount: 2, min: 1, raiseCount: 3, sampleCount: 3 }, bytesSent: { clearCount: 2, min: 1, raiseCount: 3, sampleCount: 3 }, jitter: { max: 30 }, mos: { min: 3 }, packetsLostFraction: [{ max: 1, }, { clearValue: 1, maxAverage: 3, sampleCount: 7, }], rtt: { max: 400 }, }; /** * Count the number of values that cross the max threshold. * @private * @param max - The max allowable value. * @param values - The values to iterate over. * @returns The amount of values in which the stat crossed the threshold. */ function countHigh(max, values) { return values.reduce(function (highCount, value) { return highCount += (value > max) ? 1 : 0; }, 0); } /** * Count the number of values that cross the min threshold. * @private * @param min - The minimum allowable value. * @param values - The values to iterate over. * @returns The amount of values in which the stat crossed the threshold. */ function countLow(min, values) { return values.reduce(function (lowCount, value) { return lowCount += (value < min) ? 1 : 0; }, 0); } /** * Calculate the standard deviation from a list of numbers. * @private * @param values The list of numbers to calculate the standard deviation from. * @returns The standard deviation of a list of numbers. */ function calculateStandardDeviation(values) { if (values.length <= 0) { return null; } var valueAverage = values.reduce(function (partialSum, value) { return partialSum + value; }, 0) / values.length; var diffSquared = values.map(function (value) { return Math.pow(value - valueAverage, 2); }); var stdDev = Math.sqrt(diffSquared.reduce(function (partialSum, value) { return partialSum + value; }, 0) / diffSquared.length); return stdDev; } /** * Flatten a set of numerical sample sets into a single array of samples. * @param sampleSets */ function flattenSamples(sampleSets) { return sampleSets.reduce(function (flat, current) { return __spreadArray(__spreadArray([], flat, true), current, true); }, []); } /** * {@link StatsMonitor} polls a peerConnection via PeerConnection.getStats * and emits warnings when stats cross the specified threshold values. */ var StatsMonitor = /** @class */ (function (_super) { __extends(StatsMonitor, _super); /** * @constructor * @param [options] - Optional settings */ function StatsMonitor(options) { var _this = _super.call(this) || this; /** * A map of warnings with their raised time */ _this._activeWarnings = new Map(); /** * A map of stats with the number of exceeded thresholds */ _this._currentStreaks = new Map(); /** * Keeps track of input volumes in the last second */ _this._inputVolumes = []; /** * Keeps track of output volumes in the last second */ _this._outputVolumes = []; /** * Sample buffer. Saves most recent samples */ _this._sampleBuffer = []; /** * Keeps track of supplemental sample values. * * Currently used for constant audio detection. Contains an array of volume * samples for each sample interval. */ _this._supplementalSampleBuffers = { audioInputLevel: [], audioOutputLevel: [], }; /** * Whether warnings should be enabled */ _this._warningsEnabled = true; options = options || {}; _this._getRTCStats = options.getRTCStats || stats_1.getRTCStats; _this._mos = options.Mos || mos_1.default; _this._peerConnection = options.peerConnection; _this._thresholds = __assign(__assign({}, DEFAULT_THRESHOLDS), options.thresholds); var thresholdSampleCounts = Object.values(_this._thresholds) .map(function (threshold) { return threshold.sampleCount; }) .filter(function (sampleCount) { return !!sampleCount; }); _this._maxSampleCount = Math.max.apply(Math, __spreadArray([SAMPLE_COUNT_METRICS], thresholdSampleCounts, false)); if (_this._peerConnection) { _this.enable(_this._peerConnection); } return _this; } /** * Called when a volume sample is available * @param inputVolume - Input volume level from 0 to 32767 * @param outputVolume - Output volume level from 0 to 32767 */ StatsMonitor.prototype.addVolumes = function (inputVolume, outputVolume) { this._inputVolumes.push(inputVolume); this._outputVolumes.push(outputVolume); }; /** * Stop sampling RTC statistics for this {@link StatsMonitor}. * @returns The current {@link StatsMonitor}. */ StatsMonitor.prototype.disable = function () { if (this._sampleInterval) { clearInterval(this._sampleInterval); delete this._sampleInterval; } return this; }; /** * Disable warnings for this {@link StatsMonitor}. * @returns The current {@link StatsMonitor}. */ StatsMonitor.prototype.disableWarnings = function () { if (this._warningsEnabled) { this._activeWarnings.clear(); } this._warningsEnabled = false; return this; }; /** * Start sampling RTC statistics for this {@link StatsMonitor}. * @param peerConnection - A PeerConnection to monitor. * @returns The current {@link StatsMonitor}. */ StatsMonitor.prototype.enable = function (peerConnection) { if (peerConnection) { if (this._peerConnection && peerConnection !== this._peerConnection) { throw new errors_1.InvalidArgumentError('Attempted to replace an existing PeerConnection in StatsMonitor.enable'); } this._peerConnection = peerConnection; } if (!this._peerConnection) { throw new errors_1.InvalidArgumentError('Can not enable StatsMonitor without a PeerConnection'); } this._sampleInterval = this._sampleInterval || setInterval(this._fetchSample.bind(this), SAMPLE_INTERVAL); return this; }; /** * Enable warnings for this {@link StatsMonitor}. * @returns The current {@link StatsMonitor}. */ StatsMonitor.prototype.enableWarnings = function () { this._warningsEnabled = true; return this; }; /** * Check if there is an active warning for a specific stat and threshold * @param statName - The name of the stat to check * @param thresholdName - The name of the threshold to check * @returns Whether there is an active warning for a specific stat and threshold */ StatsMonitor.prototype.hasActiveWarning = function (statName, thresholdName) { var warningId = "".concat(statName, ":").concat(thresholdName); return !!this._activeWarnings.get(warningId); }; /** * Add a sample to our sample buffer and remove the oldest if we are over the limit. * @param sample - Sample to add */ StatsMonitor.prototype._addSample = function (sample) { var samples = this._sampleBuffer; samples.push(sample); // We store 1 extra sample so that we always have (current, previous) // available for all {sampleBufferSize} threshold validations. if (samples.length > this._maxSampleCount) { samples.splice(0, samples.length - this._maxSampleCount); } }; /** * Clear an active warning. * @param statName - The name of the stat to clear. * @param thresholdName - The name of the threshold to clear * @param [data] - Any relevant sample data. */ StatsMonitor.prototype._clearWarning = function (statName, thresholdName, data) { var warningId = "".concat(statName, ":").concat(thresholdName); var activeWarning = this._activeWarnings.get(warningId); if (!activeWarning || Date.now() - activeWarning.timeRaised < WARNING_TIMEOUT) { return; } this._activeWarnings.delete(warningId); this.emit('warning-cleared', __assign(__assign({}, data), { name: statName, threshold: { name: thresholdName, value: this._thresholds[statName][thresholdName], } })); }; /** * Create a sample object from a stats object using the previous sample, if available. * @param stats - Stats retrieved from getStatistics * @param [previousSample=null] - The previous sample to use to calculate deltas. * @returns A universally-formatted version of RTC stats. */ StatsMonitor.prototype._createSample = function (stats, previousSample) { var previousBytesSent = previousSample && previousSample.totals.bytesSent || 0; var previousBytesReceived = previousSample && previousSample.totals.bytesReceived || 0; var previousPacketsSent = previousSample && previousSample.totals.packetsSent || 0; var previousPacketsReceived = previousSample && previousSample.totals.packetsReceived || 0; var previousPacketsLost = previousSample && previousSample.totals.packetsLost || 0; var currentBytesSent = stats.bytesSent - previousBytesSent; var currentBytesReceived = stats.bytesReceived - previousBytesReceived; var currentPacketsSent = stats.packetsSent - previousPacketsSent; var currentPacketsReceived = stats.packetsReceived - previousPacketsReceived; var currentPacketsLost = stats.packetsLost - previousPacketsLost; var currentInboundPackets = currentPacketsReceived + currentPacketsLost; var currentPacketsLostFraction = (currentInboundPackets > 0) ? (currentPacketsLost / currentInboundPackets) * 100 : 0; var totalInboundPackets = stats.packetsReceived + stats.packetsLost; var totalPacketsLostFraction = (totalInboundPackets > 0) ? (stats.packetsLost / totalInboundPackets) * 100 : 100; var rttValue = (typeof stats.rtt === 'number' || !previousSample) ? stats.rtt : previousSample.rtt; var audioInputLevelValues = this._inputVolumes.splice(0); this._supplementalSampleBuffers.audioInputLevel.push(audioInputLevelValues); var audioOutputLevelValues = this._outputVolumes.splice(0); this._supplementalSampleBuffers.audioOutputLevel.push(audioOutputLevelValues); return { audioInputLevel: Math.round((0, util_1.average)(audioInputLevelValues)), audioOutputLevel: Math.round((0, util_1.average)(audioOutputLevelValues)), bytesReceived: currentBytesReceived, bytesSent: currentBytesSent, codecName: stats.codecName, jitter: stats.jitter, mos: this._mos.calculate(rttValue, stats.jitter, previousSample && currentPacketsLostFraction), packetsLost: currentPacketsLost, packetsLostFraction: currentPacketsLostFraction, packetsReceived: currentPacketsReceived, packetsSent: currentPacketsSent, rtt: rttValue, timestamp: stats.timestamp, totals: { bytesReceived: stats.bytesReceived, bytesSent: stats.bytesSent, packetsLost: stats.packetsLost, packetsLostFraction: totalPacketsLostFraction, packetsReceived: stats.packetsReceived, packetsSent: stats.packetsSent, }, }; }; /** * Get stats from the PeerConnection and add it to our list of samples. */ StatsMonitor.prototype._fetchSample = function () { var _this = this; this._getSample().then(function (sample) { _this._addSample(sample); _this._raiseWarnings(); _this.emit('sample', sample); }).catch(function (error) { _this.disable(); // We only bubble up any errors coming from pc.getStats() // No need to attach a twilioError _this.emit('error', error); }); }; /** * Get stats from the PeerConnection. * @returns A universally-formatted version of RTC stats. */ StatsMonitor.prototype._getSample = function () { var _this = this; return this._getRTCStats(this._peerConnection).then(function (stats) { var previousSample = null; if (_this._sampleBuffer.length) { previousSample = _this._sampleBuffer[_this._sampleBuffer.length - 1]; } return _this._createSample(stats, previousSample); }); }; /** * Raise a warning and log its raised time. * @param statName - The name of the stat to raise. * @param thresholdName - The name of the threshold to raise * @param [data] - Any relevant sample data. */ StatsMonitor.prototype._raiseWarning = function (statName, thresholdName, data) { var warningId = "".concat(statName, ":").concat(thresholdName); if (this._activeWarnings.has(warningId)) { return; } this._activeWarnings.set(warningId, { timeRaised: Date.now() }); var thresholds = this._thresholds[statName]; var thresholdValue; if (Array.isArray(thresholds)) { var foundThreshold = thresholds.find(function (threshold) { return thresholdName in threshold; }); if (foundThreshold) { thresholdValue = foundThreshold[thresholdName]; } } else { thresholdValue = this._thresholds[statName][thresholdName]; } this.emit('warning', __assign(__assign({}, data), { name: statName, threshold: { name: thresholdName, value: thresholdValue, } })); }; /** * Apply our thresholds to our array of RTCStat samples. */ StatsMonitor.prototype._raiseWarnings = function () { var _this = this; if (!this._warningsEnabled) { return; } Object.keys(this._thresholds).forEach(function (name) { return _this._raiseWarningsForStat(name); }); }; /** * Apply thresholds for a given stat name to our array of * RTCStat samples and raise or clear any associated warnings. * @param statName - Name of the stat to compare. */ StatsMonitor.prototype._raiseWarningsForStat = function (statName) { var _this = this; var limits = Array.isArray(this._thresholds[statName]) ? this._thresholds[statName] : [this._thresholds[statName]]; limits.forEach(function (limit) { var samples = _this._sampleBuffer; var clearCount = limit.clearCount || SAMPLE_COUNT_CLEAR; var raiseCount = limit.raiseCount || SAMPLE_COUNT_RAISE; var sampleCount = limit.sampleCount || _this._maxSampleCount; var relevantSamples = samples.slice(-sampleCount); var values = relevantSamples.map(function (sample) { return sample[statName]; }); // (rrowland) If we have a bad or missing value in the set, we don't // have enough information to throw or clear a warning. Bail out. var containsNull = values.some(function (value) { return typeof value === 'undefined' || value === null; }); if (containsNull) { return; } var count; if (typeof limit.max === 'number') { count = countHigh(limit.max, values); if (count >= raiseCount) { _this._raiseWarning(statName, 'max', { values: values, samples: relevantSamples }); } else if (count <= clearCount) { _this._clearWarning(statName, 'max', { values: values, samples: relevantSamples }); } } if (typeof limit.min === 'number') { count = countLow(limit.min, values); if (count >= raiseCount) { _this._raiseWarning(statName, 'min', { values: values, samples: relevantSamples }); } else if (count <= clearCount) { _this._clearWarning(statName, 'min', { values: values, samples: relevantSamples }); } } if (typeof limit.maxDuration === 'number' && samples.length > 1) { relevantSamples = samples.slice(-2); var prevValue = relevantSamples[0][statName]; var curValue = relevantSamples[1][statName]; var prevStreak = _this._currentStreaks.get(statName) || 0; var streak = (prevValue === curValue) ? prevStreak + 1 : 0; _this._currentStreaks.set(statName, streak); if (streak >= limit.maxDuration) { _this._raiseWarning(statName, 'maxDuration', { value: streak }); } else if (streak === 0) { _this._clearWarning(statName, 'maxDuration', { value: prevStreak }); } } if (typeof limit.minStandardDeviation === 'number') { var sampleSets = _this._supplementalSampleBuffers[statName]; if (!sampleSets || sampleSets.length < limit.sampleCount) { return; } if (sampleSets.length > limit.sampleCount) { sampleSets.splice(0, sampleSets.length - limit.sampleCount); } var flatSamples = flattenSamples(sampleSets.slice(-sampleCount)); var stdDev = calculateStandardDeviation(flatSamples); if (typeof stdDev !== 'number') { return; } if (stdDev < limit.minStandardDeviation) { _this._raiseWarning(statName, 'minStandardDeviation', { value: stdDev }); } else { _this._clearWarning(statName, 'minStandardDeviation', { value: stdDev }); } } [ ['maxAverage', function (x, y) { return x > y; }], ['minAverage', function (x, y) { return x < y; }], ].forEach(function (_a) { var thresholdName = _a[0], comparator = _a[1]; if (typeof limit[thresholdName] === 'number' && values.length >= sampleCount) { var avg = (0, util_1.average)(values); if (comparator(avg, limit[thresholdName])) { _this._raiseWarning(statName, thresholdName, { values: values, samples: relevantSamples }); } else if (!comparator(avg, limit.clearValue || limit[thresholdName])) { _this._clearWarning(statName, thresholdName, { values: values, samples: relevantSamples }); } } }); }); }; return StatsMonitor; }(events_1.EventEmitter)); exports.default = StatsMonitor; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"statsMonitor.js","sourceRoot":"","sources":["../../lib/twilio/statsMonitor.ts"],"names":[],"mappings":"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"}