@twilio/voice-sdk
Version:
Twilio's JavaScript Voice SDK
403 lines • 28.8 kB
JavaScript
/**
* @packageDocumentation
* @module Voice
* @internalapi
*/
// @ts-nocheck
/**
* This file was imported from another project. If making changes to this file, please don't
* make them here. Make them on the linked repo below, then copy back:
* https://code.hq.twilio.com/client/MockRTCStatsReport
*/
// The legacy max volume, which is the positive half of a signed short integer.
const OLD_MAX_VOLUME = 32767;
const NativeRTCStatsReport = typeof window !== 'undefined'
? window.RTCStatsReport : undefined;
/**
* Create a MockRTCStatsReport wrapper around a Map of RTCStats objects. If RTCStatsReport is available
* natively, it will be inherited so that instanceof checks pass.
* @constructor
* @extends RTCStatsReport
* @param {Map<string, RTCStats>} statsMap - A Map of RTCStats objects to wrap
* with a MockRTCStatsReport object.
*/
function MockRTCStatsReport(statsMap) {
if (!(this instanceof MockRTCStatsReport)) {
return new MockRTCStatsReport(statsMap);
}
const self = this;
Object.defineProperties(this, {
_map: { value: statsMap },
size: {
enumerable: true,
get() {
return self._map.size;
},
},
});
this[Symbol.iterator] = statsMap[Symbol.iterator];
}
// If RTCStatsReport is available natively, inherit it. Keep our constructor.
if (NativeRTCStatsReport) {
MockRTCStatsReport.prototype = Object.create(NativeRTCStatsReport.prototype);
MockRTCStatsReport.prototype.constructor = MockRTCStatsReport;
}
// Map the Map-like read methods to the underlying Map
['entries', 'forEach', 'get', 'has', 'keys', 'values'].forEach(key => {
MockRTCStatsReport.prototype[key] = function (...args) {
return this._map[key](...args);
};
});
/**
* Convert an array of RTCStats objects into a mock RTCStatsReport object.
* @param {Array<RTCStats>}
* @return {MockRTCStatsReport}
*/
MockRTCStatsReport.fromArray = function fromArray(array) {
return new MockRTCStatsReport(array.reduce((map, rtcStats) => {
map.set(rtcStats.id, rtcStats);
return map;
}, new Map()));
};
/**
* Convert a legacy RTCStatsResponse object into a mock RTCStatsReport object.
* @param {RTCStatsResponse} statsResponse - An RTCStatsResponse object returned by the
* legacy getStats(callback) method in Chrome.
* @return {MockRTCStatsReport} A mock RTCStatsReport object.
*/
MockRTCStatsReport.fromRTCStatsResponse = function fromRTCStatsResponse(statsResponse) {
let activeCandidatePairId;
const transportIds = new Map();
const statsMap = statsResponse.result().reduce((map, report) => {
const id = report.id;
switch (report.type) {
case 'googCertificate':
map.set(id, createRTCCertificateStats(report));
break;
case 'datachannel':
map.set(id, createRTCDataChannelStats(report));
break;
case 'googCandidatePair':
if (getBoolean(report, 'googActiveConnection')) {
activeCandidatePairId = id;
}
map.set(id, createRTCIceCandidatePairStats(report));
break;
case 'localcandidate':
map.set(id, createRTCIceCandidateStats(report, false));
break;
case 'remotecandidate':
map.set(id, createRTCIceCandidateStats(report, true));
break;
case 'ssrc':
if (isPresent(report, 'packetsReceived')) {
map.set(`rtp-${id}`, createRTCInboundRTPStreamStats(report));
}
else {
map.set(`rtp-${id}`, createRTCOutboundRTPStreamStats(report));
}
map.set(`track-${id}`, createRTCMediaStreamTrackStats(report));
map.set(`codec-${id}`, createRTCCodecStats(report));
break;
case 'googComponent':
const transportReport = createRTCTransportStats(report);
transportIds.set(transportReport.selectedCandidatePairId, id);
map.set(id, createRTCTransportStats(report));
break;
}
return map;
}, new Map());
if (activeCandidatePairId) {
const activeTransportId = transportIds.get(activeCandidatePairId);
if (activeTransportId) {
statsMap.get(activeTransportId).dtlsState = 'connected';
}
}
return new MockRTCStatsReport(statsMap);
};
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCTransportStats}
*/
function createRTCTransportStats(report) {
return {
bytesReceived: undefined,
bytesSent: undefined,
dtlsState: undefined,
id: report.id,
localCertificateId: report.stat('localCertificateId'),
remoteCertificateId: report.stat('remoteCertificateId'),
rtcpTransportStatsId: undefined,
selectedCandidatePairId: report.stat('selectedCandidatePairId'),
timestamp: Date.parse(report.timestamp),
type: 'transport',
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCCodecStats}
*/
function createRTCCodecStats(report) {
return {
channels: undefined,
clockRate: undefined,
id: report.id,
implementation: undefined,
mimeType: `${report.stat('mediaType')}/${report.stat('googCodecName')}`,
payloadType: undefined,
sdpFmtpLine: undefined,
timestamp: Date.parse(report.timestamp),
type: 'codec',
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCMediaStreamTrackStats}
*/
function createRTCMediaStreamTrackStats(report) {
return {
audioLevel: isPresent(report, 'audioOutputLevel')
? getInt(report, 'audioOutputLevel') / OLD_MAX_VOLUME
: (getInt(report, 'audioInputLevel') || 0) / OLD_MAX_VOLUME,
detached: undefined,
echoReturnLoss: getFloat(report, 'googEchoCancellationReturnLoss'),
echoReturnLossEnhancement: getFloat(report, 'googEchoCancellationReturnLossEnhancement'),
ended: undefined,
frameHeight: isPresent(report, 'googFrameHeightReceived')
? getInt(report, 'googFrameHeightReceived')
: getInt(report, 'googFrameHeightSent'),
frameWidth: isPresent(report, 'googFrameWidthReceived')
? getInt(report, 'googFrameWidthReceived')
: getInt(report, 'googFrameWidthSent'),
framesCorrupted: undefined,
framesDecoded: getInt(report, 'framesDecoded'),
framesDropped: undefined,
framesPerSecond: undefined,
framesReceived: undefined,
framesSent: getInt(report, 'framesEncoded'),
fullFramesLost: undefined,
id: report.id,
kind: report.stat('mediaType'),
partialFramesLost: undefined,
remoteSource: undefined,
ssrcIds: undefined,
timestamp: Date.parse(report.timestamp),
trackIdentifier: report.stat('googTrackId'),
type: 'track',
};
}
/**
* @param {RTCLegacyStatsReport} report
* @param {boolean} isInbound - Whether to create an inbound stats object, or outbound.
* @returns {RTCRTPStreamStats}
*/
function createRTCRTPStreamStats(report, isInbound) {
return {
associateStatsId: undefined,
codecId: `codec-${report.id}`,
firCount: isInbound
? getInt(report, 'googFirsSent')
: undefined,
id: report.id,
isRemote: undefined,
mediaType: report.stat('mediaType'),
nackCount: isInbound
? getInt(report, 'googNacksSent')
: getInt(report, 'googNacksReceived'),
pliCount: isInbound
? getInt(report, 'googPlisSent')
: getInt(report, 'googPlisReceived'),
qpSum: getInt(report, 'qpSum'),
sliCount: undefined,
ssrc: report.stat('ssrc'),
timestamp: Date.parse(report.timestamp),
trackId: `track-${report.id}`,
transportId: report.stat('transportId'),
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCInboundRTPStreamStats}
*/
function createRTCInboundRTPStreamStats(report) {
const rtp = createRTCRTPStreamStats(report, true);
Object.assign(rtp, {
burstDiscardCount: undefined,
burstDiscardRate: undefined,
burstLossCount: undefined,
burstLossRate: undefined,
burstPacketsDiscarded: undefined,
burstPacketsLost: undefined,
bytesReceived: getInt(report, 'bytesReceived'),
fractionLost: undefined,
framesDecoded: getInt(report, 'framesDecoded'),
gapDiscardRate: undefined,
gapLossRate: undefined,
jitter: convertMsToSeconds(report.stat('googJitterReceived')),
packetsDiscarded: undefined,
packetsLost: getInt(report, 'packetsLost'),
packetsReceived: getInt(report, 'packetsReceived'),
packetsRepaired: undefined,
roundTripTime: convertMsToSeconds(report.stat('googRtt')),
type: 'inbound-rtp',
});
return rtp;
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCOutboundRTPStreamStats}
*/
function createRTCOutboundRTPStreamStats(report) {
const rtp = createRTCRTPStreamStats(report, false);
Object.assign(rtp, {
bytesSent: getInt(report, 'bytesSent'),
framesEncoded: getInt(report, 'framesEncoded'),
packetsSent: getInt(report, 'packetsSent'),
remoteTimestamp: undefined,
targetBitrate: undefined,
type: 'outbound-rtp',
});
return rtp;
}
/**
* @param {RTCLegacyStatsReport} report
* @param {boolean} isRemote - Whether to create for a remote candidate, or local candidate.
* @returns {RTCIceCandidateStats}
*/
function createRTCIceCandidateStats(report, isRemote) {
return {
candidateType: translateCandidateType(report.stat('candidateType')),
deleted: undefined,
id: report.id,
ip: report.stat('ipAddress'),
isRemote,
port: getInt(report, 'portNumber'),
priority: getFloat(report, 'priority'),
protocol: report.stat('transport'),
relayProtocol: undefined,
timestamp: Date.parse(report.timestamp),
transportId: undefined,
type: isRemote
? 'remote-candidate'
: 'local-candidate',
url: undefined,
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCIceCandidatePairStats}
*/
function createRTCIceCandidatePairStats(report) {
return {
availableIncomingBitrate: undefined,
availableOutgoingBitrate: undefined,
bytesReceived: getInt(report, 'bytesReceived'),
bytesSent: getInt(report, 'bytesSent'),
consentRequestsSent: getInt(report, 'consentRequestsSent'),
currentRoundTripTime: convertMsToSeconds(report.stat('googRtt')),
id: report.id,
lastPacketReceivedTimestamp: undefined,
lastPacketSentTimestamp: undefined,
localCandidateId: report.stat('localCandidateId'),
nominated: undefined,
priority: undefined,
readable: undefined,
remoteCandidateId: report.stat('remoteCandidateId'),
requestsReceived: getInt(report, 'requestsReceived'),
requestsSent: getInt(report, 'requestsSent'),
responsesReceived: getInt(report, 'responsesReceived'),
responsesSent: getInt(report, 'responsesSent'),
retransmissionsReceived: undefined,
retransmissionsSent: undefined,
state: undefined,
timestamp: Date.parse(report.timestamp),
totalRoundTripTime: undefined,
transportId: report.stat('googChannelId'),
type: 'candidate-pair',
writable: getBoolean(report, 'googWritable'),
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCIceCertificateStats}
*/
function createRTCCertificateStats(report) {
return {
base64Certificate: report.stat('googDerBase64'),
fingerprint: report.stat('googFingerprint'),
fingerprintAlgorithm: report.stat('googFingerprintAlgorithm'),
id: report.id,
issuerCertificateId: report.stat('googIssuerId'),
timestamp: Date.parse(report.timestamp),
type: 'certificate',
};
}
/**
* @param {RTCLegacyStatsReport} report
* @returns {RTCDataChannelStats}
*/
function createRTCDataChannelStats(report) {
return {
bytesReceived: undefined,
bytesSent: undefined,
datachannelid: report.stat('datachannelid'),
id: report.id,
label: report.stat('label'),
messagesReceived: undefined,
messagesSent: undefined,
protocol: report.stat('protocol'),
state: report.stat('state'),
timestamp: Date.parse(report.timestamp),
transportId: report.stat('transportId'),
type: 'data-channel',
};
}
/**
* @param {number} inMs - A time in milliseconds
* @returns {number} The time in seconds
*/
function convertMsToSeconds(inMs) {
return isNaN(inMs) || inMs === ''
? undefined
: parseInt(inMs, 10) / 1000;
}
/**
* @param {string} type - A type in the legacy format
* @returns {string} The type adjusted to new standards for known naming changes
*/
function translateCandidateType(type) {
switch (type) {
case 'peerreflexive':
return 'prflx';
case 'serverreflexive':
return 'srflx';
case 'host':
case 'relay':
default:
return type;
}
}
function getInt(report, statName) {
const stat = report.stat(statName);
return isPresent(report, statName)
? parseInt(stat, 10)
: undefined;
}
function getFloat(report, statName) {
const stat = report.stat(statName);
return isPresent(report, statName)
? parseFloat(stat)
: undefined;
}
function getBoolean(report, statName) {
const stat = report.stat(statName);
return isPresent(report, statName)
? (stat === 'true' || stat === true)
: undefined;
}
function isPresent(report, statName) {
const stat = report.stat(statName);
return typeof stat !== 'undefined' && stat !== '';
}
export default MockRTCStatsReport;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9ja3J0Y3N0YXRzcmVwb3J0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL3R3aWxpby9ydGMvbW9ja3J0Y3N0YXRzcmVwb3J0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFDSCxjQUFjO0FBRWQ7Ozs7R0FJRztBQUVILCtFQUErRTtBQUMvRSxNQUFNLGNBQWMsR0FBRyxLQUFLLENBQUM7QUFFN0IsTUFBTSxvQkFBb0IsR0FBRyxPQUFPLE1BQU0sS0FBSyxXQUFXO0lBQ3hELENBQUMsQ0FBQyxNQUFNLENBQUMsY0FBYyxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUM7QUFFdEM7Ozs7Ozs7R0FPRztBQUNILFNBQVMsa0JBQWtCLENBQUMsUUFBUTtJQUNsQyxJQUFJLENBQUMsQ0FBQyxJQUFJLFlBQVksa0JBQWtCLENBQUMsRUFBRTtRQUN6QyxPQUFPLElBQUksa0JBQWtCLENBQUMsUUFBUSxDQUFDLENBQUM7S0FDekM7SUFFRCxNQUFNLElBQUksR0FBRyxJQUFJLENBQUM7SUFDbEIsTUFBTSxDQUFDLGdCQUFnQixDQUFDLElBQUksRUFBRTtRQUM1QixJQUFJLEVBQUUsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFO1FBQ3pCLElBQUksRUFBRTtZQUNKLFVBQVUsRUFBRSxJQUFJO1lBQ2hCLEdBQUc7Z0JBQ0QsT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztZQUN4QixDQUFDO1NBQ0Y7S0FDRixDQUFDLENBQUM7SUFFSCxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxHQUFHLFFBQVEsQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLENBQUM7QUFDcEQsQ0FBQztBQUVELDZFQUE2RTtBQUM3RSxJQUFJLG9CQUFvQixFQUFFO0lBQ3hCLGtCQUFrQixDQUFDLFNBQVMsR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLG9CQUFvQixDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQzdFLGtCQUFrQixD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