@webex/plugin-meetings 3.0.0-beta.364 → 3.0.0-beta.366
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/breakouts/breakout.js +1 -1
- package/dist/breakouts/index.js +1 -1
- package/dist/interpretation/index.js +1 -1
- package/dist/interpretation/siLanguage.js +1 -1
- package/dist/rtcMetrics/index.js +38 -1
- package/dist/rtcMetrics/index.js.map +1 -1
- package/dist/statsAnalyzer/index.js +3 -2
- package/dist/statsAnalyzer/index.js.map +1 -1
- package/dist/types/rtcMetrics/index.d.ts +7 -0
- package/dist/webinar/index.js +1 -1
- package/package.json +19 -19
- package/src/rtcMetrics/index.ts +43 -1
- package/src/statsAnalyzer/index.ts +4 -3
- package/test/unit/spec/rtcMetrics/index.ts +32 -3
- package/test/unit/spec/stats-analyzer/index.js +44 -1
package/dist/breakouts/index.js
CHANGED
|
@@ -1041,7 +1041,7 @@ var Breakouts = _webexCore.WebexPlugin.extend({
|
|
|
1041
1041
|
this.trigger(_constants.BREAKOUTS.EVENTS.ASK_RETURN_TO_MAIN);
|
|
1042
1042
|
}
|
|
1043
1043
|
},
|
|
1044
|
-
version: "3.0.0-beta.
|
|
1044
|
+
version: "3.0.0-beta.366"
|
|
1045
1045
|
});
|
|
1046
1046
|
var _default = Breakouts;
|
|
1047
1047
|
exports.default = _default;
|
package/dist/rtcMetrics/index.js
CHANGED
|
@@ -11,8 +11,21 @@ var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime-corejs2/he
|
|
|
11
11
|
var _createClass2 = _interopRequireDefault(require("@babel/runtime-corejs2/helpers/createClass"));
|
|
12
12
|
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime-corejs2/helpers/defineProperty"));
|
|
13
13
|
var _internalPluginMetrics = require("@webex/internal-plugin-metrics");
|
|
14
|
+
var _uuid = _interopRequireDefault(require("uuid"));
|
|
14
15
|
var _constants = _interopRequireDefault(require("./constants"));
|
|
15
16
|
/* eslint-disable class-methods-use-this */
|
|
17
|
+
|
|
18
|
+
var parseJsonPayload = function parseJsonPayload(payload) {
|
|
19
|
+
try {
|
|
20
|
+
if (payload && payload[0]) {
|
|
21
|
+
return JSON.parse(payload[0]);
|
|
22
|
+
}
|
|
23
|
+
return null;
|
|
24
|
+
} catch (_) {
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
};
|
|
28
|
+
|
|
16
29
|
/**
|
|
17
30
|
* Rtc Metrics
|
|
18
31
|
*/
|
|
@@ -35,11 +48,13 @@ var RtcMetrics = /*#__PURE__*/function () {
|
|
|
35
48
|
(0, _defineProperty2.default)(this, "webex", void 0);
|
|
36
49
|
(0, _defineProperty2.default)(this, "meetingId", void 0);
|
|
37
50
|
(0, _defineProperty2.default)(this, "correlationId", void 0);
|
|
51
|
+
(0, _defineProperty2.default)(this, "connectionId", void 0);
|
|
38
52
|
// `window` is used to prevent typescript from returning a NodeJS.Timer.
|
|
39
53
|
this.intervalId = window.setInterval(this.sendMetricsInQueue.bind(this), 30 * 1000);
|
|
40
54
|
this.meetingId = meetingId;
|
|
41
55
|
this.webex = webex;
|
|
42
56
|
this.correlationId = correlationId;
|
|
57
|
+
this.setNewConnectionId();
|
|
43
58
|
// Send the first set of metrics at 5 seconds in the case of a user leaving the call shortly after joining.
|
|
44
59
|
setTimeout(this.sendMetricsInQueue.bind(this), 5 * 1000);
|
|
45
60
|
}
|
|
@@ -73,6 +88,16 @@ var RtcMetrics = /*#__PURE__*/function () {
|
|
|
73
88
|
data.payload = data.payload.map(this.anonymizeIp);
|
|
74
89
|
}
|
|
75
90
|
this.metricsQueue.push(data);
|
|
91
|
+
try {
|
|
92
|
+
// If a connection fails, send the rest of the metrics in queue and get a new connection id.
|
|
93
|
+
var parsedPayload = parseJsonPayload(data.payload);
|
|
94
|
+
if (data.name === 'onconnectionstatechange' && parsedPayload && parsedPayload.value === 'failed') {
|
|
95
|
+
this.sendMetricsInQueue();
|
|
96
|
+
this.setNewConnectionId();
|
|
97
|
+
}
|
|
98
|
+
} catch (e) {
|
|
99
|
+
console.error(e);
|
|
100
|
+
}
|
|
76
101
|
}
|
|
77
102
|
}
|
|
78
103
|
|
|
@@ -107,6 +132,17 @@ var RtcMetrics = /*#__PURE__*/function () {
|
|
|
107
132
|
return (0, _stringify.default)(data);
|
|
108
133
|
}
|
|
109
134
|
|
|
135
|
+
/**
|
|
136
|
+
* Set a new connection id.
|
|
137
|
+
*
|
|
138
|
+
* @returns {void}
|
|
139
|
+
*/
|
|
140
|
+
}, {
|
|
141
|
+
key: "setNewConnectionId",
|
|
142
|
+
value: function setNewConnectionId() {
|
|
143
|
+
this.connectionId = _uuid.default.v4();
|
|
144
|
+
}
|
|
145
|
+
|
|
110
146
|
/**
|
|
111
147
|
* Send metrics to the metrics service.
|
|
112
148
|
*
|
|
@@ -126,10 +162,11 @@ var RtcMetrics = /*#__PURE__*/function () {
|
|
|
126
162
|
body: {
|
|
127
163
|
metrics: [{
|
|
128
164
|
type: 'webrtc',
|
|
129
|
-
version: '1.0
|
|
165
|
+
version: '1.1.0',
|
|
130
166
|
userId: this.webex.internal.device.userId,
|
|
131
167
|
meetingId: this.meetingId,
|
|
132
168
|
correlationId: this.correlationId,
|
|
169
|
+
connectionId: this.connectionId,
|
|
133
170
|
data: this.metricsQueue
|
|
134
171
|
}]
|
|
135
172
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["RtcMetrics","webex","meetingId","correlationId","intervalId","window","setInterval","sendMetricsInQueue","bind","setTimeout","metricsQueue","length","sendMetrics","data","
|
|
1
|
+
{"version":3,"names":["parseJsonPayload","payload","JSON","parse","_","RtcMetrics","webex","meetingId","correlationId","intervalId","window","setInterval","sendMetricsInQueue","bind","setNewConnectionId","setTimeout","metricsQueue","length","sendMetrics","data","name","map","anonymizeIp","push","parsedPayload","value","e","console","error","clearInterval","stats","type","ip","CallDiagnosticUtils","anonymizeIPAddress","undefined","address","relatedAddress","connectionId","uuid","v4","request","method","service","resource","headers","appId","RTC_METRICS","APP_ID","body","metrics","version","userId","internal","device"],"sources":["index.ts"],"sourcesContent":["/* eslint-disable class-methods-use-this */\nimport {CallDiagnosticUtils} from '@webex/internal-plugin-metrics';\nimport uuid from 'uuid';\nimport RTC_METRICS from './constants';\n\nconst parseJsonPayload = (payload: any[]): any | null => {\n try {\n if (payload && payload[0]) {\n return JSON.parse(payload[0]);\n }\n\n return null;\n } catch (_) {\n return null;\n }\n};\n\n/**\n * Rtc Metrics\n */\nexport default class RtcMetrics {\n /**\n * Array of MetricData items to be sent to the metrics service.\n */\n metricsQueue = [];\n\n intervalId: number;\n\n webex: any;\n\n meetingId: string;\n\n correlationId: string;\n\n connectionId: string;\n\n /**\n * Initialize the interval.\n *\n * @param {object} webex - The main `webex` object.\n * @param {string} meetingId - The meeting id.\n * @param {string} correlationId - The correlation id.\n */\n constructor(webex, meetingId, correlationId) {\n // `window` is used to prevent typescript from returning a NodeJS.Timer.\n this.intervalId = window.setInterval(this.sendMetricsInQueue.bind(this), 30 * 1000);\n this.meetingId = meetingId;\n this.webex = webex;\n this.correlationId = correlationId;\n this.setNewConnectionId();\n // Send the first set of metrics at 5 seconds in the case of a user leaving the call shortly after joining.\n setTimeout(this.sendMetricsInQueue.bind(this), 5 * 1000);\n }\n\n /**\n * Check to see if the metrics queue has any items.\n *\n * @returns {void}\n */\n public sendMetricsInQueue() {\n if (this.metricsQueue.length) {\n this.sendMetrics();\n this.metricsQueue = [];\n }\n }\n\n /**\n * Add metrics items to the metrics queue.\n *\n * @param {object} data - An object with a payload array of metrics items.\n *\n * @returns {void}\n */\n addMetrics(data) {\n if (data.payload.length) {\n if (data.name === 'stats-report') {\n data.payload = data.payload.map(this.anonymizeIp);\n }\n\n this.metricsQueue.push(data);\n\n try {\n // If a connection fails, send the rest of the metrics in queue and get a new connection id.\n const parsedPayload = parseJsonPayload(data.payload);\n if (\n data.name === 'onconnectionstatechange' &&\n parsedPayload &&\n parsedPayload.value === 'failed'\n ) {\n this.sendMetricsInQueue();\n this.setNewConnectionId();\n }\n } catch (e) {\n console.error(e);\n }\n }\n }\n\n /**\n * Clear the metrics interval.\n *\n * @returns {void}\n */\n closeMetrics() {\n this.sendMetricsInQueue();\n clearInterval(this.intervalId);\n }\n\n /**\n * Anonymize IP addresses.\n *\n * @param {array} stats - An RTCStatsReport organized into an array of strings.\n * @returns {string}\n */\n anonymizeIp(stats: string): string {\n const data = JSON.parse(stats);\n // on local and remote candidates, anonymize the last 4 bits.\n if (data.type === 'local-candidate' || data.type === 'remote-candidate') {\n data.ip = CallDiagnosticUtils.anonymizeIPAddress(data.ip) || undefined;\n data.address = CallDiagnosticUtils.anonymizeIPAddress(data.address) || undefined;\n data.relatedAddress =\n CallDiagnosticUtils.anonymizeIPAddress(data.relatedAddress) || undefined;\n }\n\n return JSON.stringify(data);\n }\n\n /**\n * Set a new connection id.\n *\n * @returns {void}\n */\n private setNewConnectionId() {\n this.connectionId = uuid.v4();\n }\n\n /**\n * Send metrics to the metrics service.\n *\n * @returns {void}\n */\n private sendMetrics() {\n this.webex.request({\n method: 'POST',\n service: 'unifiedTelemetry',\n resource: 'metric/v2',\n headers: {\n type: 'webrtcMedia',\n appId: RTC_METRICS.APP_ID,\n },\n body: {\n metrics: [\n {\n type: 'webrtc',\n version: '1.1.0',\n userId: this.webex.internal.device.userId,\n meetingId: this.meetingId,\n correlationId: this.correlationId,\n connectionId: this.connectionId,\n data: this.metricsQueue,\n },\n ],\n },\n });\n }\n}\n"],"mappings":"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"}
|
|
@@ -823,6 +823,7 @@ var StatsAnalyzer = /*#__PURE__*/function (_EventsScope) {
|
|
|
823
823
|
if (result.bytesReceived) {
|
|
824
824
|
var kilobytes = 0;
|
|
825
825
|
var receiveSlot = this.receiveSlotCallback(result.ssrc);
|
|
826
|
+
var sourceState = receiveSlot === null || receiveSlot === void 0 ? void 0 : receiveSlot.sourceState;
|
|
826
827
|
var idAndCsi = receiveSlot ? "id: \"".concat(receiveSlot.id || '', "\"").concat(receiveSlot.csi ? " and csi: ".concat(receiveSlot.csi) : '') : '';
|
|
827
828
|
if (result.frameWidth && result.frameHeight) {
|
|
828
829
|
this.statsResults[mediaType][sendrecvType].width = result.frameWidth;
|
|
@@ -839,8 +840,8 @@ var StatsAnalyzer = /*#__PURE__*/function (_EventsScope) {
|
|
|
839
840
|
var currentPacketsReceived = result.packetsReceived - this.statsResults[mediaType][sendrecvType].totalPacketsReceived;
|
|
840
841
|
this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;
|
|
841
842
|
if (currentPacketsReceived === 0) {
|
|
842
|
-
if (receiveSlot) {
|
|
843
|
-
_loggerProxy.default.logger.info("StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ".concat(idAndCsi),
|
|
843
|
+
if (receiveSlot && sourceState === 'live') {
|
|
844
|
+
_loggerProxy.default.logger.info("StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ".concat(idAndCsi, ". Total packets received on slot: "), result.packetsReceived);
|
|
844
845
|
}
|
|
845
846
|
}
|
|
846
847
|
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"names":["EVENTS","MEDIA_QUALITY","LOCAL_MEDIA_STARTED","LOCAL_MEDIA_STOPPED","REMOTE_MEDIA_STARTED","REMOTE_MEDIA_STOPPED","emptySender","trackLabel","maxPacketLossRatio","availableBandwidth","bytesSent","meanRemoteJitter","meanRoundTripTime","emptyReceiver","bytesReceived","meanRtpJitter","StatsAnalyzer","config","receiveSlotCallback","undefined","networkQualityMonitor","statsResults","defaultStats","mediaType","previousValue","currentValue","isLocal","Error","lastEmittedStartStopEvent","lastEmittedEvent","local","remote","newEvent","emit","file","function","type","successfulCandidatePairId","candidates","newIpAddress","localCandidate","candidateType","address","relayProtocol","relatedAddress","localIpAddress","result","isSender","isRemote","id","successfulCandidatePair","localCandidateId","transport","toUpperCase","protocol","sendRecvType","STATS","SEND_DIRECTION","RECEIVE_DIRECTION","ipType","REMOTE","LOCAL","ipAddress","ip","relatedPort","portNumber","port","networkType","priority","timestamp","time","connectionType","NETWORK_TYPE","VPN","UNKNOWN","totalRoundTripTime","statsStarted","lastStatsResults","correlationId","mqaSentCount","lastMqaDataSent","forEach","includes","recv","send","status","meetingMediaStatus","newMqa","emptyMqaInterval","audioSender","emptyAudioTransmit","getAudioSenderMqa","audioTransmit","push","audioReceiver","emptyAudioReceive","getAudioReceiverMqa","audioReceive","videoSender","emptyVideoTransmit","getVideoSenderMqa","videoTransmit","videoReceiver","emptyVideoReceive","getVideoReceiverMqa","videoReceive","intervalMetadata","peerReflexiveIP","peripherals","information","_UNKNOWN_","name","MEDIA_DEVICES","SPEAKER","MICROPHONE","CAMERA","intervalNumber","resetStatsResults","data","mediaConnection","getStatsAndParse","then","statsInterval","setInterval","analyzerInterval","sendMqaData","mqaInterval","MQA_INTERVAL","resolve","sendOneLastMqa","clearInterval","getStatsResult","processOutboundRTPResult","processInboundRTPResult","compareSentAndReceived","parseCandidate","parseAudioSource","statsItem","types","DEFAULT_GET_STATS_FILTER","report","state","parseGetStatsResult","direction","currentDirection","localTrackLabel","csi","extractAndSetLocalIpAddressInfoForDiagnostics","audioLevel","totalAudioEnergy","getCurrentStatsTotals","keyPrefix","value","filter","key","startsWith","reduce","prev","cur","getPreviousStatsTotals","expected","sendAudio","currentStats","previousStats","totalPacketsSent","LoggerProxy","logger","info","emitStartStopEvents","receiveAudio","currentPacketsReceived","previousPacketsReceived","currentSamplesReceived","previousSamplesReceived","sendVideo","framesEncoded","framesSent","receiveVideo","currentFramesReceived","previousFramesReceived","currentFramesDecoded","previousFramesDecoded","currentFramesDropped","previousFramesDropped","sendShare","getConnectionState","ConnectionState","Failed","trace","getTransceiverStats","transceiverStats","video","receivers","receiver","i","filterAndParseGetStatsResults","audio","screenShareVideo","screenShareAudio","senders","sender","compareLastStatsResult","JSON","parse","sendrecvType","kilobytes","frameWidth","frameHeight","width","height","hugeFramesSent","toFixed","keyFramesEncoded","packetsSent","totalKeyFramesEncoded","totalNackCount","nackCount","totalPliCount","pliCount","totalFirCount","firCount","encoderImplementation","qualityLimitationReason","qualityLimitationResolutionChanges","retransmittedPacketsSent","totalBytesSent","headerBytesSent","retransmittedBytesSent","receiveSlot","ssrc","idAndCsi","framesReceived","bytes","totalBytesReceived","currentPacketsLost","packetsLost","totalPacketsLost","packetsReceived","totalPacketsReceived","currentPacketLossRatio","lastPacketReceivedTimestamp","framesDecoded","keyFramesDecoded","decoderImplementation","fecPacketsDiscarded","fecPacketsReceived","headerBytesReceived","jitter","totalSamplesReceived","totalSamplesDecoded","concealedSamples","currentPacketLoss","totalPacketsLostOnReceiver","packetsLostOnReceiver","roundTripTime","reportsReceived","overAllPacketLossRatio","determineUplinkNetworkQuality","remoteRtpResults","statsAnalyzerCurrentStats","EventsScope"],"sources":["index.ts"],"sourcesContent":["/* eslint-disable prefer-destructuring */\n\nimport {cloneDeep, isEmpty} from 'lodash';\nimport {ConnectionState} from '@webex/internal-media-core';\n\nimport EventsScope from '../common/events/events-scope';\nimport {\n DEFAULT_GET_STATS_FILTER,\n STATS,\n MQA_INTERVAL,\n NETWORK_TYPE,\n MEDIA_DEVICES,\n _UNKNOWN_,\n} from '../constants';\nimport {\n emptyAudioReceive,\n emptyAudioTransmit,\n emptyMqaInterval,\n emptyVideoReceive,\n emptyVideoTransmit,\n} from '../mediaQualityMetrics/config';\nimport LoggerProxy from '../common/logs/logger-proxy';\n\nimport defaultStats from './global';\nimport {\n getAudioSenderMqa,\n getAudioReceiverMqa,\n getVideoSenderMqa,\n getVideoReceiverMqa,\n} from './mqaUtil';\nimport {ReceiveSlot} from '../multistream/receiveSlot';\n\nexport const EVENTS = {\n MEDIA_QUALITY: 'MEDIA_QUALITY',\n LOCAL_MEDIA_STARTED: 'LOCAL_MEDIA_STARTED',\n LOCAL_MEDIA_STOPPED: 'LOCAL_MEDIA_STOPPED',\n REMOTE_MEDIA_STARTED: 'REMOTE_MEDIA_STARTED',\n REMOTE_MEDIA_STOPPED: 'REMOTE_MEDIA_STOPPED',\n};\n\nconst emptySender = {\n trackLabel: '',\n maxPacketLossRatio: 0,\n availableBandwidth: 0,\n bytesSent: 0,\n meanRemoteJitter: [],\n meanRoundTripTime: [],\n};\n\nconst emptyReceiver = {\n availableBandwidth: 0,\n bytesReceived: 0,\n meanRtpJitter: [],\n meanRoundTripTime: [],\n};\n\ntype ReceiveSlotCallback = (csi: number) => ReceiveSlot | undefined;\n\n/**\n * Stats Analyzer class that will emit events based on detected quality\n *\n * @export\n * @class StatsAnalyzer\n * @extends {EventsScope}\n */\nexport class StatsAnalyzer extends EventsScope {\n config: any;\n correlationId: any;\n lastEmittedStartStopEvent: any;\n lastMqaDataSent: any;\n lastStatsResults: any;\n meetingMediaStatus: any;\n mqaInterval: NodeJS.Timeout;\n mqaSentCount: any;\n networkQualityMonitor: any;\n mediaConnection: any;\n statsInterval: NodeJS.Timeout;\n statsResults: any;\n statsStarted: any;\n successfulCandidatePair: any;\n localIpAddress: string; // Returns the local IP address for diagnostics. this is the local IP of the interface used for the current media connection a host can have many local Ip Addresses\n receiveSlotCallback: ReceiveSlotCallback;\n\n /**\n * Creates a new instance of StatsAnalyzer\n * @constructor\n * @public\n * @param {Object} config SDK Configuration Object\n * @param {Function} receiveSlotCallback Callback used to access receive slots.\n * @param {Object} networkQualityMonitor class for assessing network characteristics (jitter, packetLoss, latency)\n * @param {Object} statsResults Default properties for stats\n */\n constructor(\n config: any,\n receiveSlotCallback: ReceiveSlotCallback = () => undefined,\n networkQualityMonitor: object = {},\n statsResults: object = defaultStats\n ) {\n super();\n this.statsStarted = false;\n this.statsResults = statsResults;\n this.lastStatsResults = null;\n this.config = config;\n this.networkQualityMonitor = networkQualityMonitor;\n this.correlationId = config.correlationId;\n this.mqaSentCount = -1;\n this.lastMqaDataSent = {};\n this.lastEmittedStartStopEvent = {};\n this.receiveSlotCallback = receiveSlotCallback;\n this.successfulCandidatePair = {};\n this.localIpAddress = '';\n }\n\n /**\n * Resets cumulative stats arrays.\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n resetStatsResults() {\n Object.keys(this.statsResults).forEach((mediaType) => {\n if (mediaType.includes('recv')) {\n this.statsResults[mediaType].recv.meanRtpJitter = [];\n }\n\n if (mediaType.includes('send')) {\n this.statsResults[mediaType].send.meanRemoteJitter = [];\n this.statsResults[mediaType].send.meanRoundTripTime = [];\n }\n });\n }\n\n /**\n * sets mediaStatus status for analyzing metrics\n *\n * @public\n * @param {Object} status for the audio and video\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n public updateMediaStatus(status: object) {\n this.meetingMediaStatus = status;\n }\n\n /**\n * captures MQA data from media connection\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n sendMqaData() {\n const newMqa = cloneDeep(emptyMqaInterval);\n\n Object.keys(this.statsResults).forEach((mediaType) => {\n if (!this.lastMqaDataSent[mediaType]) {\n this.lastMqaDataSent[mediaType] = {};\n }\n\n if (!this.lastMqaDataSent[mediaType].send && mediaType.includes('-send')) {\n this.lastMqaDataSent[mediaType].send = {};\n }\n\n if (!this.lastMqaDataSent[mediaType].recv && mediaType.includes('-recv')) {\n this.lastMqaDataSent[mediaType].recv = {};\n }\n\n if (mediaType.includes('audio-send') || mediaType.includes('audio-share-send')) {\n const audioSender = cloneDeep(emptyAudioTransmit);\n\n getAudioSenderMqa({\n audioSender,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.audioTransmit.push(audioSender);\n\n this.lastMqaDataSent[mediaType].send = cloneDeep(this.statsResults[mediaType].send);\n } else if (mediaType.includes('audio-recv') || mediaType.includes('audio-share-recv')) {\n const audioReceiver = cloneDeep(emptyAudioReceive);\n\n getAudioReceiverMqa({\n audioReceiver,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.audioReceive.push(audioReceiver);\n\n this.lastMqaDataSent[mediaType].recv = cloneDeep(this.statsResults[mediaType].recv);\n } else if (mediaType.includes('video-send') || mediaType.includes('video-share-send')) {\n const videoSender = cloneDeep(emptyVideoTransmit);\n\n getVideoSenderMqa({\n videoSender,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.videoTransmit.push(videoSender);\n\n this.lastMqaDataSent[mediaType].send = cloneDeep(this.statsResults[mediaType].send);\n } else if (mediaType.includes('video-recv') || mediaType.includes('video-share-recv')) {\n const videoReceiver = cloneDeep(emptyVideoReceive);\n\n getVideoReceiverMqa({\n videoReceiver,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.videoReceive.push(videoReceiver);\n\n this.lastMqaDataSent[mediaType].recv = cloneDeep(this.statsResults[mediaType].recv);\n }\n });\n\n newMqa.intervalMetadata.peerReflexiveIP = this.statsResults.connectionType.local.ipAddress;\n\n // Adding peripheral information\n newMqa.intervalMetadata.peripherals.push({information: _UNKNOWN_, name: MEDIA_DEVICES.SPEAKER});\n if (this.statsResults['audio-send']) {\n newMqa.intervalMetadata.peripherals.push({\n information: this.statsResults['audio-send'].trackLabel || _UNKNOWN_,\n name: MEDIA_DEVICES.MICROPHONE,\n });\n }\n if (this.statsResults['video-send']) {\n newMqa.intervalMetadata.peripherals.push({\n information: this.statsResults['video-send'].trackLabel || _UNKNOWN_,\n name: MEDIA_DEVICES.CAMERA,\n });\n }\n\n newMqa.networkType = this.statsResults.connectionType.local.networkType;\n\n this.mqaSentCount += 1;\n\n newMqa.intervalNumber = this.mqaSentCount;\n\n this.resetStatsResults();\n\n this.emit(\n {\n file: 'statsAnalyzer',\n function: 'sendMqaData',\n },\n EVENTS.MEDIA_QUALITY,\n {\n data: newMqa,\n // @ts-ignore\n networkType: newMqa.networkType,\n }\n );\n }\n\n /**\n * updated the media connection when changed\n *\n * @private\n * @memberof StatsAnalyzer\n * @param {RoapMediaConnection} mediaConnection\n * @returns {void}\n */\n updateMediaConnection(mediaConnection: any) {\n this.mediaConnection = mediaConnection;\n }\n\n /**\n * Returns the local IP address for diagnostics.\n * this is the local IP of the interface used for the current media connection\n * a host can have many local Ip Addresses\n * @returns {string | undefined} The local IP address.\n */\n getLocalIpAddress(): string {\n return this.localIpAddress;\n }\n\n /**\n * Starts the stats analyzer on interval\n *\n * @public\n * @memberof StatsAnalyzer\n * @param {RoapMediaConnection} mediaConnection\n * @returns {Promise}\n */\n public startAnalyzer(mediaConnection: any) {\n if (!this.statsStarted) {\n this.statsStarted = true;\n this.mediaConnection = mediaConnection;\n\n return this.getStatsAndParse().then(() => {\n this.statsInterval = setInterval(() => {\n this.getStatsAndParse();\n }, this.config.analyzerInterval);\n // Trigger initial fetch\n this.sendMqaData();\n this.mqaInterval = setInterval(() => {\n this.sendMqaData();\n }, MQA_INTERVAL);\n });\n }\n\n return Promise.resolve();\n }\n\n /**\n * Cleans up the analyzer when done\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n public stopAnalyzer() {\n const sendOneLastMqa = this.mqaInterval && this.statsInterval;\n\n if (this.statsInterval) {\n clearInterval(this.statsInterval);\n this.statsInterval = undefined;\n }\n\n if (this.mqaInterval) {\n clearInterval(this.mqaInterval);\n this.mqaInterval = undefined;\n }\n\n if (sendOneLastMqa) {\n return this.getStatsAndParse().then(() => {\n this.sendMqaData();\n this.mediaConnection = null;\n });\n }\n\n return Promise.resolve();\n }\n\n /**\n * Parse a single result of get stats\n *\n * @private\n * @param {*} getStatsResult\n * @param {String} type\n * @param {boolean} isSender\n * @returns {void}\n * @memberof StatsAnalyzer\n */\n private parseGetStatsResult(getStatsResult: any, type: string, isSender: boolean) {\n if (!getStatsResult) {\n return;\n }\n\n // Generate empty stats results\n if (!this.statsResults[type]) {\n this.statsResults[type] = {};\n }\n\n if (isSender && !this.statsResults[type].send) {\n this.statsResults[type].send = cloneDeep(emptySender);\n } else if (!isSender && !this.statsResults[type].recv) {\n this.statsResults[type].recv = cloneDeep(emptyReceiver);\n }\n\n switch (getStatsResult.type) {\n case 'outbound-rtp':\n this.processOutboundRTPResult(getStatsResult, type);\n break;\n case 'inbound-rtp':\n this.processInboundRTPResult(getStatsResult, type);\n break;\n case 'remote-inbound-rtp':\n case 'remote-outbound-rtp':\n this.compareSentAndReceived(getStatsResult, type);\n break;\n case 'remotecandidate':\n case 'remote-candidate':\n this.parseCandidate(getStatsResult, type, isSender, true);\n break;\n case 'local-candidate':\n this.parseCandidate(getStatsResult, type, isSender, false);\n break;\n case 'media-source':\n // @ts-ignore\n this.parseAudioSource(getStatsResult, type);\n break;\n default:\n break;\n }\n }\n\n /**\n * Filters the get stats results for types\n * @private\n * @param {Array} statsItem\n * @param {String} type\n * @param {boolean} isSender\n * @returns {void}\n */\n filterAndParseGetStatsResults(statsItem: any, type: string, isSender: boolean) {\n const {types} = DEFAULT_GET_STATS_FILTER;\n\n // get the successful candidate pair before parsing stats.\n statsItem.report.forEach((report) => {\n if (report.type === 'candidate-pair' && report.state === 'succeeded') {\n this.successfulCandidatePair = report;\n }\n });\n\n statsItem.report.forEach((result) => {\n if (types.includes(result.type)) {\n this.parseGetStatsResult(result, type, isSender);\n }\n });\n\n if (this.statsResults[type]) {\n this.statsResults[type].direction = statsItem.currentDirection;\n this.statsResults[type].trackLabel = statsItem.localTrackLabel;\n this.statsResults[type].csi = statsItem.csi;\n this.extractAndSetLocalIpAddressInfoForDiagnostics(\n this.successfulCandidatePair?.localCandidateId,\n this.statsResults?.candidates\n );\n // reset the successful candidate pair.\n this.successfulCandidatePair = {};\n }\n }\n\n /**\n * parse the audio\n * @param {String} result\n * @param {boolean} type\n * @returns {void}\n */\n parseAudioSource(result: any, type: any) {\n if (!result) {\n return;\n }\n\n if (type.includes('audio-send')) {\n this.statsResults[type].send.audioLevel = result.audioLevel;\n this.statsResults[type].send.totalAudioEnergy = result.totalAudioEnergy;\n }\n }\n\n /**\n * emits started/stopped events for local/remote media by checking\n * if given values are increasing or not. The previousValue, currentValue\n * params can be any numerical value like number of receive packets or\n * decoded frames, etc.\n *\n * @private\n * @param {string} mediaType\n * @param {number} previousValue - value to compare\n * @param {number} currentValue - value to compare (must be same type of value as previousValue)\n * @param {boolean} isLocal - true if stats are for local media being sent out, false for remote media being received\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n emitStartStopEvents = (\n mediaType: string,\n previousValue: number,\n currentValue: number,\n isLocal: boolean\n ) => {\n if (mediaType !== 'audio' && mediaType !== 'video' && mediaType !== 'share') {\n throw new Error(`Unsupported mediaType: ${mediaType}`);\n }\n\n // eslint-disable-next-line no-param-reassign\n if (previousValue === undefined) previousValue = 0;\n // eslint-disable-next-line no-param-reassign\n if (currentValue === undefined) currentValue = 0;\n\n if (!this.lastEmittedStartStopEvent[mediaType]) {\n this.lastEmittedStartStopEvent[mediaType] = {};\n }\n\n const lastEmittedEvent = isLocal\n ? this.lastEmittedStartStopEvent[mediaType].local\n : this.lastEmittedStartStopEvent[mediaType].remote;\n\n let newEvent;\n\n if (currentValue - previousValue > 0) {\n newEvent = isLocal ? EVENTS.LOCAL_MEDIA_STARTED : EVENTS.REMOTE_MEDIA_STARTED;\n } else if (currentValue === previousValue && currentValue > 0) {\n newEvent = isLocal ? EVENTS.LOCAL_MEDIA_STOPPED : EVENTS.REMOTE_MEDIA_STOPPED;\n }\n\n if (newEvent && lastEmittedEvent !== newEvent) {\n if (isLocal) {\n this.lastEmittedStartStopEvent[mediaType].local = newEvent;\n } else {\n this.lastEmittedStartStopEvent[mediaType].remote = newEvent;\n }\n this.emit(\n {\n file: 'statsAnalyzer/index',\n function: 'compareLastStatsResult',\n },\n newEvent,\n {\n type: mediaType,\n }\n );\n }\n };\n\n /**\n * compares current and previous stats to check if packets are not sent\n *\n * @private\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n private compareLastStatsResult() {\n if (this.lastStatsResults !== null && this.meetingMediaStatus) {\n const getCurrentStatsTotals = (keyPrefix: string, value: string): number =>\n Object.keys(this.statsResults)\n .filter((key) => key.startsWith(keyPrefix))\n .reduce((prev, cur) => prev + (this.statsResults[cur]?.recv[value] || 0), 0);\n\n const getPreviousStatsTotals = (keyPrefix: string, value: string): number =>\n Object.keys(this.statsResults)\n .filter((key) => key.startsWith(keyPrefix))\n .reduce((prev, cur) => prev + (this.lastStatsResults[cur]?.recv[value] || 0), 0);\n\n if (this.meetingMediaStatus.expected.sendAudio && this.lastStatsResults['audio-send']) {\n // compare audio stats sent\n // NOTE: relies on there being only one sender.\n const currentStats = this.statsResults['audio-send'].send;\n const previousStats = this.lastStatsResults['audio-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.totalAudioEnergy === previousStats.totalAudioEnergy ||\n currentStats.totalAudioEnergy === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio Energy present`,\n currentStats.totalAudioEnergy\n );\n }\n\n if (currentStats.audioLevel === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> audio level is 0 for the user`\n );\n }\n }\n\n this.emitStartStopEvents(\n 'audio',\n previousStats.totalPacketsSent,\n currentStats.totalPacketsSent,\n true\n );\n }\n\n if (this.meetingMediaStatus.expected.receiveAudio) {\n // compare audio stats received\n const currentPacketsReceived = getCurrentStatsTotals('audio-recv', 'totalPacketsReceived');\n const previousPacketsReceived = getPreviousStatsTotals(\n 'audio-recv',\n 'totalPacketsReceived'\n );\n const currentSamplesReceived = getCurrentStatsTotals('audio-recv', 'totalSamplesReceived');\n const previousSamplesReceived = getPreviousStatsTotals(\n 'audio-recv',\n 'totalSamplesReceived'\n );\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio RTP packets received`,\n currentPacketsReceived\n );\n } else if (\n currentSamplesReceived === previousSamplesReceived ||\n currentSamplesReceived === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio samples received`,\n currentSamplesReceived\n );\n }\n\n this.emitStartStopEvents('audio', previousPacketsReceived, currentPacketsReceived, false);\n }\n\n if (this.meetingMediaStatus.expected.sendVideo && this.lastStatsResults['video-send']) {\n // compare video stats sent\n const currentStats = this.statsResults['video-send'].send;\n const previousStats = this.lastStatsResults['video-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.framesEncoded === previousStats.framesEncoded ||\n currentStats.framesEncoded === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video Frames Encoded`,\n currentStats.framesEncoded\n );\n }\n\n if (\n this.statsResults['video-send'].send.framesSent ===\n this.lastStatsResults['video-send'].send.framesSent ||\n this.statsResults['video-send'].send.framesSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video Frames sent`,\n this.statsResults['video-send'].send.framesSent\n );\n }\n }\n\n this.emitStartStopEvents('video', previousStats.framesSent, currentStats.framesSent, true);\n }\n\n if (this.meetingMediaStatus.expected.receiveVideo) {\n // compare video stats received\n const currentPacketsReceived = getCurrentStatsTotals('video-recv', 'totalPacketsReceived');\n const previousPacketsReceived = getPreviousStatsTotals(\n 'video-recv',\n 'totalPacketsReceived'\n );\n const currentFramesReceived = getCurrentStatsTotals('video-recv', 'framesReceived');\n const previousFramesReceived = getPreviousStatsTotals('video-recv', 'framesReceived');\n const currentFramesDecoded = getCurrentStatsTotals('video-recv', 'framesDecoded');\n const previousFramesDecoded = getPreviousStatsTotals('video-recv', 'framesDecoded');\n const currentFramesDropped = getCurrentStatsTotals('video-recv', 'framesDropped');\n const previousFramesDropped = getPreviousStatsTotals('video-recv', 'framesDropped');\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video RTP packets received`,\n currentPacketsReceived\n );\n } else {\n if (currentFramesReceived === previousFramesReceived || currentFramesReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video frames received`,\n currentFramesReceived\n );\n }\n\n if (currentFramesDecoded === previousFramesDecoded || currentFramesDecoded === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video frames decoded`,\n currentFramesDecoded\n );\n }\n\n if (currentFramesDropped - previousFramesDropped > 10) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> video frames are getting dropped`,\n currentFramesDropped - previousFramesDropped\n );\n }\n }\n\n this.emitStartStopEvents('video', previousFramesDecoded, currentFramesDecoded, false);\n }\n\n if (this.meetingMediaStatus.expected.sendShare && this.lastStatsResults['video-share-send']) {\n // compare share stats sent\n\n const currentStats = this.statsResults['video-share-send'].send;\n const previousStats = this.lastStatsResults['video-share-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.framesEncoded === previousStats.framesEncoded ||\n currentStats.framesEncoded === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames getting encoded`,\n currentStats.framesEncoded\n );\n }\n\n if (\n this.statsResults['video-share-send'].send.framesSent ===\n this.lastStatsResults['video-share-send'].send.framesSent ||\n this.statsResults['video-share-send'].send.framesSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames sent`,\n this.statsResults['video-share-send'].send.framesSent\n );\n }\n }\n }\n\n if (this.meetingMediaStatus.expected.sendShare) {\n // TODO:need to check receive share value\n // compare share stats received\n const currentPacketsReceived = getCurrentStatsTotals(\n 'video-share-recv',\n 'totalPacketsReceived'\n );\n const previousPacketsReceived = getPreviousStatsTotals(\n 'video-share-recv',\n 'totalPacketsReceived'\n );\n const currentFramesReceived = getCurrentStatsTotals('video-share-recv', 'framesReceived');\n const previousFramesReceived = getPreviousStatsTotals('video-share-recv', 'framesReceived');\n const currentFramesDecoded = getCurrentStatsTotals('video-share-recv', 'framesDecoded');\n const previousFramesDecoded = getPreviousStatsTotals('video-share-recv', 'framesDecoded');\n const currentFramesDropped = getCurrentStatsTotals('video-share-recv', 'framesDropped');\n const previousFramesDropped = getPreviousStatsTotals('video-share-recv', 'framesDropped');\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share RTP packets received`,\n currentPacketsReceived\n );\n } else {\n if (currentFramesReceived === previousFramesReceived || currentFramesReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames received`,\n currentFramesReceived\n );\n }\n\n if (currentFramesDecoded === previousFramesDecoded || currentFramesDecoded === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames decoded`,\n currentFramesDecoded\n );\n }\n\n if (currentFramesDropped - previousFramesDropped > 10) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> share frames are getting dropped`,\n currentFramesDropped - previousFramesDropped\n );\n }\n }\n\n // we are not calling emitStartStopEvents() for sending or receiving share because sharing is often started and stopped\n // in meetings and this.meetingMediaStatus.expected values can be out of sync with the actual packet flow\n // so we would send \"sharing stopped\" events incorrectly\n }\n }\n }\n\n /**\n * Does a `getStats` on all the transceivers and parses the results\n *\n * @private\n * @memberof StatsAnalyzer\n * @returns {Promise}\n */\n private getStatsAndParse() {\n if (!this.mediaConnection) {\n return Promise.resolve();\n }\n\n if (\n this.mediaConnection &&\n this.mediaConnection.getConnectionState() === ConnectionState.Failed\n ) {\n LoggerProxy.logger.trace(\n 'StatsAnalyzer:index#getStatsAndParse --> media connection is in failed state'\n );\n\n return Promise.resolve();\n }\n\n LoggerProxy.logger.trace('StatsAnalyzer:index#getStatsAndParse --> Collecting Stats');\n\n return this.mediaConnection.getTransceiverStats().then((transceiverStats) => {\n transceiverStats.video.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `video-recv-${i}`, false)\n );\n transceiverStats.audio.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `audio-recv-${i}`, false)\n );\n transceiverStats.screenShareVideo.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `video-share-recv-${i}`, false)\n );\n transceiverStats.screenShareAudio.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `audio-share-recv-${i}`, false)\n );\n\n transceiverStats.video.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'video-send', true);\n });\n transceiverStats.audio.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'audio-send', true);\n });\n transceiverStats.screenShareVideo.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'video-share-send', true);\n });\n transceiverStats.screenShareAudio.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'audio-share-send', true);\n });\n\n this.compareLastStatsResult();\n\n // Save the last results to compare with the current\n // DO Deep copy, for some reason it takes the reference all the time rather then old value set\n this.lastStatsResults = JSON.parse(JSON.stringify(this.statsResults));\n\n LoggerProxy.logger.trace(\n 'StatsAnalyzer:index#getStatsAndParse --> Finished Collecting Stats'\n );\n });\n }\n\n /**\n * Processes OutboundRTP stats result and stores\n * @private\n * @param {*} result\n * @param {*} mediaType\n * @returns {void}\n */\n private processOutboundRTPResult(result: any, mediaType: any) {\n const sendrecvType = STATS.SEND_DIRECTION;\n\n if (result.bytesSent) {\n const kilobytes = 0;\n\n if (result.frameWidth && result.frameHeight) {\n this.statsResults[mediaType][sendrecvType].width = result.frameWidth;\n this.statsResults[mediaType][sendrecvType].height = result.frameHeight;\n this.statsResults[mediaType][sendrecvType].framesSent = result.framesSent;\n this.statsResults[mediaType][sendrecvType].hugeFramesSent = result.hugeFramesSent;\n }\n\n this.statsResults[mediaType][sendrecvType].availableBandwidth = kilobytes.toFixed(1);\n\n this.statsResults[mediaType][sendrecvType].framesEncoded = result.framesEncoded;\n this.statsResults[mediaType][sendrecvType].keyFramesEncoded = result.keyFramesEncoded;\n this.statsResults[mediaType][sendrecvType].packetsSent = result.packetsSent;\n\n // Data saved to send MQA metrics\n\n this.statsResults[mediaType][sendrecvType].totalKeyFramesEncoded = result.keyFramesEncoded;\n this.statsResults[mediaType][sendrecvType].totalNackCount = result.nackCount;\n this.statsResults[mediaType][sendrecvType].totalPliCount = result.pliCount;\n this.statsResults[mediaType][sendrecvType].totalPacketsSent = result.packetsSent;\n this.statsResults[mediaType][sendrecvType].totalFirCount = result.firCount;\n this.statsResults[mediaType][sendrecvType].framesSent = result.framesSent;\n this.statsResults[mediaType][sendrecvType].framesEncoded = result.framesEncoded;\n this.statsResults[mediaType][sendrecvType].encoderImplementation =\n result.encoderImplementation;\n this.statsResults[mediaType][sendrecvType].qualityLimitationReason =\n result.qualityLimitationReason;\n this.statsResults[mediaType][sendrecvType].qualityLimitationResolutionChanges =\n result.qualityLimitationResolutionChanges;\n this.statsResults[mediaType][sendrecvType].retransmittedPacketsSent =\n result.retransmittedPacketsSent;\n this.statsResults[mediaType][sendrecvType].totalBytesSent = result.bytesSent;\n this.statsResults[mediaType][sendrecvType].headerBytesSent = result.headerBytesSent;\n this.statsResults[mediaType][sendrecvType].retransmittedBytesSent =\n result.retransmittedBytesSent;\n }\n }\n\n /**\n * Processes InboundRTP stats result and stores\n * @private\n * @param {*} result\n * @param {*} mediaType\n * @returns {void}\n */\n private processInboundRTPResult(result: any, mediaType: any) {\n const sendrecvType = STATS.RECEIVE_DIRECTION;\n\n if (result.bytesReceived) {\n let kilobytes = 0;\n const receiveSlot = this.receiveSlotCallback(result.ssrc);\n const idAndCsi = receiveSlot\n ? `id: \"${receiveSlot.id || ''}\"${receiveSlot.csi ? ` and csi: ${receiveSlot.csi}` : ''}`\n : '';\n\n if (result.frameWidth && result.frameHeight) {\n this.statsResults[mediaType][sendrecvType].width = result.frameWidth;\n this.statsResults[mediaType][sendrecvType].height = result.frameHeight;\n this.statsResults[mediaType][sendrecvType].framesReceived = result.framesReceived;\n }\n\n const bytes =\n result.bytesReceived - this.statsResults[mediaType][sendrecvType].totalBytesReceived;\n\n kilobytes = bytes / 1024;\n this.statsResults[mediaType][sendrecvType].availableBandwidth = kilobytes.toFixed(1);\n\n let currentPacketsLost =\n result.packetsLost - this.statsResults[mediaType][sendrecvType].totalPacketsLost;\n if (currentPacketsLost < 0) {\n currentPacketsLost = 0;\n }\n\n const currentPacketsReceived =\n result.packetsReceived - this.statsResults[mediaType][sendrecvType].totalPacketsReceived;\n this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;\n\n if (currentPacketsReceived === 0) {\n if (receiveSlot) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ${idAndCsi}`,\n currentPacketsReceived\n );\n }\n }\n\n // Check the over all packet Lost ratio\n this.statsResults[mediaType][sendrecvType].currentPacketLossRatio =\n currentPacketsLost > 0\n ? currentPacketsLost / (currentPacketsReceived + currentPacketsLost)\n : 0;\n if (this.statsResults[mediaType][sendrecvType].currentPacketLossRatio > 3) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#processInboundRTPResult --> Packets getting lost from the receiver with slot ${idAndCsi}`,\n this.statsResults[mediaType][sendrecvType].currentPacketLossRatio\n );\n }\n\n // TODO: check the packet loss value is negative values here\n\n if (result.packetsLost) {\n this.statsResults[mediaType][sendrecvType].totalPacketsLost =\n result.packetsLost > 0 ? result.packetsLost : -result.packetsLost;\n } else {\n this.statsResults[mediaType][sendrecvType].totalPacketsLost = 0;\n }\n\n this.statsResults[mediaType][sendrecvType].lastPacketReceivedTimestamp =\n result.lastPacketReceivedTimestamp;\n\n // From Thin\n this.statsResults[mediaType][sendrecvType].totalNackCount = result.nackCount;\n this.statsResults[mediaType][sendrecvType].totalPliCount = result.pliCount;\n this.statsResults[mediaType][sendrecvType].framesDecoded = result.framesDecoded;\n this.statsResults[mediaType][sendrecvType].keyFramesDecoded = result.keyFramesDecoded;\n\n this.statsResults[mediaType][sendrecvType].decoderImplementation =\n result.decoderImplementation;\n this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;\n\n this.statsResults[mediaType][sendrecvType].fecPacketsDiscarded = result.fecPacketsDiscarded;\n this.statsResults[mediaType][sendrecvType].fecPacketsReceived = result.fecPacketsReceived;\n this.statsResults[mediaType][sendrecvType].totalBytesReceived = result.bytesReceived;\n this.statsResults[mediaType][sendrecvType].headerBytesReceived = result.headerBytesReceived;\n\n this.statsResults[mediaType][sendrecvType].meanRtpJitter.push(result.jitter);\n\n // Audio stats\n\n this.statsResults[mediaType][sendrecvType].audioLevel = result.audioLevel;\n this.statsResults[mediaType][sendrecvType].totalAudioEnergy = result.totalAudioEnergy;\n this.statsResults[mediaType][sendrecvType].totalSamplesReceived =\n result.totalSamplesReceived || 0;\n this.statsResults[mediaType][sendrecvType].totalSamplesDecoded =\n result.totalSamplesDecoded || 0;\n this.statsResults[mediaType][sendrecvType].concealedSamples = result.concealedSamples || 0;\n }\n }\n\n /**\n * extracts the local Ip address from the statsResult object by looking at stats results candidates\n * and matches that ID with the successful candidate pair. It looks at the type of local candidate it is\n * and then extracts the IP address from the relatedAddress or address property based on conditions known in webrtc\n * note, there are known incompatibilities and it is possible for this to set undefined, or for the IP address to be the public IP address\n * for example, firefox does not set the relayProtocol, and if the user is behind a NAT it might be the public IP\n * @private\n * @param {string} successfulCandidatePairId - The ID of the successful candidate pair.\n * @param {Object} candidates - the stats result candidates\n * @returns {void}\n */\n extractAndSetLocalIpAddressInfoForDiagnostics = (\n successfulCandidatePairId: string,\n candidates: {[key: string]: Record<string, unknown>}\n ) => {\n let newIpAddress = '';\n if (successfulCandidatePairId && !isEmpty(candidates)) {\n const localCandidate = candidates[successfulCandidatePairId];\n if (localCandidate) {\n if (localCandidate.candidateType === 'host') {\n // if it's a host candidate, use the address property - it will be the local IP\n newIpAddress = `${localCandidate.address}`;\n } else if (localCandidate.candidateType === 'prflx') {\n // if it's a peer reflexive candidate and we're not using a relay (there is no relayProtocol set)\n // then look at the relatedAddress - it will be the local\n //\n // Firefox doesn't populate the relayProtocol property\n if (!localCandidate.relayProtocol) {\n newIpAddress = `${localCandidate.relatedAddress}`;\n } else {\n // if it's a peer reflexive candidate and we are using a relay -\n // in that case the relatedAddress will be the IP of the TURN server (Linus),\n // so we can only look at the address, but it might be local IP or public IP,\n // depending on if the user is behind a NAT or not\n newIpAddress = `${localCandidate.address}`;\n }\n }\n }\n }\n this.localIpAddress = newIpAddress;\n };\n\n /**\n * Processes remote and local candidate result and stores\n * @private\n * @param {*} result\n * @param {*} type\n * @param {boolean} isSender\n * @param {boolean} isRemote\n *\n * @returns {void}\n */\n parseCandidate = (result: any, type: any, isSender: boolean, isRemote: boolean) => {\n if (!result || !result.id) {\n return;\n }\n\n // We only care about the successful local candidate\n if (this.successfulCandidatePair?.localCandidateId !== result.id) {\n return;\n }\n\n let transport;\n if (result.relayProtocol) {\n transport = result.relayProtocol.toUpperCase();\n } else if (result.protocol) {\n transport = result.protocol.toUpperCase();\n }\n\n const sendRecvType = isSender ? STATS.SEND_DIRECTION : STATS.RECEIVE_DIRECTION;\n const ipType = isRemote ? STATS.REMOTE : STATS.LOCAL;\n\n if (!this.statsResults.candidates) {\n this.statsResults.candidates = {};\n }\n\n this.statsResults.candidates[result.id] = {\n candidateType: result.candidateType,\n ipAddress: result.ip, // TODO: add ports\n relatedAddress: result.relatedAddress,\n relatedPort: result.relatedPort,\n relayProtocol: result.relayProtocol,\n protocol: result.protocol,\n address: result.address,\n portNumber: result.port,\n networkType: result.networkType,\n priority: result.priority,\n transport,\n timestamp: result.time,\n id: result.id,\n type: result.type,\n };\n\n this.statsResults.connectionType[ipType].candidateType = result.candidateType;\n this.statsResults.connectionType[ipType].ipAddress = result.ipAddress;\n\n this.statsResults.connectionType[ipType].networkType =\n result.networkType === NETWORK_TYPE.VPN ? NETWORK_TYPE.UNKNOWN : result.networkType;\n this.statsResults.connectionType[ipType].transport = transport;\n\n this.statsResults[type][sendRecvType].totalRoundTripTime = result.totalRoundTripTime;\n };\n\n /**\n *\n * @private\n * @param {*} result\n * @param {*} type\n * @returns {void}\n * @memberof StatsAnalyzer\n */\n compareSentAndReceived(result, type) {\n // Don't compare on transceivers without a sender.\n if (!type || !this.statsResults[type].send) {\n return;\n }\n\n const mediaType = type;\n\n const currentPacketLoss =\n result.packetsLost - this.statsResults[mediaType].send.totalPacketsLostOnReceiver;\n\n this.statsResults[mediaType].send.packetsLostOnReceiver = currentPacketLoss;\n this.statsResults[mediaType].send.totalPacketsLostOnReceiver = result.packetsLost;\n\n this.statsResults[mediaType].send.meanRemoteJitter.push(result.jitter);\n this.statsResults[mediaType].send.meanRoundTripTime.push(result.roundTripTime);\n\n this.statsResults[mediaType].send.timestamp = result.timestamp;\n this.statsResults[mediaType].send.ssrc = result.ssrc;\n this.statsResults[mediaType].send.reportsReceived = result.reportsReceived;\n\n // Total packloss ratio on this video section of the call\n this.statsResults[mediaType].send.overAllPacketLossRatio =\n this.statsResults[mediaType].send.totalPacketsLostOnReceiver > 0\n ? this.statsResults[mediaType].send.totalPacketsLostOnReceiver /\n this.statsResults[mediaType].send.totalPacketsSent\n : 0;\n this.statsResults[mediaType].send.currentPacketLossRatio =\n this.statsResults[mediaType].send.packetsLostOnReceiver > 0\n ? (this.statsResults[mediaType].send.packetsLostOnReceiver * 100) /\n (this.statsResults[mediaType].send.packetsSent +\n this.statsResults[mediaType].send.packetsLostOnReceiver)\n : 0;\n\n if (\n this.statsResults[mediaType].send.maxPacketLossRatio <\n this.statsResults[mediaType].send.currentPacketLossRatio\n ) {\n this.statsResults[mediaType].send.maxPacketLossRatio =\n this.statsResults[mediaType].send.currentPacketLossRatio;\n }\n\n if (result.type === 'remote-inbound-rtp') {\n this.networkQualityMonitor.determineUplinkNetworkQuality({\n mediaType,\n remoteRtpResults: result,\n statsAnalyzerCurrentStats: this.statsResults,\n });\n }\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"names":["EVENTS","MEDIA_QUALITY","LOCAL_MEDIA_STARTED","LOCAL_MEDIA_STOPPED","REMOTE_MEDIA_STARTED","REMOTE_MEDIA_STOPPED","emptySender","trackLabel","maxPacketLossRatio","availableBandwidth","bytesSent","meanRemoteJitter","meanRoundTripTime","emptyReceiver","bytesReceived","meanRtpJitter","StatsAnalyzer","config","receiveSlotCallback","undefined","networkQualityMonitor","statsResults","defaultStats","mediaType","previousValue","currentValue","isLocal","Error","lastEmittedStartStopEvent","lastEmittedEvent","local","remote","newEvent","emit","file","function","type","successfulCandidatePairId","candidates","newIpAddress","localCandidate","candidateType","address","relayProtocol","relatedAddress","localIpAddress","result","isSender","isRemote","id","successfulCandidatePair","localCandidateId","transport","toUpperCase","protocol","sendRecvType","STATS","SEND_DIRECTION","RECEIVE_DIRECTION","ipType","REMOTE","LOCAL","ipAddress","ip","relatedPort","portNumber","port","networkType","priority","timestamp","time","connectionType","NETWORK_TYPE","VPN","UNKNOWN","totalRoundTripTime","statsStarted","lastStatsResults","correlationId","mqaSentCount","lastMqaDataSent","forEach","includes","recv","send","status","meetingMediaStatus","newMqa","emptyMqaInterval","audioSender","emptyAudioTransmit","getAudioSenderMqa","audioTransmit","push","audioReceiver","emptyAudioReceive","getAudioReceiverMqa","audioReceive","videoSender","emptyVideoTransmit","getVideoSenderMqa","videoTransmit","videoReceiver","emptyVideoReceive","getVideoReceiverMqa","videoReceive","intervalMetadata","peerReflexiveIP","peripherals","information","_UNKNOWN_","name","MEDIA_DEVICES","SPEAKER","MICROPHONE","CAMERA","intervalNumber","resetStatsResults","data","mediaConnection","getStatsAndParse","then","statsInterval","setInterval","analyzerInterval","sendMqaData","mqaInterval","MQA_INTERVAL","resolve","sendOneLastMqa","clearInterval","getStatsResult","processOutboundRTPResult","processInboundRTPResult","compareSentAndReceived","parseCandidate","parseAudioSource","statsItem","types","DEFAULT_GET_STATS_FILTER","report","state","parseGetStatsResult","direction","currentDirection","localTrackLabel","csi","extractAndSetLocalIpAddressInfoForDiagnostics","audioLevel","totalAudioEnergy","getCurrentStatsTotals","keyPrefix","value","filter","key","startsWith","reduce","prev","cur","getPreviousStatsTotals","expected","sendAudio","currentStats","previousStats","totalPacketsSent","LoggerProxy","logger","info","emitStartStopEvents","receiveAudio","currentPacketsReceived","previousPacketsReceived","currentSamplesReceived","previousSamplesReceived","sendVideo","framesEncoded","framesSent","receiveVideo","currentFramesReceived","previousFramesReceived","currentFramesDecoded","previousFramesDecoded","currentFramesDropped","previousFramesDropped","sendShare","getConnectionState","ConnectionState","Failed","trace","getTransceiverStats","transceiverStats","video","receivers","receiver","i","filterAndParseGetStatsResults","audio","screenShareVideo","screenShareAudio","senders","sender","compareLastStatsResult","JSON","parse","sendrecvType","kilobytes","frameWidth","frameHeight","width","height","hugeFramesSent","toFixed","keyFramesEncoded","packetsSent","totalKeyFramesEncoded","totalNackCount","nackCount","totalPliCount","pliCount","totalFirCount","firCount","encoderImplementation","qualityLimitationReason","qualityLimitationResolutionChanges","retransmittedPacketsSent","totalBytesSent","headerBytesSent","retransmittedBytesSent","receiveSlot","ssrc","sourceState","idAndCsi","framesReceived","bytes","totalBytesReceived","currentPacketsLost","packetsLost","totalPacketsLost","packetsReceived","totalPacketsReceived","currentPacketLossRatio","lastPacketReceivedTimestamp","framesDecoded","keyFramesDecoded","decoderImplementation","fecPacketsDiscarded","fecPacketsReceived","headerBytesReceived","jitter","totalSamplesReceived","totalSamplesDecoded","concealedSamples","currentPacketLoss","totalPacketsLostOnReceiver","packetsLostOnReceiver","roundTripTime","reportsReceived","overAllPacketLossRatio","determineUplinkNetworkQuality","remoteRtpResults","statsAnalyzerCurrentStats","EventsScope"],"sources":["index.ts"],"sourcesContent":["/* eslint-disable prefer-destructuring */\n\nimport {cloneDeep, isEmpty} from 'lodash';\nimport {ConnectionState} from '@webex/internal-media-core';\n\nimport EventsScope from '../common/events/events-scope';\nimport {\n DEFAULT_GET_STATS_FILTER,\n STATS,\n MQA_INTERVAL,\n NETWORK_TYPE,\n MEDIA_DEVICES,\n _UNKNOWN_,\n} from '../constants';\nimport {\n emptyAudioReceive,\n emptyAudioTransmit,\n emptyMqaInterval,\n emptyVideoReceive,\n emptyVideoTransmit,\n} from '../mediaQualityMetrics/config';\nimport LoggerProxy from '../common/logs/logger-proxy';\n\nimport defaultStats from './global';\nimport {\n getAudioSenderMqa,\n getAudioReceiverMqa,\n getVideoSenderMqa,\n getVideoReceiverMqa,\n} from './mqaUtil';\nimport {ReceiveSlot} from '../multistream/receiveSlot';\n\nexport const EVENTS = {\n MEDIA_QUALITY: 'MEDIA_QUALITY',\n LOCAL_MEDIA_STARTED: 'LOCAL_MEDIA_STARTED',\n LOCAL_MEDIA_STOPPED: 'LOCAL_MEDIA_STOPPED',\n REMOTE_MEDIA_STARTED: 'REMOTE_MEDIA_STARTED',\n REMOTE_MEDIA_STOPPED: 'REMOTE_MEDIA_STOPPED',\n};\n\nconst emptySender = {\n trackLabel: '',\n maxPacketLossRatio: 0,\n availableBandwidth: 0,\n bytesSent: 0,\n meanRemoteJitter: [],\n meanRoundTripTime: [],\n};\n\nconst emptyReceiver = {\n availableBandwidth: 0,\n bytesReceived: 0,\n meanRtpJitter: [],\n meanRoundTripTime: [],\n};\n\ntype ReceiveSlotCallback = (csi: number) => ReceiveSlot | undefined;\n\n/**\n * Stats Analyzer class that will emit events based on detected quality\n *\n * @export\n * @class StatsAnalyzer\n * @extends {EventsScope}\n */\nexport class StatsAnalyzer extends EventsScope {\n config: any;\n correlationId: any;\n lastEmittedStartStopEvent: any;\n lastMqaDataSent: any;\n lastStatsResults: any;\n meetingMediaStatus: any;\n mqaInterval: NodeJS.Timeout;\n mqaSentCount: any;\n networkQualityMonitor: any;\n mediaConnection: any;\n statsInterval: NodeJS.Timeout;\n statsResults: any;\n statsStarted: any;\n successfulCandidatePair: any;\n localIpAddress: string; // Returns the local IP address for diagnostics. this is the local IP of the interface used for the current media connection a host can have many local Ip Addresses\n receiveSlotCallback: ReceiveSlotCallback;\n\n /**\n * Creates a new instance of StatsAnalyzer\n * @constructor\n * @public\n * @param {Object} config SDK Configuration Object\n * @param {Function} receiveSlotCallback Callback used to access receive slots.\n * @param {Object} networkQualityMonitor class for assessing network characteristics (jitter, packetLoss, latency)\n * @param {Object} statsResults Default properties for stats\n */\n constructor(\n config: any,\n receiveSlotCallback: ReceiveSlotCallback = () => undefined,\n networkQualityMonitor: object = {},\n statsResults: object = defaultStats\n ) {\n super();\n this.statsStarted = false;\n this.statsResults = statsResults;\n this.lastStatsResults = null;\n this.config = config;\n this.networkQualityMonitor = networkQualityMonitor;\n this.correlationId = config.correlationId;\n this.mqaSentCount = -1;\n this.lastMqaDataSent = {};\n this.lastEmittedStartStopEvent = {};\n this.receiveSlotCallback = receiveSlotCallback;\n this.successfulCandidatePair = {};\n this.localIpAddress = '';\n }\n\n /**\n * Resets cumulative stats arrays.\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n resetStatsResults() {\n Object.keys(this.statsResults).forEach((mediaType) => {\n if (mediaType.includes('recv')) {\n this.statsResults[mediaType].recv.meanRtpJitter = [];\n }\n\n if (mediaType.includes('send')) {\n this.statsResults[mediaType].send.meanRemoteJitter = [];\n this.statsResults[mediaType].send.meanRoundTripTime = [];\n }\n });\n }\n\n /**\n * sets mediaStatus status for analyzing metrics\n *\n * @public\n * @param {Object} status for the audio and video\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n public updateMediaStatus(status: object) {\n this.meetingMediaStatus = status;\n }\n\n /**\n * captures MQA data from media connection\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n sendMqaData() {\n const newMqa = cloneDeep(emptyMqaInterval);\n\n Object.keys(this.statsResults).forEach((mediaType) => {\n if (!this.lastMqaDataSent[mediaType]) {\n this.lastMqaDataSent[mediaType] = {};\n }\n\n if (!this.lastMqaDataSent[mediaType].send && mediaType.includes('-send')) {\n this.lastMqaDataSent[mediaType].send = {};\n }\n\n if (!this.lastMqaDataSent[mediaType].recv && mediaType.includes('-recv')) {\n this.lastMqaDataSent[mediaType].recv = {};\n }\n\n if (mediaType.includes('audio-send') || mediaType.includes('audio-share-send')) {\n const audioSender = cloneDeep(emptyAudioTransmit);\n\n getAudioSenderMqa({\n audioSender,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.audioTransmit.push(audioSender);\n\n this.lastMqaDataSent[mediaType].send = cloneDeep(this.statsResults[mediaType].send);\n } else if (mediaType.includes('audio-recv') || mediaType.includes('audio-share-recv')) {\n const audioReceiver = cloneDeep(emptyAudioReceive);\n\n getAudioReceiverMqa({\n audioReceiver,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.audioReceive.push(audioReceiver);\n\n this.lastMqaDataSent[mediaType].recv = cloneDeep(this.statsResults[mediaType].recv);\n } else if (mediaType.includes('video-send') || mediaType.includes('video-share-send')) {\n const videoSender = cloneDeep(emptyVideoTransmit);\n\n getVideoSenderMqa({\n videoSender,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.videoTransmit.push(videoSender);\n\n this.lastMqaDataSent[mediaType].send = cloneDeep(this.statsResults[mediaType].send);\n } else if (mediaType.includes('video-recv') || mediaType.includes('video-share-recv')) {\n const videoReceiver = cloneDeep(emptyVideoReceive);\n\n getVideoReceiverMqa({\n videoReceiver,\n statsResults: this.statsResults,\n lastMqaDataSent: this.lastMqaDataSent,\n mediaType,\n });\n newMqa.videoReceive.push(videoReceiver);\n\n this.lastMqaDataSent[mediaType].recv = cloneDeep(this.statsResults[mediaType].recv);\n }\n });\n\n newMqa.intervalMetadata.peerReflexiveIP = this.statsResults.connectionType.local.ipAddress;\n\n // Adding peripheral information\n newMqa.intervalMetadata.peripherals.push({information: _UNKNOWN_, name: MEDIA_DEVICES.SPEAKER});\n if (this.statsResults['audio-send']) {\n newMqa.intervalMetadata.peripherals.push({\n information: this.statsResults['audio-send'].trackLabel || _UNKNOWN_,\n name: MEDIA_DEVICES.MICROPHONE,\n });\n }\n if (this.statsResults['video-send']) {\n newMqa.intervalMetadata.peripherals.push({\n information: this.statsResults['video-send'].trackLabel || _UNKNOWN_,\n name: MEDIA_DEVICES.CAMERA,\n });\n }\n\n newMqa.networkType = this.statsResults.connectionType.local.networkType;\n\n this.mqaSentCount += 1;\n\n newMqa.intervalNumber = this.mqaSentCount;\n\n this.resetStatsResults();\n\n this.emit(\n {\n file: 'statsAnalyzer',\n function: 'sendMqaData',\n },\n EVENTS.MEDIA_QUALITY,\n {\n data: newMqa,\n // @ts-ignore\n networkType: newMqa.networkType,\n }\n );\n }\n\n /**\n * updated the media connection when changed\n *\n * @private\n * @memberof StatsAnalyzer\n * @param {RoapMediaConnection} mediaConnection\n * @returns {void}\n */\n updateMediaConnection(mediaConnection: any) {\n this.mediaConnection = mediaConnection;\n }\n\n /**\n * Returns the local IP address for diagnostics.\n * this is the local IP of the interface used for the current media connection\n * a host can have many local Ip Addresses\n * @returns {string | undefined} The local IP address.\n */\n getLocalIpAddress(): string {\n return this.localIpAddress;\n }\n\n /**\n * Starts the stats analyzer on interval\n *\n * @public\n * @memberof StatsAnalyzer\n * @param {RoapMediaConnection} mediaConnection\n * @returns {Promise}\n */\n public startAnalyzer(mediaConnection: any) {\n if (!this.statsStarted) {\n this.statsStarted = true;\n this.mediaConnection = mediaConnection;\n\n return this.getStatsAndParse().then(() => {\n this.statsInterval = setInterval(() => {\n this.getStatsAndParse();\n }, this.config.analyzerInterval);\n // Trigger initial fetch\n this.sendMqaData();\n this.mqaInterval = setInterval(() => {\n this.sendMqaData();\n }, MQA_INTERVAL);\n });\n }\n\n return Promise.resolve();\n }\n\n /**\n * Cleans up the analyzer when done\n *\n * @public\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n public stopAnalyzer() {\n const sendOneLastMqa = this.mqaInterval && this.statsInterval;\n\n if (this.statsInterval) {\n clearInterval(this.statsInterval);\n this.statsInterval = undefined;\n }\n\n if (this.mqaInterval) {\n clearInterval(this.mqaInterval);\n this.mqaInterval = undefined;\n }\n\n if (sendOneLastMqa) {\n return this.getStatsAndParse().then(() => {\n this.sendMqaData();\n this.mediaConnection = null;\n });\n }\n\n return Promise.resolve();\n }\n\n /**\n * Parse a single result of get stats\n *\n * @private\n * @param {*} getStatsResult\n * @param {String} type\n * @param {boolean} isSender\n * @returns {void}\n * @memberof StatsAnalyzer\n */\n private parseGetStatsResult(getStatsResult: any, type: string, isSender: boolean) {\n if (!getStatsResult) {\n return;\n }\n\n // Generate empty stats results\n if (!this.statsResults[type]) {\n this.statsResults[type] = {};\n }\n\n if (isSender && !this.statsResults[type].send) {\n this.statsResults[type].send = cloneDeep(emptySender);\n } else if (!isSender && !this.statsResults[type].recv) {\n this.statsResults[type].recv = cloneDeep(emptyReceiver);\n }\n\n switch (getStatsResult.type) {\n case 'outbound-rtp':\n this.processOutboundRTPResult(getStatsResult, type);\n break;\n case 'inbound-rtp':\n this.processInboundRTPResult(getStatsResult, type);\n break;\n case 'remote-inbound-rtp':\n case 'remote-outbound-rtp':\n this.compareSentAndReceived(getStatsResult, type);\n break;\n case 'remotecandidate':\n case 'remote-candidate':\n this.parseCandidate(getStatsResult, type, isSender, true);\n break;\n case 'local-candidate':\n this.parseCandidate(getStatsResult, type, isSender, false);\n break;\n case 'media-source':\n // @ts-ignore\n this.parseAudioSource(getStatsResult, type);\n break;\n default:\n break;\n }\n }\n\n /**\n * Filters the get stats results for types\n * @private\n * @param {Array} statsItem\n * @param {String} type\n * @param {boolean} isSender\n * @returns {void}\n */\n filterAndParseGetStatsResults(statsItem: any, type: string, isSender: boolean) {\n const {types} = DEFAULT_GET_STATS_FILTER;\n\n // get the successful candidate pair before parsing stats.\n statsItem.report.forEach((report) => {\n if (report.type === 'candidate-pair' && report.state === 'succeeded') {\n this.successfulCandidatePair = report;\n }\n });\n\n statsItem.report.forEach((result) => {\n if (types.includes(result.type)) {\n this.parseGetStatsResult(result, type, isSender);\n }\n });\n\n if (this.statsResults[type]) {\n this.statsResults[type].direction = statsItem.currentDirection;\n this.statsResults[type].trackLabel = statsItem.localTrackLabel;\n this.statsResults[type].csi = statsItem.csi;\n this.extractAndSetLocalIpAddressInfoForDiagnostics(\n this.successfulCandidatePair?.localCandidateId,\n this.statsResults?.candidates\n );\n // reset the successful candidate pair.\n this.successfulCandidatePair = {};\n }\n }\n\n /**\n * parse the audio\n * @param {String} result\n * @param {boolean} type\n * @returns {void}\n */\n parseAudioSource(result: any, type: any) {\n if (!result) {\n return;\n }\n\n if (type.includes('audio-send')) {\n this.statsResults[type].send.audioLevel = result.audioLevel;\n this.statsResults[type].send.totalAudioEnergy = result.totalAudioEnergy;\n }\n }\n\n /**\n * emits started/stopped events for local/remote media by checking\n * if given values are increasing or not. The previousValue, currentValue\n * params can be any numerical value like number of receive packets or\n * decoded frames, etc.\n *\n * @private\n * @param {string} mediaType\n * @param {number} previousValue - value to compare\n * @param {number} currentValue - value to compare (must be same type of value as previousValue)\n * @param {boolean} isLocal - true if stats are for local media being sent out, false for remote media being received\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n emitStartStopEvents = (\n mediaType: string,\n previousValue: number,\n currentValue: number,\n isLocal: boolean\n ) => {\n if (mediaType !== 'audio' && mediaType !== 'video' && mediaType !== 'share') {\n throw new Error(`Unsupported mediaType: ${mediaType}`);\n }\n\n // eslint-disable-next-line no-param-reassign\n if (previousValue === undefined) previousValue = 0;\n // eslint-disable-next-line no-param-reassign\n if (currentValue === undefined) currentValue = 0;\n\n if (!this.lastEmittedStartStopEvent[mediaType]) {\n this.lastEmittedStartStopEvent[mediaType] = {};\n }\n\n const lastEmittedEvent = isLocal\n ? this.lastEmittedStartStopEvent[mediaType].local\n : this.lastEmittedStartStopEvent[mediaType].remote;\n\n let newEvent;\n\n if (currentValue - previousValue > 0) {\n newEvent = isLocal ? EVENTS.LOCAL_MEDIA_STARTED : EVENTS.REMOTE_MEDIA_STARTED;\n } else if (currentValue === previousValue && currentValue > 0) {\n newEvent = isLocal ? EVENTS.LOCAL_MEDIA_STOPPED : EVENTS.REMOTE_MEDIA_STOPPED;\n }\n\n if (newEvent && lastEmittedEvent !== newEvent) {\n if (isLocal) {\n this.lastEmittedStartStopEvent[mediaType].local = newEvent;\n } else {\n this.lastEmittedStartStopEvent[mediaType].remote = newEvent;\n }\n this.emit(\n {\n file: 'statsAnalyzer/index',\n function: 'compareLastStatsResult',\n },\n newEvent,\n {\n type: mediaType,\n }\n );\n }\n };\n\n /**\n * compares current and previous stats to check if packets are not sent\n *\n * @private\n * @memberof StatsAnalyzer\n * @returns {void}\n */\n private compareLastStatsResult() {\n if (this.lastStatsResults !== null && this.meetingMediaStatus) {\n const getCurrentStatsTotals = (keyPrefix: string, value: string): number =>\n Object.keys(this.statsResults)\n .filter((key) => key.startsWith(keyPrefix))\n .reduce((prev, cur) => prev + (this.statsResults[cur]?.recv[value] || 0), 0);\n\n const getPreviousStatsTotals = (keyPrefix: string, value: string): number =>\n Object.keys(this.statsResults)\n .filter((key) => key.startsWith(keyPrefix))\n .reduce((prev, cur) => prev + (this.lastStatsResults[cur]?.recv[value] || 0), 0);\n\n if (this.meetingMediaStatus.expected.sendAudio && this.lastStatsResults['audio-send']) {\n // compare audio stats sent\n // NOTE: relies on there being only one sender.\n const currentStats = this.statsResults['audio-send'].send;\n const previousStats = this.lastStatsResults['audio-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.totalAudioEnergy === previousStats.totalAudioEnergy ||\n currentStats.totalAudioEnergy === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio Energy present`,\n currentStats.totalAudioEnergy\n );\n }\n\n if (currentStats.audioLevel === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> audio level is 0 for the user`\n );\n }\n }\n\n this.emitStartStopEvents(\n 'audio',\n previousStats.totalPacketsSent,\n currentStats.totalPacketsSent,\n true\n );\n }\n\n if (this.meetingMediaStatus.expected.receiveAudio) {\n // compare audio stats received\n const currentPacketsReceived = getCurrentStatsTotals('audio-recv', 'totalPacketsReceived');\n const previousPacketsReceived = getPreviousStatsTotals(\n 'audio-recv',\n 'totalPacketsReceived'\n );\n const currentSamplesReceived = getCurrentStatsTotals('audio-recv', 'totalSamplesReceived');\n const previousSamplesReceived = getPreviousStatsTotals(\n 'audio-recv',\n 'totalSamplesReceived'\n );\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio RTP packets received`,\n currentPacketsReceived\n );\n } else if (\n currentSamplesReceived === previousSamplesReceived ||\n currentSamplesReceived === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No audio samples received`,\n currentSamplesReceived\n );\n }\n\n this.emitStartStopEvents('audio', previousPacketsReceived, currentPacketsReceived, false);\n }\n\n if (this.meetingMediaStatus.expected.sendVideo && this.lastStatsResults['video-send']) {\n // compare video stats sent\n const currentStats = this.statsResults['video-send'].send;\n const previousStats = this.lastStatsResults['video-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.framesEncoded === previousStats.framesEncoded ||\n currentStats.framesEncoded === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video Frames Encoded`,\n currentStats.framesEncoded\n );\n }\n\n if (\n this.statsResults['video-send'].send.framesSent ===\n this.lastStatsResults['video-send'].send.framesSent ||\n this.statsResults['video-send'].send.framesSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video Frames sent`,\n this.statsResults['video-send'].send.framesSent\n );\n }\n }\n\n this.emitStartStopEvents('video', previousStats.framesSent, currentStats.framesSent, true);\n }\n\n if (this.meetingMediaStatus.expected.receiveVideo) {\n // compare video stats received\n const currentPacketsReceived = getCurrentStatsTotals('video-recv', 'totalPacketsReceived');\n const previousPacketsReceived = getPreviousStatsTotals(\n 'video-recv',\n 'totalPacketsReceived'\n );\n const currentFramesReceived = getCurrentStatsTotals('video-recv', 'framesReceived');\n const previousFramesReceived = getPreviousStatsTotals('video-recv', 'framesReceived');\n const currentFramesDecoded = getCurrentStatsTotals('video-recv', 'framesDecoded');\n const previousFramesDecoded = getPreviousStatsTotals('video-recv', 'framesDecoded');\n const currentFramesDropped = getCurrentStatsTotals('video-recv', 'framesDropped');\n const previousFramesDropped = getPreviousStatsTotals('video-recv', 'framesDropped');\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video RTP packets received`,\n currentPacketsReceived\n );\n } else {\n if (currentFramesReceived === previousFramesReceived || currentFramesReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video frames received`,\n currentFramesReceived\n );\n }\n\n if (currentFramesDecoded === previousFramesDecoded || currentFramesDecoded === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No video frames decoded`,\n currentFramesDecoded\n );\n }\n\n if (currentFramesDropped - previousFramesDropped > 10) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> video frames are getting dropped`,\n currentFramesDropped - previousFramesDropped\n );\n }\n }\n\n this.emitStartStopEvents('video', previousFramesDecoded, currentFramesDecoded, false);\n }\n\n if (this.meetingMediaStatus.expected.sendShare && this.lastStatsResults['video-share-send']) {\n // compare share stats sent\n\n const currentStats = this.statsResults['video-share-send'].send;\n const previousStats = this.lastStatsResults['video-share-send'].send;\n\n if (\n currentStats.totalPacketsSent === previousStats.totalPacketsSent ||\n currentStats.totalPacketsSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share RTP packets sent`,\n currentStats.totalPacketsSent\n );\n } else {\n if (\n currentStats.framesEncoded === previousStats.framesEncoded ||\n currentStats.framesEncoded === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames getting encoded`,\n currentStats.framesEncoded\n );\n }\n\n if (\n this.statsResults['video-share-send'].send.framesSent ===\n this.lastStatsResults['video-share-send'].send.framesSent ||\n this.statsResults['video-share-send'].send.framesSent === 0\n ) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames sent`,\n this.statsResults['video-share-send'].send.framesSent\n );\n }\n }\n }\n\n if (this.meetingMediaStatus.expected.sendShare) {\n // TODO:need to check receive share value\n // compare share stats received\n const currentPacketsReceived = getCurrentStatsTotals(\n 'video-share-recv',\n 'totalPacketsReceived'\n );\n const previousPacketsReceived = getPreviousStatsTotals(\n 'video-share-recv',\n 'totalPacketsReceived'\n );\n const currentFramesReceived = getCurrentStatsTotals('video-share-recv', 'framesReceived');\n const previousFramesReceived = getPreviousStatsTotals('video-share-recv', 'framesReceived');\n const currentFramesDecoded = getCurrentStatsTotals('video-share-recv', 'framesDecoded');\n const previousFramesDecoded = getPreviousStatsTotals('video-share-recv', 'framesDecoded');\n const currentFramesDropped = getCurrentStatsTotals('video-share-recv', 'framesDropped');\n const previousFramesDropped = getPreviousStatsTotals('video-share-recv', 'framesDropped');\n\n if (currentPacketsReceived === previousPacketsReceived || currentPacketsReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share RTP packets received`,\n currentPacketsReceived\n );\n } else {\n if (currentFramesReceived === previousFramesReceived || currentFramesReceived === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames received`,\n currentFramesReceived\n );\n }\n\n if (currentFramesDecoded === previousFramesDecoded || currentFramesDecoded === 0) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> No share frames decoded`,\n currentFramesDecoded\n );\n }\n\n if (currentFramesDropped - previousFramesDropped > 10) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#compareLastStatsResult --> share frames are getting dropped`,\n currentFramesDropped - previousFramesDropped\n );\n }\n }\n\n // we are not calling emitStartStopEvents() for sending or receiving share because sharing is often started and stopped\n // in meetings and this.meetingMediaStatus.expected values can be out of sync with the actual packet flow\n // so we would send \"sharing stopped\" events incorrectly\n }\n }\n }\n\n /**\n * Does a `getStats` on all the transceivers and parses the results\n *\n * @private\n * @memberof StatsAnalyzer\n * @returns {Promise}\n */\n private getStatsAndParse() {\n if (!this.mediaConnection) {\n return Promise.resolve();\n }\n\n if (\n this.mediaConnection &&\n this.mediaConnection.getConnectionState() === ConnectionState.Failed\n ) {\n LoggerProxy.logger.trace(\n 'StatsAnalyzer:index#getStatsAndParse --> media connection is in failed state'\n );\n\n return Promise.resolve();\n }\n\n LoggerProxy.logger.trace('StatsAnalyzer:index#getStatsAndParse --> Collecting Stats');\n\n return this.mediaConnection.getTransceiverStats().then((transceiverStats) => {\n transceiverStats.video.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `video-recv-${i}`, false)\n );\n transceiverStats.audio.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `audio-recv-${i}`, false)\n );\n transceiverStats.screenShareVideo.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `video-share-recv-${i}`, false)\n );\n transceiverStats.screenShareAudio.receivers.forEach((receiver, i) =>\n this.filterAndParseGetStatsResults(receiver, `audio-share-recv-${i}`, false)\n );\n\n transceiverStats.video.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'video-send', true);\n });\n transceiverStats.audio.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'audio-send', true);\n });\n transceiverStats.screenShareVideo.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'video-share-send', true);\n });\n transceiverStats.screenShareAudio.senders.forEach((sender, i) => {\n if (i > 0) {\n throw new Error('Stats Analyzer does not support multiple senders.');\n }\n this.filterAndParseGetStatsResults(sender, 'audio-share-send', true);\n });\n\n this.compareLastStatsResult();\n\n // Save the last results to compare with the current\n // DO Deep copy, for some reason it takes the reference all the time rather then old value set\n this.lastStatsResults = JSON.parse(JSON.stringify(this.statsResults));\n\n LoggerProxy.logger.trace(\n 'StatsAnalyzer:index#getStatsAndParse --> Finished Collecting Stats'\n );\n });\n }\n\n /**\n * Processes OutboundRTP stats result and stores\n * @private\n * @param {*} result\n * @param {*} mediaType\n * @returns {void}\n */\n private processOutboundRTPResult(result: any, mediaType: any) {\n const sendrecvType = STATS.SEND_DIRECTION;\n\n if (result.bytesSent) {\n const kilobytes = 0;\n\n if (result.frameWidth && result.frameHeight) {\n this.statsResults[mediaType][sendrecvType].width = result.frameWidth;\n this.statsResults[mediaType][sendrecvType].height = result.frameHeight;\n this.statsResults[mediaType][sendrecvType].framesSent = result.framesSent;\n this.statsResults[mediaType][sendrecvType].hugeFramesSent = result.hugeFramesSent;\n }\n\n this.statsResults[mediaType][sendrecvType].availableBandwidth = kilobytes.toFixed(1);\n\n this.statsResults[mediaType][sendrecvType].framesEncoded = result.framesEncoded;\n this.statsResults[mediaType][sendrecvType].keyFramesEncoded = result.keyFramesEncoded;\n this.statsResults[mediaType][sendrecvType].packetsSent = result.packetsSent;\n\n // Data saved to send MQA metrics\n\n this.statsResults[mediaType][sendrecvType].totalKeyFramesEncoded = result.keyFramesEncoded;\n this.statsResults[mediaType][sendrecvType].totalNackCount = result.nackCount;\n this.statsResults[mediaType][sendrecvType].totalPliCount = result.pliCount;\n this.statsResults[mediaType][sendrecvType].totalPacketsSent = result.packetsSent;\n this.statsResults[mediaType][sendrecvType].totalFirCount = result.firCount;\n this.statsResults[mediaType][sendrecvType].framesSent = result.framesSent;\n this.statsResults[mediaType][sendrecvType].framesEncoded = result.framesEncoded;\n this.statsResults[mediaType][sendrecvType].encoderImplementation =\n result.encoderImplementation;\n this.statsResults[mediaType][sendrecvType].qualityLimitationReason =\n result.qualityLimitationReason;\n this.statsResults[mediaType][sendrecvType].qualityLimitationResolutionChanges =\n result.qualityLimitationResolutionChanges;\n this.statsResults[mediaType][sendrecvType].retransmittedPacketsSent =\n result.retransmittedPacketsSent;\n this.statsResults[mediaType][sendrecvType].totalBytesSent = result.bytesSent;\n this.statsResults[mediaType][sendrecvType].headerBytesSent = result.headerBytesSent;\n this.statsResults[mediaType][sendrecvType].retransmittedBytesSent =\n result.retransmittedBytesSent;\n }\n }\n\n /**\n * Processes InboundRTP stats result and stores\n * @private\n * @param {*} result\n * @param {*} mediaType\n * @returns {void}\n */\n private processInboundRTPResult(result: any, mediaType: any) {\n const sendrecvType = STATS.RECEIVE_DIRECTION;\n\n if (result.bytesReceived) {\n let kilobytes = 0;\n const receiveSlot = this.receiveSlotCallback(result.ssrc);\n const sourceState = receiveSlot?.sourceState;\n const idAndCsi = receiveSlot\n ? `id: \"${receiveSlot.id || ''}\"${receiveSlot.csi ? ` and csi: ${receiveSlot.csi}` : ''}`\n : '';\n\n if (result.frameWidth && result.frameHeight) {\n this.statsResults[mediaType][sendrecvType].width = result.frameWidth;\n this.statsResults[mediaType][sendrecvType].height = result.frameHeight;\n this.statsResults[mediaType][sendrecvType].framesReceived = result.framesReceived;\n }\n\n const bytes =\n result.bytesReceived - this.statsResults[mediaType][sendrecvType].totalBytesReceived;\n\n kilobytes = bytes / 1024;\n this.statsResults[mediaType][sendrecvType].availableBandwidth = kilobytes.toFixed(1);\n\n let currentPacketsLost =\n result.packetsLost - this.statsResults[mediaType][sendrecvType].totalPacketsLost;\n if (currentPacketsLost < 0) {\n currentPacketsLost = 0;\n }\n\n const currentPacketsReceived =\n result.packetsReceived - this.statsResults[mediaType][sendrecvType].totalPacketsReceived;\n this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;\n\n if (currentPacketsReceived === 0) {\n if (receiveSlot && sourceState === 'live') {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ${idAndCsi}. Total packets received on slot: `,\n result.packetsReceived\n );\n }\n }\n\n // Check the over all packet Lost ratio\n this.statsResults[mediaType][sendrecvType].currentPacketLossRatio =\n currentPacketsLost > 0\n ? currentPacketsLost / (currentPacketsReceived + currentPacketsLost)\n : 0;\n if (this.statsResults[mediaType][sendrecvType].currentPacketLossRatio > 3) {\n LoggerProxy.logger.info(\n `StatsAnalyzer:index#processInboundRTPResult --> Packets getting lost from the receiver with slot ${idAndCsi}`,\n this.statsResults[mediaType][sendrecvType].currentPacketLossRatio\n );\n }\n\n // TODO: check the packet loss value is negative values here\n\n if (result.packetsLost) {\n this.statsResults[mediaType][sendrecvType].totalPacketsLost =\n result.packetsLost > 0 ? result.packetsLost : -result.packetsLost;\n } else {\n this.statsResults[mediaType][sendrecvType].totalPacketsLost = 0;\n }\n\n this.statsResults[mediaType][sendrecvType].lastPacketReceivedTimestamp =\n result.lastPacketReceivedTimestamp;\n\n // From Thin\n this.statsResults[mediaType][sendrecvType].totalNackCount = result.nackCount;\n this.statsResults[mediaType][sendrecvType].totalPliCount = result.pliCount;\n this.statsResults[mediaType][sendrecvType].framesDecoded = result.framesDecoded;\n this.statsResults[mediaType][sendrecvType].keyFramesDecoded = result.keyFramesDecoded;\n\n this.statsResults[mediaType][sendrecvType].decoderImplementation =\n result.decoderImplementation;\n this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;\n\n this.statsResults[mediaType][sendrecvType].fecPacketsDiscarded = result.fecPacketsDiscarded;\n this.statsResults[mediaType][sendrecvType].fecPacketsReceived = result.fecPacketsReceived;\n this.statsResults[mediaType][sendrecvType].totalBytesReceived = result.bytesReceived;\n this.statsResults[mediaType][sendrecvType].headerBytesReceived = result.headerBytesReceived;\n\n this.statsResults[mediaType][sendrecvType].meanRtpJitter.push(result.jitter);\n\n // Audio stats\n\n this.statsResults[mediaType][sendrecvType].audioLevel = result.audioLevel;\n this.statsResults[mediaType][sendrecvType].totalAudioEnergy = result.totalAudioEnergy;\n this.statsResults[mediaType][sendrecvType].totalSamplesReceived =\n result.totalSamplesReceived || 0;\n this.statsResults[mediaType][sendrecvType].totalSamplesDecoded =\n result.totalSamplesDecoded || 0;\n this.statsResults[mediaType][sendrecvType].concealedSamples = result.concealedSamples || 0;\n }\n }\n\n /**\n * extracts the local Ip address from the statsResult object by looking at stats results candidates\n * and matches that ID with the successful candidate pair. It looks at the type of local candidate it is\n * and then extracts the IP address from the relatedAddress or address property based on conditions known in webrtc\n * note, there are known incompatibilities and it is possible for this to set undefined, or for the IP address to be the public IP address\n * for example, firefox does not set the relayProtocol, and if the user is behind a NAT it might be the public IP\n * @private\n * @param {string} successfulCandidatePairId - The ID of the successful candidate pair.\n * @param {Object} candidates - the stats result candidates\n * @returns {void}\n */\n extractAndSetLocalIpAddressInfoForDiagnostics = (\n successfulCandidatePairId: string,\n candidates: {[key: string]: Record<string, unknown>}\n ) => {\n let newIpAddress = '';\n if (successfulCandidatePairId && !isEmpty(candidates)) {\n const localCandidate = candidates[successfulCandidatePairId];\n if (localCandidate) {\n if (localCandidate.candidateType === 'host') {\n // if it's a host candidate, use the address property - it will be the local IP\n newIpAddress = `${localCandidate.address}`;\n } else if (localCandidate.candidateType === 'prflx') {\n // if it's a peer reflexive candidate and we're not using a relay (there is no relayProtocol set)\n // then look at the relatedAddress - it will be the local\n //\n // Firefox doesn't populate the relayProtocol property\n if (!localCandidate.relayProtocol) {\n newIpAddress = `${localCandidate.relatedAddress}`;\n } else {\n // if it's a peer reflexive candidate and we are using a relay -\n // in that case the relatedAddress will be the IP of the TURN server (Linus),\n // so we can only look at the address, but it might be local IP or public IP,\n // depending on if the user is behind a NAT or not\n newIpAddress = `${localCandidate.address}`;\n }\n }\n }\n }\n this.localIpAddress = newIpAddress;\n };\n\n /**\n * Processes remote and local candidate result and stores\n * @private\n * @param {*} result\n * @param {*} type\n * @param {boolean} isSender\n * @param {boolean} isRemote\n *\n * @returns {void}\n */\n parseCandidate = (result: any, type: any, isSender: boolean, isRemote: boolean) => {\n if (!result || !result.id) {\n return;\n }\n\n // We only care about the successful local candidate\n if (this.successfulCandidatePair?.localCandidateId !== result.id) {\n return;\n }\n\n let transport;\n if (result.relayProtocol) {\n transport = result.relayProtocol.toUpperCase();\n } else if (result.protocol) {\n transport = result.protocol.toUpperCase();\n }\n\n const sendRecvType = isSender ? STATS.SEND_DIRECTION : STATS.RECEIVE_DIRECTION;\n const ipType = isRemote ? STATS.REMOTE : STATS.LOCAL;\n\n if (!this.statsResults.candidates) {\n this.statsResults.candidates = {};\n }\n\n this.statsResults.candidates[result.id] = {\n candidateType: result.candidateType,\n ipAddress: result.ip, // TODO: add ports\n relatedAddress: result.relatedAddress,\n relatedPort: result.relatedPort,\n relayProtocol: result.relayProtocol,\n protocol: result.protocol,\n address: result.address,\n portNumber: result.port,\n networkType: result.networkType,\n priority: result.priority,\n transport,\n timestamp: result.time,\n id: result.id,\n type: result.type,\n };\n\n this.statsResults.connectionType[ipType].candidateType = result.candidateType;\n this.statsResults.connectionType[ipType].ipAddress = result.ipAddress;\n\n this.statsResults.connectionType[ipType].networkType =\n result.networkType === NETWORK_TYPE.VPN ? NETWORK_TYPE.UNKNOWN : result.networkType;\n this.statsResults.connectionType[ipType].transport = transport;\n\n this.statsResults[type][sendRecvType].totalRoundTripTime = result.totalRoundTripTime;\n };\n\n /**\n *\n * @private\n * @param {*} result\n * @param {*} type\n * @returns {void}\n * @memberof StatsAnalyzer\n */\n compareSentAndReceived(result, type) {\n // Don't compare on transceivers without a sender.\n if (!type || !this.statsResults[type].send) {\n return;\n }\n\n const mediaType = type;\n\n const currentPacketLoss =\n result.packetsLost - this.statsResults[mediaType].send.totalPacketsLostOnReceiver;\n\n this.statsResults[mediaType].send.packetsLostOnReceiver = currentPacketLoss;\n this.statsResults[mediaType].send.totalPacketsLostOnReceiver = result.packetsLost;\n\n this.statsResults[mediaType].send.meanRemoteJitter.push(result.jitter);\n this.statsResults[mediaType].send.meanRoundTripTime.push(result.roundTripTime);\n\n this.statsResults[mediaType].send.timestamp = result.timestamp;\n this.statsResults[mediaType].send.ssrc = result.ssrc;\n this.statsResults[mediaType].send.reportsReceived = result.reportsReceived;\n\n // Total packloss ratio on this video section of the call\n this.statsResults[mediaType].send.overAllPacketLossRatio =\n this.statsResults[mediaType].send.totalPacketsLostOnReceiver > 0\n ? this.statsResults[mediaType].send.totalPacketsLostOnReceiver /\n this.statsResults[mediaType].send.totalPacketsSent\n : 0;\n this.statsResults[mediaType].send.currentPacketLossRatio =\n this.statsResults[mediaType].send.packetsLostOnReceiver > 0\n ? (this.statsResults[mediaType].send.packetsLostOnReceiver * 100) /\n (this.statsResults[mediaType].send.packetsSent +\n this.statsResults[mediaType].send.packetsLostOnReceiver)\n : 0;\n\n if (\n this.statsResults[mediaType].send.maxPacketLossRatio <\n this.statsResults[mediaType].send.currentPacketLossRatio\n ) {\n this.statsResults[mediaType].send.maxPacketLossRatio =\n this.statsResults[mediaType].send.currentPacketLossRatio;\n }\n\n if (result.type === 'remote-inbound-rtp') {\n this.networkQualityMonitor.determineUplinkNetworkQuality({\n mediaType,\n remoteRtpResults: result,\n statsAnalyzerCurrentStats: this.statsResults,\n });\n }\n }\n}\n"],"mappings":"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"}
|
|
@@ -10,6 +10,7 @@ export default class RtcMetrics {
|
|
|
10
10
|
webex: any;
|
|
11
11
|
meetingId: string;
|
|
12
12
|
correlationId: string;
|
|
13
|
+
connectionId: string;
|
|
13
14
|
/**
|
|
14
15
|
* Initialize the interval.
|
|
15
16
|
*
|
|
@@ -45,6 +46,12 @@ export default class RtcMetrics {
|
|
|
45
46
|
* @returns {string}
|
|
46
47
|
*/
|
|
47
48
|
anonymizeIp(stats: string): string;
|
|
49
|
+
/**
|
|
50
|
+
* Set a new connection id.
|
|
51
|
+
*
|
|
52
|
+
* @returns {void}
|
|
53
|
+
*/
|
|
54
|
+
private setNewConnectionId;
|
|
48
55
|
/**
|
|
49
56
|
* Send metrics to the metrics service.
|
|
50
57
|
*
|
package/dist/webinar/index.js
CHANGED
|
@@ -62,7 +62,7 @@ var Webinar = _webexCore.WebexPlugin.extend({
|
|
|
62
62
|
updateCanManageWebcast: function updateCanManageWebcast(canManageWebcast) {
|
|
63
63
|
this.set('canManageWebcast', canManageWebcast);
|
|
64
64
|
},
|
|
65
|
-
version: "3.0.0-beta.
|
|
65
|
+
version: "3.0.0-beta.366"
|
|
66
66
|
});
|
|
67
67
|
var _default = Webinar;
|
|
68
68
|
exports.default = _default;
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@webex/plugin-meetings",
|
|
3
|
-
"version": "3.0.0-beta.
|
|
3
|
+
"version": "3.0.0-beta.366",
|
|
4
4
|
"description": "",
|
|
5
5
|
"license": "Cisco EULA (https://www.cisco.com/c/en/us/products/end-user-license-agreement.html)",
|
|
6
6
|
"contributors": [
|
|
@@ -33,12 +33,12 @@
|
|
|
33
33
|
},
|
|
34
34
|
"devDependencies": {
|
|
35
35
|
"@peculiar/webcrypto": "^1.4.3",
|
|
36
|
-
"@webex/plugin-meetings": "3.0.0-beta.
|
|
37
|
-
"@webex/test-helper-chai": "3.0.0-beta.
|
|
38
|
-
"@webex/test-helper-mocha": "3.0.0-beta.
|
|
39
|
-
"@webex/test-helper-mock-webex": "3.0.0-beta.
|
|
40
|
-
"@webex/test-helper-retry": "3.0.0-beta.
|
|
41
|
-
"@webex/test-helper-test-users": "3.0.0-beta.
|
|
36
|
+
"@webex/plugin-meetings": "3.0.0-beta.366",
|
|
37
|
+
"@webex/test-helper-chai": "3.0.0-beta.366",
|
|
38
|
+
"@webex/test-helper-mocha": "3.0.0-beta.366",
|
|
39
|
+
"@webex/test-helper-mock-webex": "3.0.0-beta.366",
|
|
40
|
+
"@webex/test-helper-retry": "3.0.0-beta.366",
|
|
41
|
+
"@webex/test-helper-test-users": "3.0.0-beta.366",
|
|
42
42
|
"chai": "^4.3.4",
|
|
43
43
|
"chai-as-promised": "^7.1.1",
|
|
44
44
|
"jsdom-global": "3.0.2",
|
|
@@ -47,19 +47,19 @@
|
|
|
47
47
|
"typescript": "^4.7.4"
|
|
48
48
|
},
|
|
49
49
|
"dependencies": {
|
|
50
|
-
"@webex/common": "3.0.0-beta.
|
|
50
|
+
"@webex/common": "3.0.0-beta.366",
|
|
51
51
|
"@webex/internal-media-core": "2.2.7",
|
|
52
|
-
"@webex/internal-plugin-conversation": "3.0.0-beta.
|
|
53
|
-
"@webex/internal-plugin-device": "3.0.0-beta.
|
|
54
|
-
"@webex/internal-plugin-llm": "3.0.0-beta.
|
|
55
|
-
"@webex/internal-plugin-mercury": "3.0.0-beta.
|
|
56
|
-
"@webex/internal-plugin-metrics": "3.0.0-beta.
|
|
57
|
-
"@webex/internal-plugin-support": "3.0.0-beta.
|
|
58
|
-
"@webex/internal-plugin-user": "3.0.0-beta.
|
|
59
|
-
"@webex/media-helpers": "3.0.0-beta.
|
|
60
|
-
"@webex/plugin-people": "3.0.0-beta.
|
|
61
|
-
"@webex/plugin-rooms": "3.0.0-beta.
|
|
62
|
-
"@webex/webex-core": "3.0.0-beta.
|
|
52
|
+
"@webex/internal-plugin-conversation": "3.0.0-beta.366",
|
|
53
|
+
"@webex/internal-plugin-device": "3.0.0-beta.366",
|
|
54
|
+
"@webex/internal-plugin-llm": "3.0.0-beta.366",
|
|
55
|
+
"@webex/internal-plugin-mercury": "3.0.0-beta.366",
|
|
56
|
+
"@webex/internal-plugin-metrics": "3.0.0-beta.366",
|
|
57
|
+
"@webex/internal-plugin-support": "3.0.0-beta.366",
|
|
58
|
+
"@webex/internal-plugin-user": "3.0.0-beta.366",
|
|
59
|
+
"@webex/media-helpers": "3.0.0-beta.366",
|
|
60
|
+
"@webex/plugin-people": "3.0.0-beta.366",
|
|
61
|
+
"@webex/plugin-rooms": "3.0.0-beta.366",
|
|
62
|
+
"@webex/webex-core": "3.0.0-beta.366",
|
|
63
63
|
"ampersand-collection": "^2.0.2",
|
|
64
64
|
"bowser": "^2.11.0",
|
|
65
65
|
"btoa": "^1.2.1",
|
package/src/rtcMetrics/index.ts
CHANGED
|
@@ -1,7 +1,20 @@
|
|
|
1
1
|
/* eslint-disable class-methods-use-this */
|
|
2
2
|
import {CallDiagnosticUtils} from '@webex/internal-plugin-metrics';
|
|
3
|
+
import uuid from 'uuid';
|
|
3
4
|
import RTC_METRICS from './constants';
|
|
4
5
|
|
|
6
|
+
const parseJsonPayload = (payload: any[]): any | null => {
|
|
7
|
+
try {
|
|
8
|
+
if (payload && payload[0]) {
|
|
9
|
+
return JSON.parse(payload[0]);
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
return null;
|
|
13
|
+
} catch (_) {
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
|
|
5
18
|
/**
|
|
6
19
|
* Rtc Metrics
|
|
7
20
|
*/
|
|
@@ -19,6 +32,8 @@ export default class RtcMetrics {
|
|
|
19
32
|
|
|
20
33
|
correlationId: string;
|
|
21
34
|
|
|
35
|
+
connectionId: string;
|
|
36
|
+
|
|
22
37
|
/**
|
|
23
38
|
* Initialize the interval.
|
|
24
39
|
*
|
|
@@ -32,6 +47,7 @@ export default class RtcMetrics {
|
|
|
32
47
|
this.meetingId = meetingId;
|
|
33
48
|
this.webex = webex;
|
|
34
49
|
this.correlationId = correlationId;
|
|
50
|
+
this.setNewConnectionId();
|
|
35
51
|
// Send the first set of metrics at 5 seconds in the case of a user leaving the call shortly after joining.
|
|
36
52
|
setTimeout(this.sendMetricsInQueue.bind(this), 5 * 1000);
|
|
37
53
|
}
|
|
@@ -60,7 +76,23 @@ export default class RtcMetrics {
|
|
|
60
76
|
if (data.name === 'stats-report') {
|
|
61
77
|
data.payload = data.payload.map(this.anonymizeIp);
|
|
62
78
|
}
|
|
79
|
+
|
|
63
80
|
this.metricsQueue.push(data);
|
|
81
|
+
|
|
82
|
+
try {
|
|
83
|
+
// If a connection fails, send the rest of the metrics in queue and get a new connection id.
|
|
84
|
+
const parsedPayload = parseJsonPayload(data.payload);
|
|
85
|
+
if (
|
|
86
|
+
data.name === 'onconnectionstatechange' &&
|
|
87
|
+
parsedPayload &&
|
|
88
|
+
parsedPayload.value === 'failed'
|
|
89
|
+
) {
|
|
90
|
+
this.sendMetricsInQueue();
|
|
91
|
+
this.setNewConnectionId();
|
|
92
|
+
}
|
|
93
|
+
} catch (e) {
|
|
94
|
+
console.error(e);
|
|
95
|
+
}
|
|
64
96
|
}
|
|
65
97
|
}
|
|
66
98
|
|
|
@@ -93,6 +125,15 @@ export default class RtcMetrics {
|
|
|
93
125
|
return JSON.stringify(data);
|
|
94
126
|
}
|
|
95
127
|
|
|
128
|
+
/**
|
|
129
|
+
* Set a new connection id.
|
|
130
|
+
*
|
|
131
|
+
* @returns {void}
|
|
132
|
+
*/
|
|
133
|
+
private setNewConnectionId() {
|
|
134
|
+
this.connectionId = uuid.v4();
|
|
135
|
+
}
|
|
136
|
+
|
|
96
137
|
/**
|
|
97
138
|
* Send metrics to the metrics service.
|
|
98
139
|
*
|
|
@@ -111,10 +152,11 @@ export default class RtcMetrics {
|
|
|
111
152
|
metrics: [
|
|
112
153
|
{
|
|
113
154
|
type: 'webrtc',
|
|
114
|
-
version: '1.0
|
|
155
|
+
version: '1.1.0',
|
|
115
156
|
userId: this.webex.internal.device.userId,
|
|
116
157
|
meetingId: this.meetingId,
|
|
117
158
|
correlationId: this.correlationId,
|
|
159
|
+
connectionId: this.connectionId,
|
|
118
160
|
data: this.metricsQueue,
|
|
119
161
|
},
|
|
120
162
|
],
|
|
@@ -911,6 +911,7 @@ export class StatsAnalyzer extends EventsScope {
|
|
|
911
911
|
if (result.bytesReceived) {
|
|
912
912
|
let kilobytes = 0;
|
|
913
913
|
const receiveSlot = this.receiveSlotCallback(result.ssrc);
|
|
914
|
+
const sourceState = receiveSlot?.sourceState;
|
|
914
915
|
const idAndCsi = receiveSlot
|
|
915
916
|
? `id: "${receiveSlot.id || ''}"${receiveSlot.csi ? ` and csi: ${receiveSlot.csi}` : ''}`
|
|
916
917
|
: '';
|
|
@@ -938,10 +939,10 @@ export class StatsAnalyzer extends EventsScope {
|
|
|
938
939
|
this.statsResults[mediaType][sendrecvType].totalPacketsReceived = result.packetsReceived;
|
|
939
940
|
|
|
940
941
|
if (currentPacketsReceived === 0) {
|
|
941
|
-
if (receiveSlot) {
|
|
942
|
+
if (receiveSlot && sourceState === 'live') {
|
|
942
943
|
LoggerProxy.logger.info(
|
|
943
|
-
`StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ${idAndCsi}`,
|
|
944
|
-
|
|
944
|
+
`StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot ${idAndCsi}. Total packets received on slot: `,
|
|
945
|
+
result.packetsReceived
|
|
945
946
|
);
|
|
946
947
|
}
|
|
947
948
|
}
|
|
@@ -4,10 +4,15 @@ import {assert} from '@webex/test-helper-chai';
|
|
|
4
4
|
import sinon from 'sinon';
|
|
5
5
|
import RTC_METRICS from '../../../../src/rtcMetrics/constants';
|
|
6
6
|
|
|
7
|
-
const FAKE_METRICS_ITEM = {payload: ['fake-metrics']};
|
|
7
|
+
const FAKE_METRICS_ITEM = {payload: ['{"type":"string","value":"fake-metrics","id":""}']};
|
|
8
|
+
const FAILURE_METRICS_ITEM = {
|
|
9
|
+
name: "onconnectionstatechange",
|
|
10
|
+
payload: ['{"type":"string","value":"failed","id":""}'],
|
|
11
|
+
timestamp: 1707929986667
|
|
12
|
+
};
|
|
8
13
|
|
|
9
|
-
const STATS_WITH_IP = '{
|
|
10
|
-
const STATS_WITH_IP_RESULT = '{
|
|
14
|
+
const STATS_WITH_IP = '{"id":"RTCIceCandidate_/kQs0ZNU","type":"remote-candidate","transportId":"RTCTransport_0_1","isRemote":true,"ip":"11.22.111.255","address":"11.22.111.255","port":5004,"protocol":"udp","candidateType":"host","priority":2130706431}';
|
|
15
|
+
const STATS_WITH_IP_RESULT = '{"id":"RTCIceCandidate_/kQs0ZNU","type":"remote-candidate","transportId":"RTCTransport_0_1","isRemote":true,"ip":"11.22.111.240","address":"11.22.111.240","port":5004,"protocol":"udp","candidateType":"host","priority":2130706431}';
|
|
11
16
|
|
|
12
17
|
describe('RtcMetrics', () => {
|
|
13
18
|
let metrics: RtcMetrics;
|
|
@@ -82,6 +87,30 @@ describe('RtcMetrics', () => {
|
|
|
82
87
|
assert.callCount(webex.request, 1);
|
|
83
88
|
});
|
|
84
89
|
|
|
90
|
+
it('should clear out metrics on failure', () => {
|
|
91
|
+
assert.notCalled(webex.request);
|
|
92
|
+
|
|
93
|
+
metrics.addMetrics(FAILURE_METRICS_ITEM);
|
|
94
|
+
|
|
95
|
+
assert.callCount(webex.request, 1);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('should have the same connectionId on success', () => {
|
|
99
|
+
const originalId = metrics.connectionId;
|
|
100
|
+
|
|
101
|
+
metrics.addMetrics(FAKE_METRICS_ITEM);
|
|
102
|
+
|
|
103
|
+
assert.strictEqual(originalId, metrics.connectionId);
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
it('should have a new connectionId on failure', () => {
|
|
107
|
+
const originalId = metrics.connectionId;
|
|
108
|
+
|
|
109
|
+
metrics.addMetrics(FAILURE_METRICS_ITEM);
|
|
110
|
+
|
|
111
|
+
assert.notEqual(originalId, metrics.connectionId);
|
|
112
|
+
});
|
|
113
|
+
|
|
85
114
|
it('should anonymize IP addresses', () => {
|
|
86
115
|
assert.strictEqual(metrics.anonymizeIp(STATS_WITH_IP), STATS_WITH_IP_RESULT);
|
|
87
116
|
});
|
|
@@ -244,6 +244,7 @@ describe('plugin-meetings', () => {
|
|
|
244
244
|
let statsAnalyzer;
|
|
245
245
|
let mqeData;
|
|
246
246
|
let loggerSpy;
|
|
247
|
+
let receiveSlot;
|
|
247
248
|
|
|
248
249
|
let receivedEventsData = {
|
|
249
250
|
local: {},
|
|
@@ -273,6 +274,7 @@ describe('plugin-meetings', () => {
|
|
|
273
274
|
|
|
274
275
|
beforeEach(() => {
|
|
275
276
|
clock = sinon.useFakeTimers();
|
|
277
|
+
receiveSlot = undefined;
|
|
276
278
|
|
|
277
279
|
resetReceivedEvents();
|
|
278
280
|
|
|
@@ -437,7 +439,7 @@ describe('plugin-meetings', () => {
|
|
|
437
439
|
|
|
438
440
|
networkQualityMonitor = new NetworkQualityMonitor(initialConfig);
|
|
439
441
|
|
|
440
|
-
statsAnalyzer = new StatsAnalyzer(initialConfig, () => (
|
|
442
|
+
statsAnalyzer = new StatsAnalyzer(initialConfig, () => (receiveSlot), networkQualityMonitor);
|
|
441
443
|
|
|
442
444
|
statsAnalyzer.on(EVENTS.LOCAL_MEDIA_STARTED, (data) => {
|
|
443
445
|
receivedEventsData.local.started = data;
|
|
@@ -912,6 +914,47 @@ describe('plugin-meetings', () => {
|
|
|
912
914
|
|
|
913
915
|
assert(loggerSpy.neverCalledWith('StatsAnalyzer:index#compareLastStatsResult --> No audio RTP packets received'));
|
|
914
916
|
});
|
|
917
|
+
|
|
918
|
+
it('logs a message when no packets are recieved for a receive slot with sourceState "live"', async () => {
|
|
919
|
+
receiveSlot = {
|
|
920
|
+
sourceState: 'live',
|
|
921
|
+
csi: 2,
|
|
922
|
+
id: "4",
|
|
923
|
+
};
|
|
924
|
+
|
|
925
|
+
await startStatsAnalyzer();
|
|
926
|
+
|
|
927
|
+
// don't increase the packets when time progresses.
|
|
928
|
+
await progressTime();
|
|
929
|
+
|
|
930
|
+
assert.calledWith(loggerSpy, 'StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot id: "4" and csi: 2. Total packets received on slot: ', 0);
|
|
931
|
+
});
|
|
932
|
+
|
|
933
|
+
["avatar", "invalid", "no source", "bandwidth limited", "policy violation"].forEach((sourceState) => {
|
|
934
|
+
it(`does not log a message when no packets are recieved for a receive slot with sourceState "${sourceState}"`, async () => {
|
|
935
|
+
receiveSlot = {
|
|
936
|
+
sourceState,
|
|
937
|
+
csi: 2,
|
|
938
|
+
id: "4",
|
|
939
|
+
};
|
|
940
|
+
|
|
941
|
+
await startStatsAnalyzer();
|
|
942
|
+
|
|
943
|
+
// don't increase the packets when time progresses.
|
|
944
|
+
await progressTime();
|
|
945
|
+
|
|
946
|
+
assert.neverCalledWith(loggerSpy, 'StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot id: "4" and csi: 2. Total packets received on slot: ', 0);
|
|
947
|
+
});
|
|
948
|
+
});
|
|
949
|
+
|
|
950
|
+
it(`does not log a message if receiveSlot is undefined`, async () => {
|
|
951
|
+
await startStatsAnalyzer();
|
|
952
|
+
|
|
953
|
+
// don't increase the packets when time progresses.
|
|
954
|
+
await progressTime();
|
|
955
|
+
|
|
956
|
+
assert.neverCalledWith(loggerSpy, 'StatsAnalyzer:index#processInboundRTPResult --> No packets received for receive slot "". Total packets received on slot: ', 0);
|
|
957
|
+
});
|
|
915
958
|
});
|
|
916
959
|
});
|
|
917
960
|
});
|