UNPKG

@twilio/voice-sdk

Version:
442 lines (439 loc) 35.8 kB
import { EventEmitter } from 'events'; import { InvalidArgumentError } from './errors/index.js'; import Mos from './rtc/mos.js'; import { getRTCStats } from './rtc/stats.js'; import { average } from './util.js'; // How many samples we use when testing metric thresholds const SAMPLE_COUNT_METRICS = 5; // How many samples that need to cross the threshold to // raise or clear a warning. const SAMPLE_COUNT_CLEAR = 0; const SAMPLE_COUNT_RAISE = 3; const SAMPLE_INTERVAL = 1000; const WARNING_TIMEOUT = 5 * 1000; const 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((highCount, value) => 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((lowCount, value) => 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; } const valueAverage = values.reduce((partialSum, value) => partialSum + value, 0) / values.length; const diffSquared = values.map((value) => Math.pow(value - valueAverage, 2)); const stdDev = Math.sqrt(diffSquared.reduce((partialSum, value) => 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((flat, current) => [...flat, ...current], []); } /** * {@link StatsMonitor} polls a peerConnection via PeerConnection.getStats * and emits warnings when stats cross the specified threshold values. */ class StatsMonitor extends EventEmitter { /** * @constructor * @param [options] - Optional settings */ constructor(options) { super(); /** * 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 || getRTCStats; this._mos = options.Mos || Mos; this._peerConnection = options.peerConnection; this._thresholds = Object.assign(Object.assign({}, DEFAULT_THRESHOLDS), options.thresholds); const thresholdSampleCounts = Object.values(this._thresholds) .map((threshold) => threshold.sampleCount) .filter((sampleCount) => !!sampleCount); this._maxSampleCount = Math.max(SAMPLE_COUNT_METRICS, ...thresholdSampleCounts); if (this._peerConnection) { this.enable(this._peerConnection); } } /** * 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 */ addVolumes(inputVolume, outputVolume) { this._inputVolumes.push(inputVolume); this._outputVolumes.push(outputVolume); } /** * Stop sampling RTC statistics for this {@link StatsMonitor}. * @returns The current {@link StatsMonitor}. */ disable() { if (this._sampleInterval) { clearInterval(this._sampleInterval); delete this._sampleInterval; } return this; } /** * Disable warnings for this {@link StatsMonitor}. * @returns The current {@link StatsMonitor}. */ disableWarnings() { 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}. */ enable(peerConnection) { if (peerConnection) { if (this._peerConnection && peerConnection !== this._peerConnection) { throw new InvalidArgumentError('Attempted to replace an existing PeerConnection in StatsMonitor.enable'); } this._peerConnection = peerConnection; } if (!this._peerConnection) { throw new 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}. */ enableWarnings() { 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 */ hasActiveWarning(statName, thresholdName) { const warningId = `${statName}:${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 */ _addSample(sample) { const 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. */ _clearWarning(statName, thresholdName, data) { const warningId = `${statName}:${thresholdName}`; const activeWarning = this._activeWarnings.get(warningId); if (!activeWarning || Date.now() - activeWarning.timeRaised < WARNING_TIMEOUT) { return; } this._activeWarnings.delete(warningId); this.emit('warning-cleared', Object.assign(Object.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. */ _createSample(stats, previousSample) { const previousBytesSent = previousSample && previousSample.totals.bytesSent || 0; const previousBytesReceived = previousSample && previousSample.totals.bytesReceived || 0; const previousPacketsSent = previousSample && previousSample.totals.packetsSent || 0; const previousPacketsReceived = previousSample && previousSample.totals.packetsReceived || 0; const previousPacketsLost = previousSample && previousSample.totals.packetsLost || 0; const currentBytesSent = stats.bytesSent - previousBytesSent; const currentBytesReceived = stats.bytesReceived - previousBytesReceived; const currentPacketsSent = stats.packetsSent - previousPacketsSent; const currentPacketsReceived = stats.packetsReceived - previousPacketsReceived; const currentPacketsLost = stats.packetsLost - previousPacketsLost; const currentInboundPackets = currentPacketsReceived + currentPacketsLost; const currentPacketsLostFraction = (currentInboundPackets > 0) ? (currentPacketsLost / currentInboundPackets) * 100 : 0; const totalInboundPackets = stats.packetsReceived + stats.packetsLost; const totalPacketsLostFraction = (totalInboundPackets > 0) ? (stats.packetsLost / totalInboundPackets) * 100 : 100; const rttValue = (typeof stats.rtt === 'number' || !previousSample) ? stats.rtt : previousSample.rtt; const audioInputLevelValues = this._inputVolumes.splice(0); this._supplementalSampleBuffers.audioInputLevel.push(audioInputLevelValues); const audioOutputLevelValues = this._outputVolumes.splice(0); this._supplementalSampleBuffers.audioOutputLevel.push(audioOutputLevelValues); return { audioInputLevel: Math.round(average(audioInputLevelValues)), audioOutputLevel: Math.round(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. */ _fetchSample() { this._getSample().then(sample => { this._addSample(sample); this._raiseWarnings(); this.emit('sample', sample); }).catch(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. */ _getSample() { return this._getRTCStats(this._peerConnection).then((stats) => { let 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. */ _raiseWarning(statName, thresholdName, data) { const warningId = `${statName}:${thresholdName}`; if (this._activeWarnings.has(warningId)) { return; } this._activeWarnings.set(warningId, { timeRaised: Date.now() }); const thresholds = this._thresholds[statName]; let thresholdValue; if (Array.isArray(thresholds)) { const foundThreshold = thresholds.find(threshold => thresholdName in threshold); if (foundThreshold) { thresholdValue = foundThreshold[thresholdName]; } } else { thresholdValue = this._thresholds[statName][thresholdName]; } this.emit('warning', Object.assign(Object.assign({}, data), { name: statName, threshold: { name: thresholdName, value: thresholdValue, } })); } /** * Apply our thresholds to our array of RTCStat samples. */ _raiseWarnings() { if (!this._warningsEnabled) { return; } Object.keys(this._thresholds).forEach(name => 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. */ _raiseWarningsForStat(statName) { const limits = Array.isArray(this._thresholds[statName]) ? this._thresholds[statName] : [this._thresholds[statName]]; limits.forEach((limit) => { const samples = this._sampleBuffer; const clearCount = limit.clearCount || SAMPLE_COUNT_CLEAR; const raiseCount = limit.raiseCount || SAMPLE_COUNT_RAISE; const sampleCount = limit.sampleCount || this._maxSampleCount; let relevantSamples = samples.slice(-sampleCount); const values = relevantSamples.map(sample => 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. const containsNull = values.some(value => typeof value === 'undefined' || value === null); if (containsNull) { return; } let count; if (typeof limit.max === 'number') { count = countHigh(limit.max, values); if (count >= raiseCount) { this._raiseWarning(statName, 'max', { values, samples: relevantSamples }); } else if (count <= clearCount) { this._clearWarning(statName, 'max', { values, samples: relevantSamples }); } } if (typeof limit.min === 'number') { count = countLow(limit.min, values); if (count >= raiseCount) { this._raiseWarning(statName, 'min', { values, samples: relevantSamples }); } else if (count <= clearCount) { this._clearWarning(statName, 'min', { values, samples: relevantSamples }); } } if (typeof limit.maxDuration === 'number' && samples.length > 1) { relevantSamples = samples.slice(-2); const prevValue = relevantSamples[0][statName]; const curValue = relevantSamples[1][statName]; const prevStreak = this._currentStreaks.get(statName) || 0; const 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') { const sampleSets = this._supplementalSampleBuffers[statName]; if (!sampleSets || sampleSets.length < limit.sampleCount) { return; } if (sampleSets.length > limit.sampleCount) { sampleSets.splice(0, sampleSets.length - limit.sampleCount); } const flatSamples = flattenSamples(sampleSets.slice(-sampleCount)); const 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', (x, y) => x > y], ['minAverage', (x, y) => x < y], ].forEach(([thresholdName, comparator]) => { if (typeof limit[thresholdName] === 'number' && values.length >= sampleCount) { const avg = average(values); if (comparator(avg, limit[thresholdName])) { this._raiseWarning(statName, thresholdName, { values, samples: relevantSamples }); } else if (!comparator(avg, limit.clearValue || limit[thresholdName])) { this._clearWarning(statName, thresholdName, { values, samples: relevantSamples }); } } }); }); } } export { StatsMonitor as default }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"statsMonitor.js","sources":["../../lib/twilio/statsMonitor.ts"],"sourcesContent":[null],"names":[],"mappings":"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"}