You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

statistics.bundle.js 94KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281
  1. !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.statistics=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
  2. /**
  3. * Provides statistics for the local stream.
  4. */
  5. /**
  6. * Size of the webaudio analizer buffer.
  7. * @type {number}
  8. */
  9. var WEBAUDIO_ANALIZER_FFT_SIZE = 2048;
  10. /**
  11. * Value of the webaudio analizer smoothing time parameter.
  12. * @type {number}
  13. */
  14. var WEBAUDIO_ANALIZER_SMOOTING_TIME = 0.8;
  15. /**
  16. * Converts time domain data array to audio level.
  17. * @param array the time domain data array.
  18. * @returns {number} the audio level
  19. */
  20. function timeDomainDataToAudioLevel(samples) {
  21. var maxVolume = 0;
  22. var length = samples.length;
  23. for (var i = 0; i < length; i++) {
  24. if (maxVolume < samples[i])
  25. maxVolume = samples[i];
  26. }
  27. return parseFloat(((maxVolume - 127) / 128).toFixed(3));
  28. };
  29. /**
  30. * Animates audio level change
  31. * @param newLevel the new audio level
  32. * @param lastLevel the last audio level
  33. * @returns {Number} the audio level to be set
  34. */
  35. function animateLevel(newLevel, lastLevel)
  36. {
  37. var value = 0;
  38. var diff = lastLevel - newLevel;
  39. if(diff > 0.2)
  40. {
  41. value = lastLevel - 0.2;
  42. }
  43. else if(diff < -0.4)
  44. {
  45. value = lastLevel + 0.4;
  46. }
  47. else
  48. {
  49. value = newLevel;
  50. }
  51. return parseFloat(value.toFixed(3));
  52. }
  53. /**
  54. * <tt>LocalStatsCollector</tt> calculates statistics for the local stream.
  55. *
  56. * @param stream the local stream
  57. * @param interval stats refresh interval given in ms.
  58. * @param {function(LocalStatsCollector)} updateCallback the callback called on stats
  59. * update.
  60. * @constructor
  61. */
  62. function LocalStatsCollector(stream, interval, statisticsService, eventEmitter) {
  63. window.AudioContext = window.AudioContext || window.webkitAudioContext;
  64. this.stream = stream;
  65. this.intervalId = null;
  66. this.intervalMilis = interval;
  67. this.eventEmitter = eventEmitter;
  68. this.audioLevel = 0;
  69. this.statisticsService = statisticsService;
  70. }
  71. /**
  72. * Starts the collecting the statistics.
  73. */
  74. LocalStatsCollector.prototype.start = function () {
  75. if (!window.AudioContext)
  76. return;
  77. var context = new AudioContext();
  78. var analyser = context.createAnalyser();
  79. analyser.smoothingTimeConstant = WEBAUDIO_ANALIZER_SMOOTING_TIME;
  80. analyser.fftSize = WEBAUDIO_ANALIZER_FFT_SIZE;
  81. var source = context.createMediaStreamSource(this.stream);
  82. source.connect(analyser);
  83. var self = this;
  84. this.intervalId = setInterval(
  85. function () {
  86. var array = new Uint8Array(analyser.frequencyBinCount);
  87. analyser.getByteTimeDomainData(array);
  88. var audioLevel = timeDomainDataToAudioLevel(array);
  89. if(audioLevel != self.audioLevel) {
  90. self.audioLevel = animateLevel(audioLevel, self.audioLevel);
  91. self.eventEmitter.emit(
  92. "statistics.audioLevel",
  93. self.statisticsService.LOCAL_JID,
  94. self.audioLevel);
  95. }
  96. },
  97. this.intervalMilis
  98. );
  99. };
  100. /**
  101. * Stops collecting the statistics.
  102. */
  103. LocalStatsCollector.prototype.stop = function () {
  104. if (this.intervalId) {
  105. clearInterval(this.intervalId);
  106. this.intervalId = null;
  107. }
  108. };
  109. module.exports = LocalStatsCollector;
  110. },{}],2:[function(require,module,exports){
  111. /* global focusMucJid, ssrc2jid */
  112. /* jshint -W117 */
  113. /**
  114. * Calculates packet lost percent using the number of lost packets and the
  115. * number of all packet.
  116. * @param lostPackets the number of lost packets
  117. * @param totalPackets the number of all packets.
  118. * @returns {number} packet loss percent
  119. */
  120. function calculatePacketLoss(lostPackets, totalPackets) {
  121. if(!totalPackets || totalPackets <= 0 || !lostPackets || lostPackets <= 0)
  122. return 0;
  123. return Math.round((lostPackets/totalPackets)*100);
  124. }
  125. function getStatValue(item, name) {
  126. if(!keyMap[RTC.getBrowserType()][name])
  127. throw "The property isn't supported!";
  128. var key = keyMap[RTC.getBrowserType()][name];
  129. return RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_CHROME? item.stat(key) : item[key];
  130. }
  131. /**
  132. * Peer statistics data holder.
  133. * @constructor
  134. */
  135. function PeerStats()
  136. {
  137. this.ssrc2Loss = {};
  138. this.ssrc2AudioLevel = {};
  139. this.ssrc2bitrate = {};
  140. this.ssrc2resolution = {};
  141. }
  142. /**
  143. * The bandwidth
  144. * @type {{}}
  145. */
  146. PeerStats.bandwidth = {};
  147. /**
  148. * The bit rate
  149. * @type {{}}
  150. */
  151. PeerStats.bitrate = {};
  152. /**
  153. * The packet loss rate
  154. * @type {{}}
  155. */
  156. PeerStats.packetLoss = null;
  157. /**
  158. * Sets packets loss rate for given <tt>ssrc</tt> that blong to the peer
  159. * represented by this instance.
  160. * @param ssrc audio or video RTP stream SSRC.
  161. * @param lossRate new packet loss rate value to be set.
  162. */
  163. PeerStats.prototype.setSsrcLoss = function (ssrc, lossRate)
  164. {
  165. this.ssrc2Loss[ssrc] = lossRate;
  166. };
  167. /**
  168. * Sets resolution for given <tt>ssrc</tt> that belong to the peer
  169. * represented by this instance.
  170. * @param ssrc audio or video RTP stream SSRC.
  171. * @param resolution new resolution value to be set.
  172. */
  173. PeerStats.prototype.setSsrcResolution = function (ssrc, resolution)
  174. {
  175. if(resolution === null && this.ssrc2resolution[ssrc])
  176. {
  177. delete this.ssrc2resolution[ssrc];
  178. }
  179. else if(resolution !== null)
  180. this.ssrc2resolution[ssrc] = resolution;
  181. };
  182. /**
  183. * Sets the bit rate for given <tt>ssrc</tt> that blong to the peer
  184. * represented by this instance.
  185. * @param ssrc audio or video RTP stream SSRC.
  186. * @param bitrate new bitrate value to be set.
  187. */
  188. PeerStats.prototype.setSsrcBitrate = function (ssrc, bitrate)
  189. {
  190. if(this.ssrc2bitrate[ssrc])
  191. {
  192. this.ssrc2bitrate[ssrc].download += bitrate.download;
  193. this.ssrc2bitrate[ssrc].upload += bitrate.upload;
  194. }
  195. else {
  196. this.ssrc2bitrate[ssrc] = bitrate;
  197. }
  198. };
  199. /**
  200. * Sets new audio level(input or output) for given <tt>ssrc</tt> that identifies
  201. * the stream which belongs to the peer represented by this instance.
  202. * @param ssrc RTP stream SSRC for which current audio level value will be
  203. * updated.
  204. * @param audioLevel the new audio level value to be set. Value is truncated to
  205. * fit the range from 0 to 1.
  206. */
  207. PeerStats.prototype.setSsrcAudioLevel = function (ssrc, audioLevel)
  208. {
  209. // Range limit 0 - 1
  210. this.ssrc2AudioLevel[ssrc] = Math.min(Math.max(audioLevel, 0), 1);
  211. };
  212. /**
  213. * Array with the transport information.
  214. * @type {Array}
  215. */
  216. PeerStats.transport = [];
  217. /**
  218. * <tt>StatsCollector</tt> registers for stats updates of given
  219. * <tt>peerconnection</tt> in given <tt>interval</tt>. On each update particular
  220. * stats are extracted and put in {@link PeerStats} objects. Once the processing
  221. * is done <tt>audioLevelsUpdateCallback</tt> is called with <tt>this</tt>
  222. * instance as an event source.
  223. *
  224. * @param peerconnection webRTC peer connection object.
  225. * @param interval stats refresh interval given in ms.
  226. * @param {function(StatsCollector)} audioLevelsUpdateCallback the callback
  227. * called on stats update.
  228. * @constructor
  229. */
  230. function StatsCollector(peerconnection, audioLevelsInterval, statsInterval, eventEmitter)
  231. {
  232. this.peerconnection = peerconnection;
  233. this.baselineAudioLevelsReport = null;
  234. this.currentAudioLevelsReport = null;
  235. this.currentStatsReport = null;
  236. this.baselineStatsReport = null;
  237. this.audioLevelsIntervalId = null;
  238. this.eventEmitter = eventEmitter;
  239. /**
  240. * Gather PeerConnection stats once every this many milliseconds.
  241. */
  242. this.GATHER_INTERVAL = 10000;
  243. /**
  244. * Log stats via the focus once every this many milliseconds.
  245. */
  246. this.LOG_INTERVAL = 60000;
  247. /**
  248. * Gather stats and store them in this.statsToBeLogged.
  249. */
  250. this.gatherStatsIntervalId = null;
  251. /**
  252. * Send the stats already saved in this.statsToBeLogged to be logged via
  253. * the focus.
  254. */
  255. this.logStatsIntervalId = null;
  256. /**
  257. * Stores the statistics which will be send to the focus to be logged.
  258. */
  259. this.statsToBeLogged =
  260. {
  261. timestamps: [],
  262. stats: {}
  263. };
  264. // Updates stats interval
  265. this.audioLevelsIntervalMilis = audioLevelsInterval;
  266. this.statsIntervalId = null;
  267. this.statsIntervalMilis = statsInterval;
  268. // Map of jids to PeerStats
  269. this.jid2stats = {};
  270. }
  271. module.exports = StatsCollector;
  272. /**
  273. * Stops stats updates.
  274. */
  275. StatsCollector.prototype.stop = function ()
  276. {
  277. if (this.audioLevelsIntervalId)
  278. {
  279. clearInterval(this.audioLevelsIntervalId);
  280. this.audioLevelsIntervalId = null;
  281. clearInterval(this.statsIntervalId);
  282. this.statsIntervalId = null;
  283. clearInterval(this.logStatsIntervalId);
  284. this.logStatsIntervalId = null;
  285. clearInterval(this.gatherStatsIntervalId);
  286. this.gatherStatsIntervalId = null;
  287. }
  288. };
  289. /**
  290. * Callback passed to <tt>getStats</tt> method.
  291. * @param error an error that occurred on <tt>getStats</tt> call.
  292. */
  293. StatsCollector.prototype.errorCallback = function (error)
  294. {
  295. console.error("Get stats error", error);
  296. this.stop();
  297. };
  298. /**
  299. * Starts stats updates.
  300. */
  301. StatsCollector.prototype.start = function ()
  302. {
  303. var self = this;
  304. this.audioLevelsIntervalId = setInterval(
  305. function ()
  306. {
  307. // Interval updates
  308. self.peerconnection.getStats(
  309. function (report)
  310. {
  311. var results = null;
  312. if(!report || !report.result || typeof report.result != 'function')
  313. {
  314. results = report;
  315. }
  316. else
  317. {
  318. results = report.result();
  319. }
  320. //console.error("Got interval report", results);
  321. self.currentAudioLevelsReport = results;
  322. self.processAudioLevelReport();
  323. self.baselineAudioLevelsReport =
  324. self.currentAudioLevelsReport;
  325. },
  326. self.errorCallback
  327. );
  328. },
  329. self.audioLevelsIntervalMilis
  330. );
  331. this.statsIntervalId = setInterval(
  332. function () {
  333. // Interval updates
  334. self.peerconnection.getStats(
  335. function (report)
  336. {
  337. var results = null;
  338. if(!report || !report.result || typeof report.result != 'function')
  339. {
  340. //firefox
  341. results = report;
  342. }
  343. else
  344. {
  345. //chrome
  346. results = report.result();
  347. }
  348. //console.error("Got interval report", results);
  349. self.currentStatsReport = results;
  350. try
  351. {
  352. self.processStatsReport();
  353. }
  354. catch (e)
  355. {
  356. console.error("Unsupported key:" + e, e);
  357. }
  358. self.baselineStatsReport = self.currentStatsReport;
  359. },
  360. self.errorCallback
  361. );
  362. },
  363. self.statsIntervalMilis
  364. );
  365. if (config.logStats) {
  366. this.gatherStatsIntervalId = setInterval(
  367. function () {
  368. self.peerconnection.getStats(
  369. function (report) {
  370. self.addStatsToBeLogged(report.result());
  371. },
  372. function () {
  373. }
  374. );
  375. },
  376. this.GATHER_INTERVAL
  377. );
  378. this.logStatsIntervalId = setInterval(
  379. function() { self.logStats(); },
  380. this.LOG_INTERVAL);
  381. }
  382. };
  383. /**
  384. * Converts the stats to the format used for logging, and saves the data in
  385. * this.statsToBeLogged.
  386. * @param reports Reports as given by webkitRTCPerConnection.getStats.
  387. */
  388. StatsCollector.prototype.addStatsToBeLogged = function (reports) {
  389. var self = this;
  390. var num_records = this.statsToBeLogged.timestamps.length;
  391. this.statsToBeLogged.timestamps.push(new Date().getTime());
  392. reports.map(function (report) {
  393. var stat = self.statsToBeLogged.stats[report.id];
  394. if (!stat) {
  395. stat = self.statsToBeLogged.stats[report.id] = {};
  396. }
  397. stat.type = report.type;
  398. report.names().map(function (name) {
  399. var values = stat[name];
  400. if (!values) {
  401. values = stat[name] = [];
  402. }
  403. while (values.length < num_records) {
  404. values.push(null);
  405. }
  406. values.push(report.stat(name));
  407. });
  408. });
  409. };
  410. StatsCollector.prototype.logStats = function () {
  411. if (!focusMucJid) {
  412. return;
  413. }
  414. var deflate = true;
  415. var content = JSON.stringify(this.statsToBeLogged);
  416. if (deflate) {
  417. content = String.fromCharCode.apply(null, Pako.deflateRaw(content));
  418. }
  419. content = Base64.encode(content);
  420. xmpp.sendLogs(content);
  421. // Reset the stats
  422. this.statsToBeLogged.stats = {};
  423. this.statsToBeLogged.timestamps = [];
  424. };
  425. var keyMap = {};
  426. keyMap[RTCBrowserType.RTC_BROWSER_FIREFOX] = {
  427. "ssrc": "ssrc",
  428. "packetsReceived": "packetsReceived",
  429. "packetsLost": "packetsLost",
  430. "packetsSent": "packetsSent",
  431. "bytesReceived": "bytesReceived",
  432. "bytesSent": "bytesSent"
  433. };
  434. keyMap[RTCBrowserType.RTC_BROWSER_CHROME] = {
  435. "receiveBandwidth": "googAvailableReceiveBandwidth",
  436. "sendBandwidth": "googAvailableSendBandwidth",
  437. "remoteAddress": "googRemoteAddress",
  438. "transportType": "googTransportType",
  439. "localAddress": "googLocalAddress",
  440. "activeConnection": "googActiveConnection",
  441. "ssrc": "ssrc",
  442. "packetsReceived": "packetsReceived",
  443. "packetsSent": "packetsSent",
  444. "packetsLost": "packetsLost",
  445. "bytesReceived": "bytesReceived",
  446. "bytesSent": "bytesSent",
  447. "googFrameHeightReceived": "googFrameHeightReceived",
  448. "googFrameWidthReceived": "googFrameWidthReceived",
  449. "googFrameHeightSent": "googFrameHeightSent",
  450. "googFrameWidthSent": "googFrameWidthSent",
  451. "audioInputLevel": "audioInputLevel",
  452. "audioOutputLevel": "audioOutputLevel"
  453. };
  454. /**
  455. * Stats processing logic.
  456. */
  457. StatsCollector.prototype.processStatsReport = function () {
  458. if (!this.baselineStatsReport) {
  459. return;
  460. }
  461. for (var idx in this.currentStatsReport) {
  462. var now = this.currentStatsReport[idx];
  463. try {
  464. if (getStatValue(now, 'receiveBandwidth') ||
  465. getStatValue(now, 'sendBandwidth')) {
  466. PeerStats.bandwidth = {
  467. "download": Math.round(
  468. (getStatValue(now, 'receiveBandwidth')) / 1000),
  469. "upload": Math.round(
  470. (getStatValue(now, 'sendBandwidth')) / 1000)
  471. };
  472. }
  473. }
  474. catch(e){/*not supported*/}
  475. if(now.type == 'googCandidatePair')
  476. {
  477. var ip, type, localIP, active;
  478. try {
  479. ip = getStatValue(now, 'remoteAddress');
  480. type = getStatValue(now, "transportType");
  481. localIP = getStatValue(now, "localAddress");
  482. active = getStatValue(now, "activeConnection");
  483. }
  484. catch(e){/*not supported*/}
  485. if(!ip || !type || !localIP || active != "true")
  486. continue;
  487. var addressSaved = false;
  488. for(var i = 0; i < PeerStats.transport.length; i++)
  489. {
  490. if(PeerStats.transport[i].ip == ip &&
  491. PeerStats.transport[i].type == type &&
  492. PeerStats.transport[i].localip == localIP)
  493. {
  494. addressSaved = true;
  495. }
  496. }
  497. if(addressSaved)
  498. continue;
  499. PeerStats.transport.push({localip: localIP, ip: ip, type: type});
  500. continue;
  501. }
  502. if(now.type == "candidatepair")
  503. {
  504. if(now.state == "succeeded")
  505. continue;
  506. var local = this.currentStatsReport[now.localCandidateId];
  507. var remote = this.currentStatsReport[now.remoteCandidateId];
  508. PeerStats.transport.push({localip: local.ipAddress + ":" + local.portNumber,
  509. ip: remote.ipAddress + ":" + remote.portNumber, type: local.transport});
  510. }
  511. if (now.type != 'ssrc' && now.type != "outboundrtp" &&
  512. now.type != "inboundrtp") {
  513. continue;
  514. }
  515. var before = this.baselineStatsReport[idx];
  516. if (!before) {
  517. console.warn(getStatValue(now, 'ssrc') + ' not enough data');
  518. continue;
  519. }
  520. var ssrc = getStatValue(now, 'ssrc');
  521. if(!ssrc)
  522. continue;
  523. var jid = ssrc2jid[ssrc];
  524. if (!jid && (Date.now() - now.timestamp) < 3000) {
  525. console.warn("No jid for ssrc: " + ssrc);
  526. continue;
  527. }
  528. var jidStats = this.jid2stats[jid];
  529. if (!jidStats) {
  530. jidStats = new PeerStats();
  531. this.jid2stats[jid] = jidStats;
  532. }
  533. var isDownloadStream = true;
  534. var key = 'packetsReceived';
  535. if (!getStatValue(now, key))
  536. {
  537. isDownloadStream = false;
  538. key = 'packetsSent';
  539. if (!getStatValue(now, key))
  540. {
  541. console.warn("No packetsReceived nor packetSent stat found");
  542. continue;
  543. }
  544. }
  545. var packetsNow = getStatValue(now, key);
  546. if(!packetsNow || packetsNow < 0)
  547. packetsNow = 0;
  548. var packetsBefore = getStatValue(before, key);
  549. if(!packetsBefore || packetsBefore < 0)
  550. packetsBefore = 0;
  551. var packetRate = packetsNow - packetsBefore;
  552. if(!packetRate || packetRate < 0)
  553. packetRate = 0;
  554. var currentLoss = getStatValue(now, 'packetsLost');
  555. if(!currentLoss || currentLoss < 0)
  556. currentLoss = 0;
  557. var previousLoss = getStatValue(before, 'packetsLost');
  558. if(!previousLoss || previousLoss < 0)
  559. previousLoss = 0;
  560. var lossRate = currentLoss - previousLoss;
  561. if(!lossRate || lossRate < 0)
  562. lossRate = 0;
  563. var packetsTotal = (packetRate + lossRate);
  564. jidStats.setSsrcLoss(ssrc,
  565. {"packetsTotal": packetsTotal,
  566. "packetsLost": lossRate,
  567. "isDownloadStream": isDownloadStream});
  568. var bytesReceived = 0, bytesSent = 0;
  569. if(getStatValue(now, "bytesReceived"))
  570. {
  571. bytesReceived = getStatValue(now, "bytesReceived") -
  572. getStatValue(before, "bytesReceived");
  573. }
  574. if(getStatValue(now, "bytesSent"))
  575. {
  576. bytesSent = getStatValue(now, "bytesSent") -
  577. getStatValue(before, "bytesSent");
  578. }
  579. var time = Math.round((now.timestamp - before.timestamp) / 1000);
  580. if(bytesReceived <= 0 || time <= 0)
  581. {
  582. bytesReceived = 0;
  583. }
  584. else
  585. {
  586. bytesReceived = Math.round(((bytesReceived * 8) / time) / 1000);
  587. }
  588. if(bytesSent <= 0 || time <= 0)
  589. {
  590. bytesSent = 0;
  591. }
  592. else
  593. {
  594. bytesSent = Math.round(((bytesSent * 8) / time) / 1000);
  595. }
  596. jidStats.setSsrcBitrate(ssrc, {
  597. "download": bytesReceived,
  598. "upload": bytesSent});
  599. var resolution = {height: null, width: null};
  600. try {
  601. if (getStatValue(now, "googFrameHeightReceived") &&
  602. getStatValue(now, "googFrameWidthReceived")) {
  603. resolution.height = getStatValue(now, "googFrameHeightReceived");
  604. resolution.width = getStatValue(now, "googFrameWidthReceived");
  605. }
  606. else if (getStatValue(now, "googFrameHeightSent") &&
  607. getStatValue(now, "googFrameWidthSent")) {
  608. resolution.height = getStatValue(now, "googFrameHeightSent");
  609. resolution.width = getStatValue(now, "googFrameWidthSent");
  610. }
  611. }
  612. catch(e){/*not supported*/}
  613. if(resolution.height && resolution.width)
  614. {
  615. jidStats.setSsrcResolution(ssrc, resolution);
  616. }
  617. else
  618. {
  619. jidStats.setSsrcResolution(ssrc, null);
  620. }
  621. }
  622. var self = this;
  623. // Jid stats
  624. var totalPackets = {download: 0, upload: 0};
  625. var lostPackets = {download: 0, upload: 0};
  626. var bitrateDownload = 0;
  627. var bitrateUpload = 0;
  628. var resolutions = {};
  629. Object.keys(this.jid2stats).forEach(
  630. function (jid)
  631. {
  632. Object.keys(self.jid2stats[jid].ssrc2Loss).forEach(
  633. function (ssrc)
  634. {
  635. var type = "upload";
  636. if(self.jid2stats[jid].ssrc2Loss[ssrc].isDownloadStream)
  637. type = "download";
  638. totalPackets[type] +=
  639. self.jid2stats[jid].ssrc2Loss[ssrc].packetsTotal;
  640. lostPackets[type] +=
  641. self.jid2stats[jid].ssrc2Loss[ssrc].packetsLost;
  642. }
  643. );
  644. Object.keys(self.jid2stats[jid].ssrc2bitrate).forEach(
  645. function (ssrc) {
  646. bitrateDownload +=
  647. self.jid2stats[jid].ssrc2bitrate[ssrc].download;
  648. bitrateUpload +=
  649. self.jid2stats[jid].ssrc2bitrate[ssrc].upload;
  650. delete self.jid2stats[jid].ssrc2bitrate[ssrc];
  651. }
  652. );
  653. resolutions[jid] = self.jid2stats[jid].ssrc2resolution;
  654. }
  655. );
  656. PeerStats.bitrate = {"upload": bitrateUpload, "download": bitrateDownload};
  657. PeerStats.packetLoss = {
  658. total:
  659. calculatePacketLoss(lostPackets.download + lostPackets.upload,
  660. totalPackets.download + totalPackets.upload),
  661. download:
  662. calculatePacketLoss(lostPackets.download, totalPackets.download),
  663. upload:
  664. calculatePacketLoss(lostPackets.upload, totalPackets.upload)
  665. };
  666. this.eventEmitter.emit("statistics.connectionstats",
  667. {
  668. "bitrate": PeerStats.bitrate,
  669. "packetLoss": PeerStats.packetLoss,
  670. "bandwidth": PeerStats.bandwidth,
  671. "resolution": resolutions,
  672. "transport": PeerStats.transport
  673. });
  674. PeerStats.transport = [];
  675. };
  676. /**
  677. * Stats processing logic.
  678. */
  679. StatsCollector.prototype.processAudioLevelReport = function ()
  680. {
  681. if (!this.baselineAudioLevelsReport)
  682. {
  683. return;
  684. }
  685. for (var idx in this.currentAudioLevelsReport)
  686. {
  687. var now = this.currentAudioLevelsReport[idx];
  688. if (now.type != 'ssrc')
  689. {
  690. continue;
  691. }
  692. var before = this.baselineAudioLevelsReport[idx];
  693. if (!before)
  694. {
  695. console.warn(getStatValue(now, 'ssrc') + ' not enough data');
  696. continue;
  697. }
  698. var ssrc = getStatValue(now, 'ssrc');
  699. var jid = ssrc2jid[ssrc];
  700. if (!jid && (Date.now() - now.timestamp) < 3000)
  701. {
  702. console.warn("No jid for ssrc: " + ssrc);
  703. continue;
  704. }
  705. var jidStats = this.jid2stats[jid];
  706. if (!jidStats)
  707. {
  708. jidStats = new PeerStats();
  709. this.jid2stats[jid] = jidStats;
  710. }
  711. // Audio level
  712. var audioLevel = null;
  713. try {
  714. audioLevel = getStatValue(now, 'audioInputLevel');
  715. if (!audioLevel)
  716. audioLevel = getStatValue(now, 'audioOutputLevel');
  717. }
  718. catch(e) {/*not supported*/
  719. console.warn("Audio Levels are not available in the statistics.");
  720. clearInterval(this.audioLevelsIntervalId);
  721. return;
  722. }
  723. if (audioLevel)
  724. {
  725. // TODO: can't find specs about what this value really is,
  726. // but it seems to vary between 0 and around 32k.
  727. audioLevel = audioLevel / 32767;
  728. jidStats.setSsrcAudioLevel(ssrc, audioLevel);
  729. if(jid != xmpp.myJid())
  730. this.eventEmitter.emit("statistics.audioLevel", jid, audioLevel);
  731. }
  732. }
  733. };
  734. },{}],3:[function(require,module,exports){
  735. /**
  736. * Created by hristo on 8/4/14.
  737. */
  738. var LocalStats = require("./LocalStatsCollector.js");
  739. var RTPStats = require("./RTPStatsCollector.js");
  740. var EventEmitter = require("events");
  741. //These lines should be uncommented when require works in app.js
  742. //var StreamEventTypes = require("../../service/RTC/StreamEventTypes.js");
  743. //var RTCBrowserType = require("../../service/RTC/RTCBrowserType");
  744. //var XMPPEvents = require("../service/xmpp/XMPPEvents");
  745. var eventEmitter = new EventEmitter();
  746. var localStats = null;
  747. var rtpStats = null;
  748. function stopLocal()
  749. {
  750. if(localStats)
  751. {
  752. localStats.stop();
  753. localStats = null;
  754. }
  755. }
  756. function stopRemote()
  757. {
  758. if(rtpStats)
  759. {
  760. rtpStats.stop();
  761. eventEmitter.emit("statistics.stop");
  762. rtpStats = null;
  763. }
  764. }
  765. function startRemoteStats (peerconnection) {
  766. if (config.enableRtpStats)
  767. {
  768. if(rtpStats)
  769. {
  770. rtpStats.stop();
  771. rtpStats = null;
  772. }
  773. rtpStats = new RTPStats(peerconnection, 200, 2000, eventEmitter);
  774. rtpStats.start();
  775. }
  776. }
  777. function onStreamCreated(stream)
  778. {
  779. if(stream.getOriginalStream().getAudioTracks().length === 0)
  780. return;
  781. localStats = new LocalStats(stream.getOriginalStream(), 100, statistics,
  782. eventEmitter);
  783. localStats.start();
  784. }
  785. function onDisposeConference(onUnload) {
  786. stopRemote();
  787. if(onUnload) {
  788. stopLocal();
  789. eventEmitter.removeAllListeners();
  790. }
  791. }
  792. var statistics =
  793. {
  794. /**
  795. * Indicates that this audio level is for local jid.
  796. * @type {string}
  797. */
  798. LOCAL_JID: 'local',
  799. addAudioLevelListener: function(listener)
  800. {
  801. eventEmitter.on("statistics.audioLevel", listener);
  802. },
  803. removeAudioLevelListener: function(listener)
  804. {
  805. eventEmitter.removeListener("statistics.audioLevel", listener);
  806. },
  807. addConnectionStatsListener: function(listener)
  808. {
  809. eventEmitter.on("statistics.connectionstats", listener);
  810. },
  811. removeConnectionStatsListener: function(listener)
  812. {
  813. eventEmitter.removeListener("statistics.connectionstats", listener);
  814. },
  815. addRemoteStatsStopListener: function(listener)
  816. {
  817. eventEmitter.on("statistics.stop", listener);
  818. },
  819. removeRemoteStatsStopListener: function(listener)
  820. {
  821. eventEmitter.removeListener("statistics.stop", listener);
  822. },
  823. stop: function () {
  824. stopLocal();
  825. stopRemote();
  826. if(eventEmitter)
  827. {
  828. eventEmitter.removeAllListeners();
  829. }
  830. },
  831. stopRemoteStatistics: function()
  832. {
  833. stopRemote();
  834. },
  835. onConferenceCreated: function (event) {
  836. startRemoteStats(event.peerconnection);
  837. },
  838. start: function () {
  839. this.addConnectionStatsListener(connectionquality.updateLocalStats);
  840. this.addRemoteStatsStopListener(connectionquality.stopSendingStats);
  841. RTC.addStreamListener(onStreamCreated,
  842. StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
  843. xmpp.addListener(XMPPEvents.DISPOSE_CONFERENCE, onDisposeConference);
  844. }
  845. };
  846. module.exports = statistics;
  847. },{"./LocalStatsCollector.js":1,"./RTPStatsCollector.js":2,"events":4}],4:[function(require,module,exports){
  848. // Copyright Joyent, Inc. and other Node contributors.
  849. //
  850. // Permission is hereby granted, free of charge, to any person obtaining a
  851. // copy of this software and associated documentation files (the
  852. // "Software"), to deal in the Software without restriction, including
  853. // without limitation the rights to use, copy, modify, merge, publish,
  854. // distribute, sublicense, and/or sell copies of the Software, and to permit
  855. // persons to whom the Software is furnished to do so, subject to the
  856. // following conditions:
  857. //
  858. // The above copyright notice and this permission notice shall be included
  859. // in all copies or substantial portions of the Software.
  860. //
  861. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
  862. // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  863. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
  864. // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
  865. // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
  866. // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
  867. // USE OR OTHER DEALINGS IN THE SOFTWARE.
  868. function EventEmitter() {
  869. this._events = this._events || {};
  870. this._maxListeners = this._maxListeners || undefined;
  871. }
  872. module.exports = EventEmitter;
  873. // Backwards-compat with node 0.10.x
  874. EventEmitter.EventEmitter = EventEmitter;
  875. EventEmitter.prototype._events = undefined;
  876. EventEmitter.prototype._maxListeners = undefined;
  877. // By default EventEmitters will print a warning if more than 10 listeners are
  878. // added to it. This is a useful default which helps finding memory leaks.
  879. EventEmitter.defaultMaxListeners = 10;
  880. // Obviously not all Emitters should be limited to 10. This function allows
  881. // that to be increased. Set to zero for unlimited.
  882. EventEmitter.prototype.setMaxListeners = function(n) {
  883. if (!isNumber(n) || n < 0 || isNaN(n))
  884. throw TypeError('n must be a positive number');
  885. this._maxListeners = n;
  886. return this;
  887. };
  888. EventEmitter.prototype.emit = function(type) {
  889. var er, handler, len, args, i, listeners;
  890. if (!this._events)
  891. this._events = {};
  892. // If there is no 'error' event listener then throw.
  893. if (type === 'error') {
  894. if (!this._events.error ||
  895. (isObject(this._events.error) && !this._events.error.length)) {
  896. er = arguments[1];
  897. if (er instanceof Error) {
  898. throw er; // Unhandled 'error' event
  899. } else {
  900. throw TypeError('Uncaught, unspecified "error" event.');
  901. }
  902. return false;
  903. }
  904. }
  905. handler = this._events[type];
  906. if (isUndefined(handler))
  907. return false;
  908. if (isFunction(handler)) {
  909. switch (arguments.length) {
  910. // fast cases
  911. case 1:
  912. handler.call(this);
  913. break;
  914. case 2:
  915. handler.call(this, arguments[1]);
  916. break;
  917. case 3:
  918. handler.call(this, arguments[1], arguments[2]);
  919. break;
  920. // slower
  921. default:
  922. len = arguments.length;
  923. args = new Array(len - 1);
  924. for (i = 1; i < len; i++)
  925. args[i - 1] = arguments[i];
  926. handler.apply(this, args);
  927. }
  928. } else if (isObject(handler)) {
  929. len = arguments.length;
  930. args = new Array(len - 1);
  931. for (i = 1; i < len; i++)
  932. args[i - 1] = arguments[i];
  933. listeners = handler.slice();
  934. len = listeners.length;
  935. for (i = 0; i < len; i++)
  936. listeners[i].apply(this, args);
  937. }
  938. return true;
  939. };
  940. EventEmitter.prototype.addListener = function(type, listener) {
  941. var m;
  942. if (!isFunction(listener))
  943. throw TypeError('listener must be a function');
  944. if (!this._events)
  945. this._events = {};
  946. // To avoid recursion in the case that type === "newListener"! Before
  947. // adding it to the listeners, first emit "newListener".
  948. if (this._events.newListener)
  949. this.emit('newListener', type,
  950. isFunction(listener.listener) ?
  951. listener.listener : listener);
  952. if (!this._events[type])
  953. // Optimize the case of one listener. Don't need the extra array object.
  954. this._events[type] = listener;
  955. else if (isObject(this._events[type]))
  956. // If we've already got an array, just append.
  957. this._events[type].push(listener);
  958. else
  959. // Adding the second element, need to change to array.
  960. this._events[type] = [this._events[type], listener];
  961. // Check for listener leak
  962. if (isObject(this._events[type]) && !this._events[type].warned) {
  963. var m;
  964. if (!isUndefined(this._maxListeners)) {
  965. m = this._maxListeners;
  966. } else {
  967. m = EventEmitter.defaultMaxListeners;
  968. }
  969. if (m && m > 0 && this._events[type].length > m) {
  970. this._events[type].warned = true;
  971. console.error('(node) warning: possible EventEmitter memory ' +
  972. 'leak detected. %d listeners added. ' +
  973. 'Use emitter.setMaxListeners() to increase limit.',
  974. this._events[type].length);
  975. if (typeof console.trace === 'function') {
  976. // not supported in IE 10
  977. console.trace();
  978. }
  979. }
  980. }
  981. return this;
  982. };
  983. EventEmitter.prototype.on = EventEmitter.prototype.addListener;
  984. EventEmitter.prototype.once = function(type, listener) {
  985. if (!isFunction(listener))
  986. throw TypeError('listener must be a function');
  987. var fired = false;
  988. function g() {
  989. this.removeListener(type, g);
  990. if (!fired) {
  991. fired = true;
  992. listener.apply(this, arguments);
  993. }
  994. }
  995. g.listener = listener;
  996. this.on(type, g);
  997. return this;
  998. };
  999. // emits a 'removeListener' event iff the listener was removed
  1000. EventEmitter.prototype.removeListener = function(type, listener) {
  1001. var list, position, length, i;
  1002. if (!isFunction(listener))
  1003. throw TypeError('listener must be a function');
  1004. if (!this._events || !this._events[type])
  1005. return this;
  1006. list = this._events[type];
  1007. length = list.length;
  1008. position = -1;
  1009. if (list === listener ||
  1010. (isFunction(list.listener) && list.listener === listener)) {
  1011. delete this._events[type];
  1012. if (this._events.removeListener)
  1013. this.emit('removeListener', type, listener);
  1014. } else if (isObject(list)) {
  1015. for (i = length; i-- > 0;) {
  1016. if (list[i] === listener ||
  1017. (list[i].listener && list[i].listener === listener)) {
  1018. position = i;
  1019. break;
  1020. }
  1021. }
  1022. if (position < 0)
  1023. return this;
  1024. if (list.length === 1) {
  1025. list.length = 0;
  1026. delete this._events[type];
  1027. } else {
  1028. list.splice(position, 1);
  1029. }
  1030. if (this._events.removeListener)
  1031. this.emit('removeListener', type, listener);
  1032. }
  1033. return this;
  1034. };
  1035. EventEmitter.prototype.removeAllListeners = function(type) {
  1036. var key, listeners;
  1037. if (!this._events)
  1038. return this;
  1039. // not listening for removeListener, no need to emit
  1040. if (!this._events.removeListener) {
  1041. if (arguments.length === 0)
  1042. this._events = {};
  1043. else if (this._events[type])
  1044. delete this._events[type];
  1045. return this;
  1046. }
  1047. // emit removeListener for all listeners on all events
  1048. if (arguments.length === 0) {
  1049. for (key in this._events) {
  1050. if (key === 'removeListener') continue;
  1051. this.removeAllListeners(key);
  1052. }
  1053. this.removeAllListeners('removeListener');
  1054. this._events = {};
  1055. return this;
  1056. }
  1057. listeners = this._events[type];
  1058. if (isFunction(listeners)) {
  1059. this.removeListener(type, listeners);
  1060. } else {
  1061. // LIFO order
  1062. while (listeners.length)
  1063. this.removeListener(type, listeners[listeners.length - 1]);
  1064. }
  1065. delete this._events[type];
  1066. return this;
  1067. };
  1068. EventEmitter.prototype.listeners = function(type) {
  1069. var ret;
  1070. if (!this._events || !this._events[type])
  1071. ret = [];
  1072. else if (isFunction(this._events[type]))
  1073. ret = [this._events[type]];
  1074. else
  1075. ret = this._events[type].slice();
  1076. return ret;
  1077. };
  1078. EventEmitter.listenerCount = function(emitter, type) {
  1079. var ret;
  1080. if (!emitter._events || !emitter._events[type])
  1081. ret = 0;
  1082. else if (isFunction(emitter._events[type]))
  1083. ret = 1;
  1084. else
  1085. ret = emitter._events[type].length;
  1086. return ret;
  1087. };
  1088. function isFunction(arg) {
  1089. return typeof arg === 'function';
  1090. }
  1091. function isNumber(arg) {
  1092. return typeof arg === 'number';
  1093. }
  1094. function isObject(arg) {
  1095. return typeof arg === 'object' && arg !== null;
  1096. }
  1097. function isUndefined(arg) {
  1098. return arg === void 0;
  1099. }
  1100. },{}]},{},[3])(3)
  1101. });
  1102. //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/usr/local/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/statistics/LocalStatsCollector.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/statistics/RTPStatsCollector.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/statistics/statistics.js","/usr/local/lib/node_modules/browserify/node_modules/events/events.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/**\n * Provides statistics for the local stream.\n */\n\n\n/**\n * Size of the webaudio analizer buffer.\n * @type {number}\n */\nvar WEBAUDIO_ANALIZER_FFT_SIZE = 2048;\n\n/**\n * Value of the webaudio analizer smoothing time parameter.\n * @type {number}\n */\nvar WEBAUDIO_ANALIZER_SMOOTING_TIME = 0.8;\n\n/**\n * Converts time domain data array to audio level.\n * @param array the time domain data array.\n * @returns {number} the audio level\n */\nfunction timeDomainDataToAudioLevel(samples) {\n\n    var maxVolume = 0;\n\n    var length = samples.length;\n\n    for (var i = 0; i < length; i++) {\n        if (maxVolume < samples[i])\n            maxVolume = samples[i];\n    }\n\n    return parseFloat(((maxVolume - 127) / 128).toFixed(3));\n};\n\n/**\n * Animates audio level change\n * @param newLevel the new audio level\n * @param lastLevel the last audio level\n * @returns {Number} the audio level to be set\n */\nfunction animateLevel(newLevel, lastLevel)\n{\n    var value = 0;\n    var diff = lastLevel - newLevel;\n    if(diff > 0.2)\n    {\n        value = lastLevel - 0.2;\n    }\n    else if(diff < -0.4)\n    {\n        value = lastLevel + 0.4;\n    }\n    else\n    {\n        value = newLevel;\n    }\n\n    return parseFloat(value.toFixed(3));\n}\n\n\n/**\n * <tt>LocalStatsCollector</tt> calculates statistics for the local stream.\n *\n * @param stream the local stream\n * @param interval stats refresh interval given in ms.\n * @param {function(LocalStatsCollector)} updateCallback the callback called on stats\n *                                   update.\n * @constructor\n */\nfunction LocalStatsCollector(stream, interval, statisticsService, eventEmitter) {\n    window.AudioContext = window.AudioContext || window.webkitAudioContext;\n    this.stream = stream;\n    this.intervalId = null;\n    this.intervalMilis = interval;\n    this.eventEmitter = eventEmitter;\n    this.audioLevel = 0;\n    this.statisticsService = statisticsService;\n}\n\n/**\n * Starts the collecting the statistics.\n */\nLocalStatsCollector.prototype.start = function () {\n    if (!window.AudioContext)\n        return;\n\n    var context = new AudioContext();\n    var analyser = context.createAnalyser();\n    analyser.smoothingTimeConstant = WEBAUDIO_ANALIZER_SMOOTING_TIME;\n    analyser.fftSize = WEBAUDIO_ANALIZER_FFT_SIZE;\n\n\n    var source = context.createMediaStreamSource(this.stream);\n    source.connect(analyser);\n\n\n    var self = this;\n\n    this.intervalId = setInterval(\n        function () {\n            var array = new Uint8Array(analyser.frequencyBinCount);\n            analyser.getByteTimeDomainData(array);\n            var audioLevel = timeDomainDataToAudioLevel(array);\n            if(audioLevel != self.audioLevel) {\n                self.audioLevel = animateLevel(audioLevel, self.audioLevel);\n                self.eventEmitter.emit(\n                    \"statistics.audioLevel\",\n                    self.statisticsService.LOCAL_JID,\n                    self.audioLevel);\n            }\n        },\n        this.intervalMilis\n    );\n\n};\n\n/**\n * Stops collecting the statistics.\n */\nLocalStatsCollector.prototype.stop = function () {\n    if (this.intervalId) {\n        clearInterval(this.intervalId);\n        this.intervalId = null;\n    }\n};\n\nmodule.exports = LocalStatsCollector;","/* global focusMucJid, ssrc2jid */\n/* jshint -W117 */\n/**\n * Calculates packet lost percent using the number of lost packets and the\n * number of all packet.\n * @param lostPackets the number of lost packets\n * @param totalPackets the number of all packets.\n * @returns {number} packet loss percent\n */\nfunction calculatePacketLoss(lostPackets, totalPackets) {\n    if(!totalPackets || totalPackets <= 0 || !lostPackets || lostPackets <= 0)\n        return 0;\n    return Math.round((lostPackets/totalPackets)*100);\n}\n\nfunction getStatValue(item, name) {\n    if(!keyMap[RTC.getBrowserType()][name])\n        throw \"The property isn't supported!\";\n    var key = keyMap[RTC.getBrowserType()][name];\n    return RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_CHROME? item.stat(key) : item[key];\n}\n\n/**\n * Peer statistics data holder.\n * @constructor\n */\nfunction PeerStats()\n{\n    this.ssrc2Loss = {};\n    this.ssrc2AudioLevel = {};\n    this.ssrc2bitrate = {};\n    this.ssrc2resolution = {};\n}\n\n/**\n * The bandwidth\n * @type {{}}\n */\nPeerStats.bandwidth = {};\n\n/**\n * The bit rate\n * @type {{}}\n */\nPeerStats.bitrate = {};\n\n\n\n/**\n * The packet loss rate\n * @type {{}}\n */\nPeerStats.packetLoss = null;\n\n/**\n * Sets packets loss rate for given <tt>ssrc</tt> that blong to the peer\n * represented by this instance.\n * @param ssrc audio or video RTP stream SSRC.\n * @param lossRate new packet loss rate value to be set.\n */\nPeerStats.prototype.setSsrcLoss = function (ssrc, lossRate)\n{\n    this.ssrc2Loss[ssrc] = lossRate;\n};\n\n/**\n * Sets resolution for given <tt>ssrc</tt> that belong to the peer\n * represented by this instance.\n * @param ssrc audio or video RTP stream SSRC.\n * @param resolution new resolution value to be set.\n */\nPeerStats.prototype.setSsrcResolution = function (ssrc, resolution)\n{\n    if(resolution === null && this.ssrc2resolution[ssrc])\n    {\n        delete this.ssrc2resolution[ssrc];\n    }\n    else if(resolution !== null)\n        this.ssrc2resolution[ssrc] = resolution;\n};\n\n/**\n * Sets the bit rate for given <tt>ssrc</tt> that blong to the peer\n * represented by this instance.\n * @param ssrc audio or video RTP stream SSRC.\n * @param bitrate new bitrate value to be set.\n */\nPeerStats.prototype.setSsrcBitrate = function (ssrc, bitrate)\n{\n    if(this.ssrc2bitrate[ssrc])\n    {\n        this.ssrc2bitrate[ssrc].download += bitrate.download;\n        this.ssrc2bitrate[ssrc].upload += bitrate.upload;\n    }\n    else {\n        this.ssrc2bitrate[ssrc] = bitrate;\n    }\n};\n\n/**\n * Sets new audio level(input or output) for given <tt>ssrc</tt> that identifies\n * the stream which belongs to the peer represented by this instance.\n * @param ssrc RTP stream SSRC for which current audio level value will be\n *        updated.\n * @param audioLevel the new audio level value to be set. Value is truncated to\n *        fit the range from 0 to 1.\n */\nPeerStats.prototype.setSsrcAudioLevel = function (ssrc, audioLevel)\n{\n    // Range limit 0 - 1\n    this.ssrc2AudioLevel[ssrc] = Math.min(Math.max(audioLevel, 0), 1);\n};\n\n/**\n * Array with the transport information.\n * @type {Array}\n */\nPeerStats.transport = [];\n\n\n/**\n * <tt>StatsCollector</tt> registers for stats updates of given\n * <tt>peerconnection</tt> in given <tt>interval</tt>. On each update particular\n * stats are extracted and put in {@link PeerStats} objects. Once the processing\n * is done <tt>audioLevelsUpdateCallback</tt> is called with <tt>this</tt>\n * instance as an event source.\n *\n * @param peerconnection webRTC peer connection object.\n * @param interval stats refresh interval given in ms.\n * @param {function(StatsCollector)} audioLevelsUpdateCallback the callback\n * called on stats update.\n * @constructor\n */\nfunction StatsCollector(peerconnection, audioLevelsInterval, statsInterval, eventEmitter)\n{\n    this.peerconnection = peerconnection;\n    this.baselineAudioLevelsReport = null;\n    this.currentAudioLevelsReport = null;\n    this.currentStatsReport = null;\n    this.baselineStatsReport = null;\n    this.audioLevelsIntervalId = null;\n    this.eventEmitter = eventEmitter;\n\n    /**\n     * Gather PeerConnection stats once every this many milliseconds.\n     */\n    this.GATHER_INTERVAL = 10000;\n\n    /**\n     * Log stats via the focus once every this many milliseconds.\n     */\n    this.LOG_INTERVAL = 60000;\n\n    /**\n     * Gather stats and store them in this.statsToBeLogged.\n     */\n    this.gatherStatsIntervalId = null;\n\n    /**\n     * Send the stats already saved in this.statsToBeLogged to be logged via\n     * the focus.\n     */\n    this.logStatsIntervalId = null;\n\n    /**\n     * Stores the statistics which will be send to the focus to be logged.\n     */\n    this.statsToBeLogged =\n    {\n        timestamps: [],\n        stats: {}\n    };\n\n    // Updates stats interval\n    this.audioLevelsIntervalMilis = audioLevelsInterval;\n\n    this.statsIntervalId = null;\n    this.statsIntervalMilis = statsInterval;\n    // Map of jids to PeerStats\n    this.jid2stats = {};\n}\n\nmodule.exports = StatsCollector;\n\n/**\n * Stops stats updates.\n */\nStatsCollector.prototype.stop = function ()\n{\n    if (this.audioLevelsIntervalId)\n    {\n        clearInterval(this.audioLevelsIntervalId);\n        this.audioLevelsIntervalId = null;\n        clearInterval(this.statsIntervalId);\n        this.statsIntervalId = null;\n        clearInterval(this.logStatsIntervalId);\n        this.logStatsIntervalId = null;\n        clearInterval(this.gatherStatsIntervalId);\n        this.gatherStatsIntervalId = null;\n    }\n};\n\n/**\n * Callback passed to <tt>getStats</tt> method.\n * @param error an error that occurred on <tt>getStats</tt> call.\n */\nStatsCollector.prototype.errorCallback = function (error)\n{\n    console.error(\"Get stats error\", error);\n    this.stop();\n};\n\n/**\n * Starts stats updates.\n */\nStatsCollector.prototype.start = function ()\n{\n    var self = this;\n    this.audioLevelsIntervalId = setInterval(\n        function ()\n        {\n            // Interval updates\n            self.peerconnection.getStats(\n                function (report)\n                {\n                    var results = null;\n                    if(!report || !report.result || typeof report.result != 'function')\n                    {\n                        results = report;\n                    }\n                    else\n                    {\n                        results = report.result();\n                    }\n                    //console.error(\"Got interval report\", results);\n                    self.currentAudioLevelsReport = results;\n                    self.processAudioLevelReport();\n                    self.baselineAudioLevelsReport =\n                        self.currentAudioLevelsReport;\n                },\n                self.errorCallback\n            );\n        },\n        self.audioLevelsIntervalMilis\n    );\n\n    this.statsIntervalId = setInterval(\n        function () {\n            // Interval updates\n            self.peerconnection.getStats(\n                function (report)\n                {\n                    var results = null;\n                    if(!report || !report.result || typeof report.result != 'function')\n                    {\n                        //firefox\n                        results = report;\n                    }\n                    else\n                    {\n                        //chrome\n                        results = report.result();\n                    }\n                    //console.error(\"Got interval report\", results);\n                    self.currentStatsReport = results;\n                    try\n                    {\n                        self.processStatsReport();\n                    }\n                    catch (e)\n                    {\n                        console.error(\"Unsupported key:\" + e, e);\n                    }\n\n                    self.baselineStatsReport = self.currentStatsReport;\n                },\n                self.errorCallback\n            );\n        },\n        self.statsIntervalMilis\n    );\n\n    if (config.logStats) {\n        this.gatherStatsIntervalId = setInterval(\n            function () {\n                self.peerconnection.getStats(\n                    function (report) {\n                        self.addStatsToBeLogged(report.result());\n                    },\n                    function () {\n                    }\n                );\n            },\n            this.GATHER_INTERVAL\n        );\n\n        this.logStatsIntervalId = setInterval(\n            function() { self.logStats(); },\n            this.LOG_INTERVAL);\n    }\n};\n\n/**\n * Converts the stats to the format used for logging, and saves the data in\n * this.statsToBeLogged.\n * @param reports Reports as given by webkitRTCPerConnection.getStats.\n */\nStatsCollector.prototype.addStatsToBeLogged = function (reports) {\n    var self = this;\n    var num_records = this.statsToBeLogged.timestamps.length;\n    this.statsToBeLogged.timestamps.push(new Date().getTime());\n    reports.map(function (report) {\n        var stat = self.statsToBeLogged.stats[report.id];\n        if (!stat) {\n            stat = self.statsToBeLogged.stats[report.id] = {};\n        }\n        stat.type = report.type;\n        report.names().map(function (name) {\n            var values = stat[name];\n            if (!values) {\n                values = stat[name] = [];\n            }\n            while (values.length < num_records) {\n                values.push(null);\n            }\n            values.push(report.stat(name));\n        });\n    });\n};\n\nStatsCollector.prototype.logStats = function () {\n    if (!focusMucJid) {\n        return;\n    }\n\n    var deflate = true;\n\n    var content = JSON.stringify(this.statsToBeLogged);\n    if (deflate) {\n        content = String.fromCharCode.apply(null, Pako.deflateRaw(content));\n    }\n    content = Base64.encode(content);\n\n    xmpp.sendLogs(content);\n    // Reset the stats\n    this.statsToBeLogged.stats = {};\n    this.statsToBeLogged.timestamps = [];\n};\nvar keyMap = {};\nkeyMap[RTCBrowserType.RTC_BROWSER_FIREFOX] = {\n    \"ssrc\": \"ssrc\",\n    \"packetsReceived\": \"packetsReceived\",\n    \"packetsLost\": \"packetsLost\",\n    \"packetsSent\": \"packetsSent\",\n    \"bytesReceived\": \"bytesReceived\",\n    \"bytesSent\": \"bytesSent\"\n};\nkeyMap[RTCBrowserType.RTC_BROWSER_CHROME] = {\n    \"receiveBandwidth\": \"googAvailableReceiveBandwidth\",\n    \"sendBandwidth\": \"googAvailableSendBandwidth\",\n    \"remoteAddress\": \"googRemoteAddress\",\n    \"transportType\": \"googTransportType\",\n    \"localAddress\": \"googLocalAddress\",\n    \"activeConnection\": \"googActiveConnection\",\n    \"ssrc\": \"ssrc\",\n    \"packetsReceived\": \"packetsReceived\",\n    \"packetsSent\": \"packetsSent\",\n    \"packetsLost\": \"packetsLost\",\n    \"bytesReceived\": \"bytesReceived\",\n    \"bytesSent\": \"bytesSent\",\n    \"googFrameHeightReceived\": \"googFrameHeightReceived\",\n    \"googFrameWidthReceived\": \"googFrameWidthReceived\",\n    \"googFrameHeightSent\": \"googFrameHeightSent\",\n    \"googFrameWidthSent\": \"googFrameWidthSent\",\n    \"audioInputLevel\": \"audioInputLevel\",\n    \"audioOutputLevel\": \"audioOutputLevel\"\n};\n\n\n/**\n * Stats processing logic.\n */\nStatsCollector.prototype.processStatsReport = function () {\n    if (!this.baselineStatsReport) {\n        return;\n    }\n\n    for (var idx in this.currentStatsReport) {\n        var now = this.currentStatsReport[idx];\n        try {\n            if (getStatValue(now, 'receiveBandwidth') ||\n                getStatValue(now, 'sendBandwidth')) {\n                PeerStats.bandwidth = {\n                    \"download\": Math.round(\n                            (getStatValue(now, 'receiveBandwidth')) / 1000),\n                    \"upload\": Math.round(\n                            (getStatValue(now, 'sendBandwidth')) / 1000)\n                };\n            }\n        }\n        catch(e){/*not supported*/}\n\n        if(now.type == 'googCandidatePair')\n        {\n            var ip, type, localIP, active;\n            try {\n                ip = getStatValue(now, 'remoteAddress');\n                type = getStatValue(now, \"transportType\");\n                localIP = getStatValue(now, \"localAddress\");\n                active = getStatValue(now, \"activeConnection\");\n            }\n            catch(e){/*not supported*/}\n            if(!ip || !type || !localIP || active != \"true\")\n                continue;\n            var addressSaved = false;\n            for(var i = 0; i < PeerStats.transport.length; i++)\n            {\n                if(PeerStats.transport[i].ip == ip &&\n                    PeerStats.transport[i].type == type &&\n                    PeerStats.transport[i].localip == localIP)\n                {\n                    addressSaved = true;\n                }\n            }\n            if(addressSaved)\n                continue;\n            PeerStats.transport.push({localip: localIP, ip: ip, type: type});\n            continue;\n        }\n\n        if(now.type == \"candidatepair\")\n        {\n            if(now.state == \"succeeded\")\n                continue;\n\n            var local = this.currentStatsReport[now.localCandidateId];\n            var remote = this.currentStatsReport[now.remoteCandidateId];\n            PeerStats.transport.push({localip: local.ipAddress + \":\" + local.portNumber,\n                ip: remote.ipAddress + \":\" + remote.portNumber, type: local.transport});\n\n        }\n\n        if (now.type != 'ssrc' && now.type != \"outboundrtp\" &&\n            now.type != \"inboundrtp\") {\n            continue;\n        }\n\n        var before = this.baselineStatsReport[idx];\n        if (!before) {\n            console.warn(getStatValue(now, 'ssrc') + ' not enough data');\n            continue;\n        }\n\n        var ssrc = getStatValue(now, 'ssrc');\n        if(!ssrc)\n            continue;\n        var jid = ssrc2jid[ssrc];\n        if (!jid && (Date.now() - now.timestamp) < 3000) {\n            console.warn(\"No jid for ssrc: \" + ssrc);\n            continue;\n        }\n\n        var jidStats = this.jid2stats[jid];\n        if (!jidStats) {\n            jidStats = new PeerStats();\n            this.jid2stats[jid] = jidStats;\n        }\n\n\n        var isDownloadStream = true;\n        var key = 'packetsReceived';\n        if (!getStatValue(now, key))\n        {\n            isDownloadStream = false;\n            key = 'packetsSent';\n            if (!getStatValue(now, key))\n            {\n                console.warn(\"No packetsReceived nor packetSent stat found\");\n                continue;\n            }\n        }\n        var packetsNow = getStatValue(now, key);\n        if(!packetsNow || packetsNow < 0)\n            packetsNow = 0;\n\n        var packetsBefore = getStatValue(before, key);\n        if(!packetsBefore || packetsBefore < 0)\n            packetsBefore = 0;\n        var packetRate = packetsNow - packetsBefore;\n        if(!packetRate || packetRate < 0)\n            packetRate = 0;\n        var currentLoss = getStatValue(now, 'packetsLost');\n        if(!currentLoss || currentLoss < 0)\n            currentLoss = 0;\n        var previousLoss = getStatValue(before, 'packetsLost');\n        if(!previousLoss || previousLoss < 0)\n            previousLoss = 0;\n        var lossRate = currentLoss - previousLoss;\n        if(!lossRate || lossRate < 0)\n            lossRate = 0;\n        var packetsTotal = (packetRate + lossRate);\n\n        jidStats.setSsrcLoss(ssrc,\n            {\"packetsTotal\": packetsTotal,\n                \"packetsLost\": lossRate,\n                \"isDownloadStream\": isDownloadStream});\n\n\n        var bytesReceived = 0, bytesSent = 0;\n        if(getStatValue(now, \"bytesReceived\"))\n        {\n            bytesReceived = getStatValue(now, \"bytesReceived\") -\n                getStatValue(before, \"bytesReceived\");\n        }\n\n        if(getStatValue(now, \"bytesSent\"))\n        {\n            bytesSent = getStatValue(now, \"bytesSent\") -\n                getStatValue(before, \"bytesSent\");\n        }\n\n        var time = Math.round((now.timestamp - before.timestamp) / 1000);\n        if(bytesReceived <= 0 || time <= 0)\n        {\n            bytesReceived = 0;\n        }\n        else\n        {\n            bytesReceived = Math.round(((bytesReceived * 8) / time) / 1000);\n        }\n\n        if(bytesSent <= 0 || time <= 0)\n        {\n            bytesSent = 0;\n        }\n        else\n        {\n            bytesSent = Math.round(((bytesSent * 8) / time) / 1000);\n        }\n\n        jidStats.setSsrcBitrate(ssrc, {\n            \"download\": bytesReceived,\n            \"upload\": bytesSent});\n\n        var resolution = {height: null, width: null};\n        try {\n            if (getStatValue(now, \"googFrameHeightReceived\") &&\n                getStatValue(now, \"googFrameWidthReceived\")) {\n                resolution.height = getStatValue(now, \"googFrameHeightReceived\");\n                resolution.width = getStatValue(now, \"googFrameWidthReceived\");\n            }\n            else if (getStatValue(now, \"googFrameHeightSent\") &&\n                getStatValue(now, \"googFrameWidthSent\")) {\n                resolution.height = getStatValue(now, \"googFrameHeightSent\");\n                resolution.width = getStatValue(now, \"googFrameWidthSent\");\n            }\n        }\n        catch(e){/*not supported*/}\n\n        if(resolution.height && resolution.width)\n        {\n            jidStats.setSsrcResolution(ssrc, resolution);\n        }\n        else\n        {\n            jidStats.setSsrcResolution(ssrc, null);\n        }\n\n\n    }\n\n    var self = this;\n    // Jid stats\n    var totalPackets = {download: 0, upload: 0};\n    var lostPackets = {download: 0, upload: 0};\n    var bitrateDownload = 0;\n    var bitrateUpload = 0;\n    var resolutions = {};\n    Object.keys(this.jid2stats).forEach(\n        function (jid)\n        {\n            Object.keys(self.jid2stats[jid].ssrc2Loss).forEach(\n                function (ssrc)\n                {\n                    var type = \"upload\";\n                    if(self.jid2stats[jid].ssrc2Loss[ssrc].isDownloadStream)\n                        type = \"download\";\n                    totalPackets[type] +=\n                        self.jid2stats[jid].ssrc2Loss[ssrc].packetsTotal;\n                    lostPackets[type] +=\n                        self.jid2stats[jid].ssrc2Loss[ssrc].packetsLost;\n                }\n            );\n            Object.keys(self.jid2stats[jid].ssrc2bitrate).forEach(\n                function (ssrc) {\n                    bitrateDownload +=\n                        self.jid2stats[jid].ssrc2bitrate[ssrc].download;\n                    bitrateUpload +=\n                        self.jid2stats[jid].ssrc2bitrate[ssrc].upload;\n\n                    delete self.jid2stats[jid].ssrc2bitrate[ssrc];\n                }\n            );\n            resolutions[jid] = self.jid2stats[jid].ssrc2resolution;\n        }\n    );\n\n    PeerStats.bitrate = {\"upload\": bitrateUpload, \"download\": bitrateDownload};\n\n    PeerStats.packetLoss = {\n        total:\n            calculatePacketLoss(lostPackets.download + lostPackets.upload,\n                    totalPackets.download + totalPackets.upload),\n        download:\n            calculatePacketLoss(lostPackets.download, totalPackets.download),\n        upload:\n            calculatePacketLoss(lostPackets.upload, totalPackets.upload)\n    };\n    this.eventEmitter.emit(\"statistics.connectionstats\",\n        {\n            \"bitrate\": PeerStats.bitrate,\n            \"packetLoss\": PeerStats.packetLoss,\n            \"bandwidth\": PeerStats.bandwidth,\n            \"resolution\": resolutions,\n            \"transport\": PeerStats.transport\n        });\n    PeerStats.transport = [];\n\n};\n\n/**\n * Stats processing logic.\n */\nStatsCollector.prototype.processAudioLevelReport = function ()\n{\n    if (!this.baselineAudioLevelsReport)\n    {\n        return;\n    }\n\n    for (var idx in this.currentAudioLevelsReport)\n    {\n        var now = this.currentAudioLevelsReport[idx];\n\n        if (now.type != 'ssrc')\n        {\n            continue;\n        }\n\n        var before = this.baselineAudioLevelsReport[idx];\n        if (!before)\n        {\n            console.warn(getStatValue(now, 'ssrc') + ' not enough data');\n            continue;\n        }\n\n        var ssrc = getStatValue(now, 'ssrc');\n        var jid = ssrc2jid[ssrc];\n        if (!jid && (Date.now() - now.timestamp) < 3000)\n        {\n            console.warn(\"No jid for ssrc: \" + ssrc);\n            continue;\n        }\n\n        var jidStats = this.jid2stats[jid];\n        if (!jidStats)\n        {\n            jidStats = new PeerStats();\n            this.jid2stats[jid] = jidStats;\n        }\n\n        // Audio level\n        var audioLevel = null;\n\n        try {\n            audioLevel = getStatValue(now, 'audioInputLevel');\n            if (!audioLevel)\n                audioLevel = getStatValue(now, 'audioOutputLevel');\n        }\n        catch(e) {/*not supported*/\n            console.warn(\"Audio Levels are not available in the statistics.\");\n            clearInterval(this.audioLevelsIntervalId);\n            return;\n        }\n\n        if (audioLevel)\n        {\n            // TODO: can't find specs about what this value really is,\n            // but it seems to vary between 0 and around 32k.\n            audioLevel = audioLevel / 32767;\n            jidStats.setSsrcAudioLevel(ssrc, audioLevel);\n            if(jid != xmpp.myJid())\n                this.eventEmitter.emit(\"statistics.audioLevel\", jid, audioLevel);\n        }\n\n    }\n\n\n};","/**\n * Created by hristo on 8/4/14.\n */\nvar LocalStats = require(\"./LocalStatsCollector.js\");\nvar RTPStats = require(\"./RTPStatsCollector.js\");\nvar EventEmitter = require(\"events\");\n//These lines should be uncommented when require works in app.js\n//var StreamEventTypes = require(\"../../service/RTC/StreamEventTypes.js\");\n//var RTCBrowserType = require(\"../../service/RTC/RTCBrowserType\");\n//var XMPPEvents = require(\"../service/xmpp/XMPPEvents\");\n\nvar eventEmitter = new EventEmitter();\n\nvar localStats = null;\n\nvar rtpStats = null;\n\nfunction stopLocal()\n{\n    if(localStats)\n    {\n        localStats.stop();\n        localStats = null;\n    }\n}\n\nfunction stopRemote()\n{\n    if(rtpStats)\n    {\n        rtpStats.stop();\n        eventEmitter.emit(\"statistics.stop\");\n        rtpStats = null;\n    }\n}\n\nfunction startRemoteStats (peerconnection) {\n    if (config.enableRtpStats)\n    {\n        if(rtpStats)\n        {\n            rtpStats.stop();\n            rtpStats = null;\n        }\n\n        rtpStats = new RTPStats(peerconnection, 200, 2000, eventEmitter);\n        rtpStats.start();\n    }\n\n}\n\nfunction onStreamCreated(stream)\n{\n    if(stream.getOriginalStream().getAudioTracks().length === 0)\n        return;\n\n    localStats = new LocalStats(stream.getOriginalStream(), 100, statistics,\n        eventEmitter);\n    localStats.start();\n}\n\nfunction onDisposeConference(onUnload) {\n    stopRemote();\n    if(onUnload) {\n        stopLocal();\n        eventEmitter.removeAllListeners();\n    }\n}\n\n\nvar statistics =\n{\n    /**\n     * Indicates that this audio level is for local jid.\n     * @type {string}\n     */\n    LOCAL_JID: 'local',\n\n    addAudioLevelListener: function(listener)\n    {\n        eventEmitter.on(\"statistics.audioLevel\", listener);\n    },\n\n    removeAudioLevelListener: function(listener)\n    {\n        eventEmitter.removeListener(\"statistics.audioLevel\", listener);\n    },\n\n    addConnectionStatsListener: function(listener)\n    {\n        eventEmitter.on(\"statistics.connectionstats\", listener);\n    },\n\n    removeConnectionStatsListener: function(listener)\n    {\n        eventEmitter.removeListener(\"statistics.connectionstats\", listener);\n    },\n\n\n    addRemoteStatsStopListener: function(listener)\n    {\n        eventEmitter.on(\"statistics.stop\", listener);\n    },\n\n    removeRemoteStatsStopListener: function(listener)\n    {\n        eventEmitter.removeListener(\"statistics.stop\", listener);\n    },\n\n    stop: function () {\n        stopLocal();\n        stopRemote();\n        if(eventEmitter)\n        {\n            eventEmitter.removeAllListeners();\n        }\n    },\n\n    stopRemoteStatistics: function()\n    {\n        stopRemote();\n    },\n\n    onConferenceCreated: function (event) {\n        startRemoteStats(event.peerconnection);\n    },\n\n    start: function () {\n        this.addConnectionStatsListener(connectionquality.updateLocalStats);\n        this.addRemoteStatsStopListener(connectionquality.stopSendingStats);\n        RTC.addStreamListener(onStreamCreated,\n            StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);\n        xmpp.addListener(XMPPEvents.DISPOSE_CONFERENCE, onDisposeConference);\n    }\n\n};\n\n\n\n\nmodule.exports = statistics;","// Copyright Joyent, Inc. and other Node contributors.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a\n// copy of this software and associated documentation files (the\n// \"Software\"), to deal in the Software without restriction, including\n// without limitation the rights to use, copy, modify, merge, publish,\n// distribute, sublicense, and/or sell copies of the Software, and to permit\n// persons to whom the Software is furnished to do so, subject to the\n// following conditions:\n//\n// The above copyright notice and this permission notice shall be included\n// in all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\n// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\n// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN\n// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,\n// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\n// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE\n// USE OR OTHER DEALINGS IN THE SOFTWARE.\n\nfunction EventEmitter() {\n  this._events = this._events || {};\n  this._maxListeners = this._maxListeners || undefined;\n}\nmodule.exports = EventEmitter;\n\n// Backwards-compat with node 0.10.x\nEventEmitter.EventEmitter = EventEmitter;\n\nEventEmitter.prototype._events = undefined;\nEventEmitter.prototype._maxListeners = undefined;\n\n// By default EventEmitters will print a warning if more than 10 listeners are\n// added to it. This is a useful default which helps finding memory leaks.\nEventEmitter.defaultMaxListeners = 10;\n\n// Obviously not all Emitters should be limited to 10. This function allows\n// that to be increased. Set to zero for unlimited.\nEventEmitter.prototype.setMaxListeners = function(n) {\n  if (!isNumber(n) || n < 0 || isNaN(n))\n    throw TypeError('n must be a positive number');\n  this._maxListeners = n;\n  return this;\n};\n\nEventEmitter.prototype.emit = function(type) {\n  var er, handler, len, args, i, listeners;\n\n  if (!this._events)\n    this._events = {};\n\n  // If there is no 'error' event listener then throw.\n  if (type === 'error') {\n    if (!this._events.error ||\n        (isObject(this._events.error) && !this._events.error.length)) {\n      er = arguments[1];\n      if (er instanceof Error) {\n        throw er; // Unhandled 'error' event\n      } else {\n        throw TypeError('Uncaught, unspecified \"error\" event.');\n      }\n      return false;\n    }\n  }\n\n  handler = this._events[type];\n\n  if (isUndefined(handler))\n    return false;\n\n  if (isFunction(handler)) {\n    switch (arguments.length) {\n      // fast cases\n      case 1:\n        handler.call(this);\n        break;\n      case 2:\n        handler.call(this, arguments[1]);\n        break;\n      case 3:\n        handler.call(this, arguments[1], arguments[2]);\n        break;\n      // slower\n      default:\n        len = arguments.length;\n        args = new Array(len - 1);\n        for (i = 1; i < len; i++)\n          args[i - 1] = arguments[i];\n        handler.apply(this, args);\n    }\n  } else if (isObject(handler)) {\n    len = arguments.length;\n    args = new Array(len - 1);\n    for (i = 1; i < len; i++)\n      args[i - 1] = arguments[i];\n\n    listeners = handler.slice();\n    len = listeners.length;\n    for (i = 0; i < len; i++)\n      listeners[i].apply(this, args);\n  }\n\n  return true;\n};\n\nEventEmitter.prototype.addListener = function(type, listener) {\n  var m;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events)\n    this._events = {};\n\n  // To avoid recursion in the case that type === \"newListener\"! Before\n  // adding it to the listeners, first emit \"newListener\".\n  if (this._events.newListener)\n    this.emit('newListener', type,\n              isFunction(listener.listener) ?\n              listener.listener : listener);\n\n  if (!this._events[type])\n    // Optimize the case of one listener. Don't need the extra array object.\n    this._events[type] = listener;\n  else if (isObject(this._events[type]))\n    // If we've already got an array, just append.\n    this._events[type].push(listener);\n  else\n    // Adding the second element, need to change to array.\n    this._events[type] = [this._events[type], listener];\n\n  // Check for listener leak\n  if (isObject(this._events[type]) && !this._events[type].warned) {\n    var m;\n    if (!isUndefined(this._maxListeners)) {\n      m = this._maxListeners;\n    } else {\n      m = EventEmitter.defaultMaxListeners;\n    }\n\n    if (m && m > 0 && this._events[type].length > m) {\n      this._events[type].warned = true;\n      console.error('(node) warning: possible EventEmitter memory ' +\n                    'leak detected. %d listeners added. ' +\n                    'Use emitter.setMaxListeners() to increase limit.',\n                    this._events[type].length);\n      if (typeof console.trace === 'function') {\n        // not supported in IE 10\n        console.trace();\n      }\n    }\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.on = EventEmitter.prototype.addListener;\n\nEventEmitter.prototype.once = function(type, listener) {\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  var fired = false;\n\n  function g() {\n    this.removeListener(type, g);\n\n    if (!fired) {\n      fired = true;\n      listener.apply(this, arguments);\n    }\n  }\n\n  g.listener = listener;\n  this.on(type, g);\n\n  return this;\n};\n\n// emits a 'removeListener' event iff the listener was removed\nEventEmitter.prototype.removeListener = function(type, listener) {\n  var list, position, length, i;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events || !this._events[type])\n    return this;\n\n  list = this._events[type];\n  length = list.length;\n  position = -1;\n\n  if (list === listener ||\n      (isFunction(list.listener) && list.listener === listener)) {\n    delete this._events[type];\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n\n  } else if (isObject(list)) {\n    for (i = length; i-- > 0;) {\n      if (list[i] === listener ||\n          (list[i].listener && list[i].listener === listener)) {\n        position = i;\n        break;\n      }\n    }\n\n    if (position < 0)\n      return this;\n\n    if (list.length === 1) {\n      list.length = 0;\n      delete this._events[type];\n    } else {\n      list.splice(position, 1);\n    }\n\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.removeAllListeners = function(type) {\n  var key, listeners;\n\n  if (!this._events)\n    return this;\n\n  // not listening for removeListener, no need to emit\n  if (!this._events.removeListener) {\n    if (arguments.length === 0)\n      this._events = {};\n    else if (this._events[type])\n      delete this._events[type];\n    return this;\n  }\n\n  // emit removeListener for all listeners on all events\n  if (arguments.length === 0) {\n    for (key in this._events) {\n      if (key === 'removeListener') continue;\n      this.removeAllListeners(key);\n    }\n    this.removeAllListeners('removeListener');\n    this._events = {};\n    return this;\n  }\n\n  listeners = this._events[type];\n\n  if (isFunction(listeners)) {\n    this.removeListener(type, listeners);\n  } else {\n    // LIFO order\n    while (listeners.length)\n      this.removeListener(type, listeners[listeners.length - 1]);\n  }\n  delete this._events[type];\n\n  return this;\n};\n\nEventEmitter.prototype.listeners = function(type) {\n  var ret;\n  if (!this._events || !this._events[type])\n    ret = [];\n  else if (isFunction(this._events[type]))\n    ret = [this._events[type]];\n  else\n    ret = this._events[type].slice();\n  return ret;\n};\n\nEventEmitter.listenerCount = function(emitter, type) {\n  var ret;\n  if (!emitter._events || !emitter._events[type])\n    ret = 0;\n  else if (isFunction(emitter._events[type]))\n    ret = 1;\n  else\n    ret = emitter._events[type].length;\n  return ret;\n};\n\nfunction isFunction(arg) {\n  return typeof arg === 'function';\n}\n\nfunction isNumber(arg) {\n  return typeof arg === 'number';\n}\n\nfunction isObject(arg) {\n  return typeof arg === 'object' && arg !== null;\n}\n\nfunction isUndefined(arg) {\n  return arg === void 0;\n}\n"]}