@webex/web-client-media-engine 3.36.0 → 3.37.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/index.js +107 -100
- package/dist/cjs/index.js.map +1 -1
- package/dist/esm/index.js +107 -100
- package/dist/esm/index.js.map +1 -1
- package/dist/types/index.d.ts +9 -11
- package/package.json +1 -1
package/dist/esm/index.js
CHANGED
|
@@ -8206,6 +8206,24 @@ function getRecommendedMaxBitrateForFrameSize(requestedMaxFs) {
|
|
|
8206
8206
|
.sort((a, b) => b - a)
|
|
8207
8207
|
.find((h) => requestedMaxFs >= h);
|
|
8208
8208
|
return maxFrameSizeToMaxBitrateMap.get(expectedHeight);
|
|
8209
|
+
}
|
|
8210
|
+
function setCodecParameters(allCodecParameters, codecMimeType, newParameters) {
|
|
8211
|
+
if (!allCodecParameters.has(codecMimeType)) {
|
|
8212
|
+
allCodecParameters.set(codecMimeType, {});
|
|
8213
|
+
}
|
|
8214
|
+
const currentParameters = allCodecParameters.get(codecMimeType);
|
|
8215
|
+
Object.entries(newParameters).forEach(([param, value]) => {
|
|
8216
|
+
currentParameters[param] = value;
|
|
8217
|
+
});
|
|
8218
|
+
}
|
|
8219
|
+
function markCodecParametersForDeletion(allCodecParameters, codecMimeTypeToUpdate, codecParameters) {
|
|
8220
|
+
if (!allCodecParameters.has(codecMimeTypeToUpdate)) {
|
|
8221
|
+
allCodecParameters.set(codecMimeTypeToUpdate, {});
|
|
8222
|
+
}
|
|
8223
|
+
const currentParameters = allCodecParameters.get(codecMimeTypeToUpdate);
|
|
8224
|
+
codecParameters.forEach((param) => {
|
|
8225
|
+
currentParameters[param] = undefined;
|
|
8226
|
+
});
|
|
8209
8227
|
}
|
|
8210
8228
|
|
|
8211
8229
|
/******************************************************************************
|
|
@@ -10439,16 +10457,17 @@ function addVlaExtension(mediaDescription) {
|
|
|
10439
10457
|
mediaDescription.addExtension({ uri: vlaExtensionUri });
|
|
10440
10458
|
}
|
|
10441
10459
|
}
|
|
10442
|
-
function
|
|
10443
|
-
|
|
10460
|
+
function setFormatParameters(mediaDescription, codecMimeType, paramsMap) {
|
|
10461
|
+
const codecName = codecMimeType.split('/')[1].toLowerCase();
|
|
10462
|
+
Object.entries(paramsMap).forEach(([param, value]) => {
|
|
10444
10463
|
[...mediaDescription.codecs.values()]
|
|
10445
|
-
.filter((ci) => ci.name &&
|
|
10464
|
+
.filter((ci) => ci.name && codecName === ci.name.toLowerCase())
|
|
10446
10465
|
.forEach((ci) => {
|
|
10447
|
-
if (value ===
|
|
10466
|
+
if (value === undefined) {
|
|
10448
10467
|
ci.fmtParams.delete(param);
|
|
10449
10468
|
}
|
|
10450
10469
|
else {
|
|
10451
|
-
ci.fmtParams.set(param,
|
|
10470
|
+
ci.fmtParams.set(param, value);
|
|
10452
10471
|
}
|
|
10453
10472
|
});
|
|
10454
10473
|
});
|
|
@@ -10563,7 +10582,9 @@ class EgressSdpMunger {
|
|
|
10563
10582
|
if (options.simulcastEnabled) {
|
|
10564
10583
|
mediaDescription.addLine(new SsrcGroupLine('SIM', this.streamIds.map((streamId) => streamId.ssrc)));
|
|
10565
10584
|
}
|
|
10566
|
-
|
|
10585
|
+
this.customCodecParameters.forEach((codecParams, codecMimeType) => {
|
|
10586
|
+
setFormatParameters(mediaDescription, codecMimeType, codecParams);
|
|
10587
|
+
});
|
|
10567
10588
|
if (options.twccDisabled) {
|
|
10568
10589
|
disableTwcc(mediaDescription);
|
|
10569
10590
|
}
|
|
@@ -10593,7 +10614,9 @@ class EgressSdpMunger {
|
|
|
10593
10614
|
ci.fmtParams.set('usedtx', mungeOptions.dtxDisabled ? '0' : '1');
|
|
10594
10615
|
});
|
|
10595
10616
|
}
|
|
10596
|
-
|
|
10617
|
+
this.customCodecParameters.forEach((codecParams, codecMimeType) => {
|
|
10618
|
+
setFormatParameters(mediaDescription, codecMimeType, codecParams);
|
|
10619
|
+
});
|
|
10597
10620
|
}
|
|
10598
10621
|
getSenderIds() {
|
|
10599
10622
|
return this.streamIds;
|
|
@@ -10601,15 +10624,11 @@ class EgressSdpMunger {
|
|
|
10601
10624
|
getEncodingIndexForStreamId(streamId) {
|
|
10602
10625
|
return this.streamIds.findIndex((currStreamId) => areStreamIdsEqual(currStreamId, streamId));
|
|
10603
10626
|
}
|
|
10604
|
-
|
|
10605
|
-
|
|
10606
|
-
this.customCodecParameters.set(param, value);
|
|
10607
|
-
});
|
|
10627
|
+
setCustomCodecParameters(codecMimeType, parameters) {
|
|
10628
|
+
setCodecParameters(this.customCodecParameters, codecMimeType, parameters);
|
|
10608
10629
|
}
|
|
10609
|
-
|
|
10610
|
-
|
|
10611
|
-
this.customCodecParameters.set(param, null);
|
|
10612
|
-
});
|
|
10630
|
+
markCustomCodecParametersForDeletion(codecMimeType, parameters) {
|
|
10631
|
+
markCodecParametersForDeletion(this.customCodecParameters, codecMimeType, parameters);
|
|
10613
10632
|
}
|
|
10614
10633
|
}
|
|
10615
10634
|
|
|
@@ -11095,7 +11114,6 @@ class EventEmitter$2 extends events$1.exports.EventEmitter {
|
|
|
11095
11114
|
class IngressSdpMunger {
|
|
11096
11115
|
constructor() {
|
|
11097
11116
|
this.customCodecParameters = new Map();
|
|
11098
|
-
this.customRtxCodecParameters = new Map();
|
|
11099
11117
|
this.ssrc = generateSsrc();
|
|
11100
11118
|
}
|
|
11101
11119
|
getReceiverId() {
|
|
@@ -11107,8 +11125,9 @@ class IngressSdpMunger {
|
|
|
11107
11125
|
logErrorAndThrow(WcmeErrorType.SDP_MUNGE_MISSING_CODECS, `No codecs present in m-line with MID ${mediaDescription.mid} after filtering.`);
|
|
11108
11126
|
}
|
|
11109
11127
|
removeMidRidExtensions(mediaDescription);
|
|
11110
|
-
|
|
11111
|
-
|
|
11128
|
+
this.customCodecParameters.forEach((codecParams, codecMimeType) => {
|
|
11129
|
+
setFormatParameters(mediaDescription, codecMimeType, codecParams);
|
|
11130
|
+
});
|
|
11112
11131
|
if (options.twccDisabled) {
|
|
11113
11132
|
disableTwcc(mediaDescription);
|
|
11114
11133
|
}
|
|
@@ -11129,17 +11148,16 @@ class IngressSdpMunger {
|
|
|
11129
11148
|
if (retainCandidatesByTransportType(mediaDescription, ['udp', 'tcp'])) {
|
|
11130
11149
|
logger.log(`Some unsupported remote candidates have been removed from mid ${mediaDescription.mid}`);
|
|
11131
11150
|
}
|
|
11132
|
-
|
|
11151
|
+
const rtxParams = this.customCodecParameters.get(MediaCodecMimeType.RTX);
|
|
11152
|
+
if (rtxParams) {
|
|
11153
|
+
setFormatParameters(mediaDescription, MediaCodecMimeType.RTX, rtxParams);
|
|
11154
|
+
}
|
|
11133
11155
|
}
|
|
11134
|
-
|
|
11135
|
-
|
|
11136
|
-
this.customCodecParameters.set(param, value);
|
|
11137
|
-
});
|
|
11156
|
+
setCustomCodecParameters(codecMimeType, parameters) {
|
|
11157
|
+
setCodecParameters(this.customCodecParameters, codecMimeType, parameters);
|
|
11138
11158
|
}
|
|
11139
|
-
|
|
11140
|
-
|
|
11141
|
-
this.customRtxCodecParameters.set(param, value);
|
|
11142
|
-
});
|
|
11159
|
+
markCustomCodecParametersForDeletion(codecMimeType, parameters) {
|
|
11160
|
+
markCodecParametersForDeletion(this.customCodecParameters, codecMimeType, parameters);
|
|
11143
11161
|
}
|
|
11144
11162
|
reset() {
|
|
11145
11163
|
this.ssrc = generateSsrc();
|
|
@@ -11452,11 +11470,8 @@ class ReceiveOnlyTransceiver extends Transceiver {
|
|
|
11452
11470
|
this.metadata.lastActiveSpeakerUpdateTimestamp = getCurrentTimestamp();
|
|
11453
11471
|
}
|
|
11454
11472
|
}
|
|
11455
|
-
|
|
11456
|
-
this.munger.
|
|
11457
|
-
}
|
|
11458
|
-
setRtxCodecParameters(parameters) {
|
|
11459
|
-
this.munger.setRtxCodecParameters(parameters);
|
|
11473
|
+
setCustomCodecParameters(codecMimeType, parameters) {
|
|
11474
|
+
this.munger.setCustomCodecParameters(codecMimeType, parameters);
|
|
11460
11475
|
}
|
|
11461
11476
|
}
|
|
11462
11477
|
ReceiveOnlyTransceiver.rid = '1';
|
|
@@ -15432,12 +15447,12 @@ class SendOnlyTransceiver extends Transceiver {
|
|
|
15432
15447
|
resetSdpMunger() {
|
|
15433
15448
|
this.munger.reset();
|
|
15434
15449
|
}
|
|
15435
|
-
|
|
15436
|
-
this.munger.
|
|
15450
|
+
setCustomCodecParameters(codecMimeType, codecParams) {
|
|
15451
|
+
this.munger.setCustomCodecParameters(codecMimeType, codecParams);
|
|
15437
15452
|
this.negotiationNeeded.emit(OfferAnswerType.LocalOnly);
|
|
15438
15453
|
}
|
|
15439
|
-
|
|
15440
|
-
this.munger.
|
|
15454
|
+
markCustomCodecParametersForDeletion(codecMimeType, parameters) {
|
|
15455
|
+
this.munger.markCustomCodecParametersForDeletion(codecMimeType, parameters);
|
|
15441
15456
|
this.negotiationNeeded.emit(OfferAnswerType.LocalOnly);
|
|
15442
15457
|
}
|
|
15443
15458
|
setSourceStateOverride(state) {
|
|
@@ -15498,12 +15513,26 @@ class SendSlot {
|
|
|
15498
15513
|
}
|
|
15499
15514
|
setCodecParameters(parameters) {
|
|
15500
15515
|
return __awaiter(this, void 0, void 0, function* () {
|
|
15501
|
-
|
|
15516
|
+
[MediaCodecMimeType.H264, MediaCodecMimeType.OPUS].forEach((codecMimeType) => {
|
|
15517
|
+
this.setCustomCodecParameters(codecMimeType, parameters);
|
|
15518
|
+
});
|
|
15502
15519
|
});
|
|
15503
15520
|
}
|
|
15504
15521
|
deleteCodecParameters(parameters) {
|
|
15505
15522
|
return __awaiter(this, void 0, void 0, function* () {
|
|
15506
|
-
|
|
15523
|
+
[MediaCodecMimeType.H264, MediaCodecMimeType.OPUS].forEach((codecMimeType) => {
|
|
15524
|
+
this.markCustomCodecParametersForDeletion(codecMimeType, parameters);
|
|
15525
|
+
});
|
|
15526
|
+
});
|
|
15527
|
+
}
|
|
15528
|
+
setCustomCodecParameters(codecMimeType, parameters) {
|
|
15529
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
15530
|
+
this.sendTransceiver.setCustomCodecParameters(codecMimeType, parameters);
|
|
15531
|
+
});
|
|
15532
|
+
}
|
|
15533
|
+
markCustomCodecParametersForDeletion(codecMimeType, parameters) {
|
|
15534
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
15535
|
+
this.sendTransceiver.markCustomCodecParametersForDeletion(codecMimeType, parameters);
|
|
15507
15536
|
});
|
|
15508
15537
|
}
|
|
15509
15538
|
setSourceStateOverride(state) {
|
|
@@ -15587,31 +15616,6 @@ const organizeTransceiverStats = (sendTransceivers, recvTransceivers, peerConnec
|
|
|
15587
15616
|
return result;
|
|
15588
15617
|
});
|
|
15589
15618
|
|
|
15590
|
-
var WorkerClass = null;
|
|
15591
|
-
|
|
15592
|
-
try {
|
|
15593
|
-
var WorkerThreads =
|
|
15594
|
-
typeof module !== 'undefined' && typeof module.require === 'function' && module.require('worker_threads') ||
|
|
15595
|
-
typeof __non_webpack_require__ === 'function' && __non_webpack_require__('worker_threads') ||
|
|
15596
|
-
typeof require === 'function' && require('worker_threads');
|
|
15597
|
-
WorkerClass = WorkerThreads.Worker;
|
|
15598
|
-
} catch(e) {} // eslint-disable-line
|
|
15599
|
-
|
|
15600
|
-
function decodeBase64$1(base64, enableUnicode) {
|
|
15601
|
-
return Buffer.from(base64, 'base64').toString(enableUnicode ? 'utf16' : 'utf8');
|
|
15602
|
-
}
|
|
15603
|
-
|
|
15604
|
-
function createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg) {
|
|
15605
|
-
var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
|
|
15606
|
-
var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
|
|
15607
|
-
var source = decodeBase64$1(base64, enableUnicode);
|
|
15608
|
-
var start = source.indexOf('\n', 10) + 1;
|
|
15609
|
-
var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
|
|
15610
|
-
return function WorkerFactory(options) {
|
|
15611
|
-
return new WorkerClass(body, Object.assign({}, options, { eval: true }));
|
|
15612
|
-
};
|
|
15613
|
-
}
|
|
15614
|
-
|
|
15615
15619
|
function decodeBase64(base64, enableUnicode) {
|
|
15616
15620
|
var binaryString = atob(base64);
|
|
15617
15621
|
if (enableUnicode) {
|
|
@@ -15635,7 +15639,7 @@ function createURL(base64, sourcemapArg, enableUnicodeArg) {
|
|
|
15635
15639
|
return URL.createObjectURL(blob);
|
|
15636
15640
|
}
|
|
15637
15641
|
|
|
15638
|
-
function createBase64WorkerFactory
|
|
15642
|
+
function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
|
|
15639
15643
|
var url;
|
|
15640
15644
|
return function WorkerFactory(options) {
|
|
15641
15645
|
url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
|
|
@@ -15643,19 +15647,6 @@ function createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg) {
|
|
|
15643
15647
|
};
|
|
15644
15648
|
}
|
|
15645
15649
|
|
|
15646
|
-
var kIsNodeJS = Object.prototype.toString.call(typeof process !== 'undefined' ? process : 0) === '[object process]';
|
|
15647
|
-
|
|
15648
|
-
function isNodeJS() {
|
|
15649
|
-
return kIsNodeJS;
|
|
15650
|
-
}
|
|
15651
|
-
|
|
15652
|
-
function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
|
|
15653
|
-
if (isNodeJS()) {
|
|
15654
|
-
return createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg);
|
|
15655
|
-
}
|
|
15656
|
-
return createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg);
|
|
15657
|
-
}
|
|
15658
|
-
|
|
15659
15650
|
var WorkerFactory = /*#__PURE__*/createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
    'use strict';

    var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {};

    var logger = {exports: {}};

    /*!
     * js-logger - http://github.com/jonnyreeves/js-logger
     * Jonny Reeves, http://jonnyreeves.co.uk/
     * js-logger may be freely distributed under the MIT license.
     */

    (function (module) {
    (function (global) {

    	// Top level module for the global, static logger instance.
    	var Logger = { };

    	// For those that are at home that are keeping score.
    	Logger.VERSION = "1.6.1";

    	// Function which handles all incoming log messages.
    	var logHandler;

    	// Map of ContextualLogger instances by name; used by Logger.get() to return the same named instance.
    	var contextualLoggersByNameMap = {};

    	// Polyfill for ES5's Function.bind.
    	var bind = function(scope, func) {
    		return function() {
    			return func.apply(scope, arguments);
    		};
    	};

    	// Super exciting object merger-matron 9000 adding another 100 bytes to your download.
    	var merge = function () {
    		var args = arguments, target = args[0], key, i;
    		for (i = 1; i < args.length; i++) {
    			for (key in args[i]) {
    				if (!(key in target) && args[i].hasOwnProperty(key)) {
    					target[key] = args[i][key];
    				}
    			}
    		}
    		return target;
    	};

    	// Helper to define a logging level object; helps with optimisation.
    	var defineLogLevel = function(value, name) {
    		return { value: value, name: name };
    	};

    	// Predefined logging levels.
    	Logger.TRACE = defineLogLevel(1, 'TRACE');
    	Logger.DEBUG = defineLogLevel(2, 'DEBUG');
    	Logger.INFO = defineLogLevel(3, 'INFO');
    	Logger.TIME = defineLogLevel(4, 'TIME');
    	Logger.WARN = defineLogLevel(5, 'WARN');
    	Logger.ERROR = defineLogLevel(8, 'ERROR');
    	Logger.OFF = defineLogLevel(99, 'OFF');

    	// Inner class which performs the bulk of the work; ContextualLogger instances can be configured independently
    	// of each other.
    	var ContextualLogger = function(defaultContext) {
    		this.context = defaultContext;
    		this.setLevel(defaultContext.filterLevel);
    		this.log = this.info;  // Convenience alias.
    	};

    	ContextualLogger.prototype = {
    		// Changes the current logging level for the logging instance.
    		setLevel: function (newLevel) {
    			// Ensure the supplied Level object looks valid.
    			if (newLevel && "value" in newLevel) {
    				this.context.filterLevel = newLevel;
    			}
    		},
    		
    		// Gets the current logging level for the logging instance
    		getLevel: function () {
    			return this.context.filterLevel;
    		},

    		// Is the logger configured to output messages at the supplied level?
    		enabledFor: function (lvl) {
    			var filterLevel = this.context.filterLevel;
    			return lvl.value >= filterLevel.value;
    		},

    		trace: function () {
    			this.invoke(Logger.TRACE, arguments);
    		},

    		debug: function () {
    			this.invoke(Logger.DEBUG, arguments);
    		},

    		info: function () {
    			this.invoke(Logger.INFO, arguments);
    		},

    		warn: function () {
    			this.invoke(Logger.WARN, arguments);
    		},

    		error: function () {
    			this.invoke(Logger.ERROR, arguments);
    		},

    		time: function (label) {
    			if (typeof label === 'string' && label.length > 0) {
    				this.invoke(Logger.TIME, [ label, 'start' ]);
    			}
    		},

    		timeEnd: function (label) {
    			if (typeof label === 'string' && label.length > 0) {
    				this.invoke(Logger.TIME, [ label, 'end' ]);
    			}
    		},

    		// Invokes the logger callback if it's not being filtered.
    		invoke: function (level, msgArgs) {
    			if (logHandler && this.enabledFor(level)) {
    				logHandler(msgArgs, merge({ level: level }, this.context));
    			}
    		}
    	};

    	// Protected instance which all calls to the to level `Logger` module will be routed through.
    	var globalLogger = new ContextualLogger({ filterLevel: Logger.OFF });

    	// Configure the global Logger instance.
    	(function() {
    		// Shortcut for optimisers.
    		var L = Logger;

    		L.enabledFor = bind(globalLogger, globalLogger.enabledFor);
    		L.trace = bind(globalLogger, globalLogger.trace);
    		L.debug = bind(globalLogger, globalLogger.debug);
    		L.time = bind(globalLogger, globalLogger.time);
    		L.timeEnd = bind(globalLogger, globalLogger.timeEnd);
    		L.info = bind(globalLogger, globalLogger.info);
    		L.warn = bind(globalLogger, globalLogger.warn);
    		L.error = bind(globalLogger, globalLogger.error);

    		// Don't forget the convenience alias!
    		L.log = L.info;
    	}());

    	// Set the global logging handler.  The supplied function should expect two arguments, the first being an arguments
    	// object with the supplied log messages and the second being a context object which contains a hash of stateful
    	// parameters which the logging function can consume.
    	Logger.setHandler = function (func) {
    		logHandler = func;
    	};

    	// Sets the global logging filter level which applies to *all* previously registered, and future Logger instances.
    	// (note that named loggers (retrieved via `Logger.get`) can be configured independently if required).
    	Logger.setLevel = function(level) {
    		// Set the globalLogger's level.
    		globalLogger.setLevel(level);

    		// Apply this level to all registered contextual loggers.
    		for (var key in contextualLoggersByNameMap) {
    			if (contextualLoggersByNameMap.hasOwnProperty(key)) {
    				contextualLoggersByNameMap[key].setLevel(level);
    			}
    		}
    	};

    	// Gets the global logging filter level
    	Logger.getLevel = function() {
    		return globalLogger.getLevel();
    	};

    	// Retrieve a ContextualLogger instance.  Note that named loggers automatically inherit the global logger's level,
    	// default context and log handler.
    	Logger.get = function (name) {
    		// All logger instances are cached so they can be configured ahead of use.
    		return contextualLoggersByNameMap[name] ||
    			(contextualLoggersByNameMap[name] = new ContextualLogger(merge({ name: name }, globalLogger.context)));
    	};

    	// CreateDefaultHandler returns a handler function which can be passed to `Logger.setHandler()` which will
    	// write to the window's console object (if present); the optional options object can be used to customise the
    	// formatter used to format each log message.
    	Logger.createDefaultHandler = function (options) {
    		options = options || {};

    		options.formatter = options.formatter || function defaultMessageFormatter(messages, context) {
    			// Prepend the logger's name to the log message for easy identification.
    			if (context.name) {
    				messages.unshift("[" + context.name + "]");
    			}
    		};

    		// Map of timestamps by timer labels used to track `#time` and `#timeEnd()` invocations in environments
    		// that don't offer a native console method.
    		var timerStartTimeByLabelMap = {};

    		// Support for IE8+ (and other, slightly more sane environments)
    		var invokeConsoleMethod = function (hdlr, messages) {
    			Function.prototype.apply.call(hdlr, console, messages);
    		};

    		// Check for the presence of a logger.
    		if (typeof console === "undefined") {
    			return function () { /* no console */ };
    		}

    		return function(messages, context) {
    			// Convert arguments object to Array.
    			messages = Array.prototype.slice.call(messages);

    			var hdlr = console.log;
    			var timerLabel;

    			if (context.level === Logger.TIME) {
    				timerLabel = (context.name ? '[' + context.name + '] ' : '') + messages[0];

    				if (messages[1] === 'start') {
    					if (console.time) {
    						console.time(timerLabel);
    					}
    					else {
    						timerStartTimeByLabelMap[timerLabel] = new Date().getTime();
    					}
    				}
    				else {
    					if (console.timeEnd) {
    						console.timeEnd(timerLabel);
    					}
    					else {
    						invokeConsoleMethod(hdlr, [ timerLabel + ': ' +
    							(new Date().getTime() - timerStartTimeByLabelMap[timerLabel]) + 'ms' ]);
    					}
    				}
    			}
    			else {
    				// Delegate through to custom warn/error loggers if present on the console.
    				if (context.level === Logger.WARN && console.warn) {
    					hdlr = console.warn;
    				} else if (context.level === Logger.ERROR && console.error) {
    					hdlr = console.error;
    				} else if (context.level === Logger.INFO && console.info) {
    					hdlr = console.info;
    				} else if (context.level === Logger.DEBUG && console.debug) {
    					hdlr = console.debug;
    				} else if (context.level === Logger.TRACE && console.trace) {
    					hdlr = console.trace;
    				}

    				options.formatter(messages, context);
    				invokeConsoleMethod(hdlr, messages);
    			}
    		};
    	};

    	// Configure and example a Default implementation which writes to the `window.console` (if present).  The
    	// `options` hash can be used to configure the default logLevel and provide a custom message formatter.
    	Logger.useDefaults = function(options) {
    		Logger.setLevel(options && options.defaultLevel || Logger.DEBUG);
    		Logger.setHandler(Logger.createDefaultHandler(options));
    	};

    	// Createa an alias to useDefaults to avoid reaking a react-hooks rule.
    	Logger.setDefaults = Logger.useDefaults;

    	// Export to popular environments boilerplate.
    	if (module.exports) {
    		module.exports = Logger;
    	}
    	else {
    		Logger._prevLogger = global.Logger;

    		Logger.noConflict = function () {
    			global.Logger = Logger._prevLogger;
    			return Logger;
    		};

    		global.Logger = Logger;
    	}
    }(commonjsGlobal));
    }(logger));

    var Logger = logger.exports;

    Logger.useDefaults({
        defaultLevel: Logger.DEBUG,
        formatter: (messages, context) => {
            messages.unshift(`[${context.name}] `);
        },
    });

    var MediaFamily;
    (function (MediaFamily) {
        MediaFamily["Audio"] = "AUDIO";
        MediaFamily["Video"] = "VIDEO";
    })(MediaFamily || (MediaFamily = {}));
    var MediaContent;
    (function (MediaContent) {
        MediaContent["Main"] = "MAIN";
        MediaContent["Slides"] = "SLIDES";
    })(MediaContent || (MediaContent = {}));
    var Policy;
    (function (Policy) {
        Policy["ActiveSpeaker"] = "active-speaker";
        Policy["ReceiverSelected"] = "receiver-selected";
    })(Policy || (Policy = {}));
    var MediaType;
    (function (MediaType) {
        MediaType["VideoMain"] = "VIDEO-MAIN";
        MediaType["VideoSlides"] = "VIDEO-SLIDES";
        MediaType["AudioMain"] = "AUDIO-MAIN";
        MediaType["AudioSlides"] = "AUDIO-SLIDES";
    })(MediaType || (MediaType = {}));

    var JmpMsgType;
    (function (JmpMsgType) {
        JmpMsgType["MediaRequest"] = "mediaRequest";
        JmpMsgType["MediaRequestAck"] = "mediaRequestAck";
        JmpMsgType["MediaRequestStatus"] = "mediaRequestStatus";
        JmpMsgType["MediaRequestStatusAck"] = "mediaRequestStatusAck";
        JmpMsgType["SourceAdvertisement"] = "sourceAdvertisement";
        JmpMsgType["SourceAdvertisementAck"] = "sourceAdvertisementAck";
        JmpMsgType["ActiveSpeakerNotification"] = "activeSpeakerNotification";
    })(JmpMsgType || (JmpMsgType = {}));

    var JmpSessionEvents;
    (function (JmpSessionEvents) {
        JmpSessionEvents["ActiveSpeaker"] = "active-speaker";
        JmpSessionEvents["MediaRequestReceived"] = "media-request-received";
        JmpSessionEvents["MediaRequestStatusReceived"] = "media-request-status-received";
        JmpSessionEvents["SourceAdvertisementReceived"] = "source-advertisement-received";
    })(JmpSessionEvents || (JmpSessionEvents = {}));

    var EncodedTransformType;
    (function (EncodedTransformType) {
        EncodedTransformType["AudioLevelMonitor"] = "audio-level-monitor";
    })(EncodedTransformType || (EncodedTransformType = {}));
    var MainMsgType;
    (function (MainMsgType) {
        MainMsgType["GetMetadata"] = "get-metadata";
        MainMsgType["ClearMetadata"] = "clear-metadata";
    })(MainMsgType || (MainMsgType = {}));
    var WorkerMsgType;
    (function (WorkerMsgType) {
        WorkerMsgType["Response"] = "response";
        WorkerMsgType["Log"] = "log";
    })(WorkerMsgType || (WorkerMsgType = {}));

    const workerLogger = {
        info: (...args) => {
            postMessage({ type: WorkerMsgType.Log, level: 'info', messageArgs: args });
        },
        warn: (...args) => {
            postMessage({ type: WorkerMsgType.Log, level: 'warn', messageArgs: args });
        },
        error: (...args) => {
            postMessage({ type: WorkerMsgType.Log, level: 'error', messageArgs: args });
        },
    };
    const collectedMetadata = {};
    const createAudioLevelMonitorTransform = (multistreamConnectionId, encodedTransformId) => {
        if (!(multistreamConnectionId in collectedMetadata)) {
            collectedMetadata[multistreamConnectionId] = {
                [MediaType.AudioMain]: {},
                [MediaType.AudioSlides]: {},
                [MediaType.VideoMain]: {},
                [MediaType.VideoSlides]: {},
            };
        }
        collectedMetadata[multistreamConnectionId][MediaType.AudioMain][encodedTransformId] = {
            maxAudioLevel: undefined,
        };
        const collectedStreamMetadata = collectedMetadata[multistreamConnectionId][MediaType.AudioMain][encodedTransformId];
        workerLogger.info(`creating encoded transform for ${multistreamConnectionId}:${encodedTransformId}`);
        return new TransformStream({
            transform: (encodedFrame, controller) => {
                const metadata = encodedFrame.getMetadata();
                if (!collectedStreamMetadata.maxAudioLevel ||
                    metadata.audioLevel > collectedStreamMetadata.maxAudioLevel) {
                    collectedStreamMetadata.maxAudioLevel = metadata.audioLevel;
                }
                controller.enqueue(encodedFrame);
            },
        });
    };
    addEventListener('rtctransform', (event) => {
        let transform;
        if (!event.transformer.options || !event.transformer.options.type) {
            workerLogger.warn('missing or invalid transformer options=', event.transformer.options);
            return;
        }
        const options = event.transformer.options;
        const { encodedTransformId, multistreamConnectionId, type } = options;
        switch (type) {
            case EncodedTransformType.AudioLevelMonitor:
                transform = createAudioLevelMonitorTransform(multistreamConnectionId, encodedTransformId);
                break;
            default:
                workerLogger.warn('unexpected transformer type=', type);
        }
        if (transform) {
            event.transformer.readable.pipeThrough(transform).pipeTo(event.transformer.writable);
        }
    });
    addEventListener('message', (event) => {
        var _a, _b;
        const { requestId } = event.data;
        const { type } = event.data.payload;
        let responseData;
        switch (type) {
            case MainMsgType.GetMetadata: {
                const { multistreamConnectionId, mediaType, encodedTransformId } = event.data.payload;
                responseData = (_b = (_a = collectedMetadata[multistreamConnectionId]) === null || _a === void 0 ? void 0 : _a[mediaType]) === null || _b === void 0 ? void 0 : _b[encodedTransformId];
                break;
            }
            case MainMsgType.ClearMetadata: {
                const { multistreamConnectionId } = event.data.payload;
                delete collectedMetadata[multistreamConnectionId];
                break;
            }
            default:
                workerLogger.warn('unexpected message type=', type);
                break;
        }
        postMessage({
            type: WorkerMsgType.Response,
            requestId,
            responseData,
        });
    });
    workerLogger.info('Worker fully started.');

})();

', null, false);
|
|
15660
15651
|
/* eslint-enable */
|
|
15661
15652
|
|
|
@@ -15735,7 +15726,7 @@ function getWorkerManager() {
|
|
|
15735
15726
|
return WorkerManager.workerManagerInstance;
|
|
15736
15727
|
}
|
|
15737
15728
|
|
|
15738
|
-
function
|
|
15729
|
+
function mediaTypeToTrackKind(mediaType) {
|
|
15739
15730
|
return [MediaType.VideoMain, MediaType.VideoSlides].includes(mediaType)
|
|
15740
15731
|
? MediaStreamTrackKind.Video
|
|
15741
15732
|
: MediaStreamTrackKind.Audio;
|
|
@@ -15805,9 +15796,10 @@ class MultistreamConnection extends EventEmitter$2 {
|
|
|
15805
15796
|
const slidesSceneId = generateSceneId();
|
|
15806
15797
|
const videoMainEncodingOptions = this.getVideoEncodingOptions(MediaContent.Main);
|
|
15807
15798
|
const videoSlidesEncodingOptions = this.getVideoEncodingOptions(MediaContent.Slides);
|
|
15808
|
-
this.
|
|
15799
|
+
const sendVideoCodecParams = this.getCustomSendVideoCodecParams();
|
|
15800
|
+
this.createSendTransceiver(MediaType.VideoMain, mainSceneId, videoMainEncodingOptions, sendVideoCodecParams);
|
|
15809
15801
|
this.createSendTransceiver(MediaType.AudioMain, mainSceneId);
|
|
15810
|
-
this.createSendTransceiver(MediaType.VideoSlides, slidesSceneId, videoSlidesEncodingOptions);
|
|
15802
|
+
this.createSendTransceiver(MediaType.VideoSlides, slidesSceneId, videoSlidesEncodingOptions, sendVideoCodecParams);
|
|
15811
15803
|
this.createSendTransceiver(MediaType.AudioSlides, slidesSceneId);
|
|
15812
15804
|
}
|
|
15813
15805
|
startWorkerIfNeeded() {
|
|
@@ -15885,10 +15877,29 @@ class MultistreamConnection extends EventEmitter$2 {
|
|
|
15885
15877
|
]
|
|
15886
15878
|
: [{ active: false }];
|
|
15887
15879
|
}
|
|
15888
|
-
|
|
15880
|
+
getCustomSendVideoCodecParams() {
|
|
15881
|
+
return {
|
|
15882
|
+
[MediaCodecMimeType.H264]: {
|
|
15883
|
+
'x-google-start-bitrate': `${this.options.preferredStartingBitrateKbps}`,
|
|
15884
|
+
'max-mbps': `${defaultMaxVideoEncodeMbps}`,
|
|
15885
|
+
'max-fs': `${defaultMaxVideoEncodeFrameSize}`,
|
|
15886
|
+
},
|
|
15887
|
+
};
|
|
15888
|
+
}
|
|
15889
|
+
getCustomReceiveVideoCodecParams(mediaType) {
|
|
15890
|
+
return {
|
|
15891
|
+
[MediaCodecMimeType.H264]: {
|
|
15892
|
+
'sps-pps-idr-in-keyframe': '1',
|
|
15893
|
+
},
|
|
15894
|
+
[MediaCodecMimeType.RTX]: {
|
|
15895
|
+
'rtx-time': mediaType === MediaType.VideoMain ? defaultVideoMainRtxTime : defaultVideoSlidesRtxTime,
|
|
15896
|
+
},
|
|
15897
|
+
};
|
|
15898
|
+
}
|
|
15899
|
+
createSendTransceiver(mediaType, sceneId, sendEncodingsOptions, codecParams) {
|
|
15889
15900
|
let rtcRtpTransceiver;
|
|
15890
15901
|
try {
|
|
15891
|
-
rtcRtpTransceiver = this.pc.addTransceiver(
|
|
15902
|
+
rtcRtpTransceiver = this.pc.addTransceiver(mediaTypeToTrackKind(mediaType), {
|
|
15892
15903
|
direction: 'sendrecv',
|
|
15893
15904
|
sendEncodings: sendEncodingsOptions,
|
|
15894
15905
|
});
|
|
@@ -15907,14 +15918,12 @@ class MultistreamConnection extends EventEmitter$2 {
|
|
|
15907
15918
|
munger,
|
|
15908
15919
|
csi,
|
|
15909
15920
|
});
|
|
15910
|
-
let codecParameters = {
|
|
15911
|
-
'x-google-start-bitrate': `${this.options.preferredStartingBitrateKbps}`,
|
|
15912
|
-
};
|
|
15913
15921
|
if (getMediaFamily(mediaType) === MediaFamily.Video) {
|
|
15914
15922
|
transceiver.rtxEnabled = true;
|
|
15915
|
-
codecParameters = Object.assign(Object.assign({}, codecParameters), { 'max-mbps': `${defaultMaxVideoEncodeMbps}`, 'max-fs': `${defaultMaxVideoEncodeFrameSize}` });
|
|
15916
15923
|
}
|
|
15917
|
-
|
|
15924
|
+
Object.entries(codecParams !== null && codecParams !== void 0 ? codecParams : {}).forEach(([codec, params]) => {
|
|
15925
|
+
transceiver.setCustomCodecParameters(codec, params);
|
|
15926
|
+
});
|
|
15918
15927
|
transceiver.twccDisabled =
|
|
15919
15928
|
getMediaFamily(mediaType) === MediaFamily.Audio ? this.options.disableAudioTwcc : false;
|
|
15920
15929
|
transceiver.dtxDisabled = mediaType !== MediaType.AudioMain || this.options.disableAudioMainDtx;
|
|
@@ -16215,8 +16224,8 @@ SCTP Max Message Size: ${maxMessageSize}`);
|
|
|
16215
16224
|
static getEncodedTransformId(mediaType, mid) {
|
|
16216
16225
|
return `INBOUND-${mediaType}-${mid}`;
|
|
16217
16226
|
}
|
|
16218
|
-
createReceiveTransceiver(mediaType) {
|
|
16219
|
-
const rtcRtpTransceiver = this.pc.addTransceiver(
|
|
16227
|
+
createReceiveTransceiver(mediaType, codecParams) {
|
|
16228
|
+
const rtcRtpTransceiver = this.pc.addTransceiver(mediaTypeToTrackKind(mediaType), {
|
|
16220
16229
|
direction: 'recvonly',
|
|
16221
16230
|
});
|
|
16222
16231
|
const mid = this.midPredictor.getNextMid(mediaType);
|
|
@@ -16244,14 +16253,9 @@ SCTP Max Message Size: ${maxMessageSize}`);
|
|
|
16244
16253
|
logger.warn(`Failed to setup encoded transform for audio level monitoring of mid=${mid}: ${e}`);
|
|
16245
16254
|
}
|
|
16246
16255
|
}
|
|
16247
|
-
|
|
16248
|
-
recvOnlyTransceiver.
|
|
16249
|
-
|
|
16250
|
-
});
|
|
16251
|
-
recvOnlyTransceiver.setRtxCodecParameters({
|
|
16252
|
-
'rtx-time': mediaType === MediaType.VideoMain ? defaultVideoMainRtxTime : defaultVideoSlidesRtxTime,
|
|
16253
|
-
});
|
|
16254
|
-
}
|
|
16256
|
+
Object.entries(codecParams !== null && codecParams !== void 0 ? codecParams : {}).forEach(([codec, params]) => {
|
|
16257
|
+
recvOnlyTransceiver.setCustomCodecParameters(codec, params);
|
|
16258
|
+
});
|
|
16255
16259
|
recvOnlyTransceiver.twccDisabled =
|
|
16256
16260
|
getMediaFamily(mediaType) === MediaFamily.Audio ? this.options.disableAudioTwcc : false;
|
|
16257
16261
|
this.recvTransceivers.set(mediaType, [
|
|
@@ -16270,8 +16274,11 @@ SCTP Max Message Size: ${maxMessageSize}`);
|
|
|
16270
16274
|
return new Promise((createReceiveSlotsResolve) => {
|
|
16271
16275
|
this.offerAnswerQueue.push(() => __awaiter(this, void 0, void 0, function* () {
|
|
16272
16276
|
const createdReceiveOnlyTransceivers = [];
|
|
16277
|
+
const codecParams = mediaTypeToTrackKind(mediaType) === MediaStreamTrackKind.Video
|
|
16278
|
+
? this.getCustomReceiveVideoCodecParams(mediaType)
|
|
16279
|
+
: undefined;
|
|
16273
16280
|
for (let i = 0; i < count; i++) {
|
|
16274
|
-
const recvOnlyTransceiver = this.createReceiveTransceiver(mediaType);
|
|
16281
|
+
const recvOnlyTransceiver = this.createReceiveTransceiver(mediaType, codecParams);
|
|
16275
16282
|
createdReceiveOnlyTransceivers.push(recvOnlyTransceiver);
|
|
16276
16283
|
}
|
|
16277
16284
|
if (this.pc.getRemoteDescription()) {
|
|
@@ -16648,7 +16655,7 @@ SCTP Max Message Size: ${maxMessageSize}`);
|
|
|
16648
16655
|
const mediaContent = getMediaContent(mediaType);
|
|
16649
16656
|
const sceneId = mediaContent === MediaContent.Main ? mainSceneId : slidesSceneId;
|
|
16650
16657
|
const mid = this.midPredictor.getNextMid(mediaType);
|
|
16651
|
-
transceiver.replaceTransceiver(this.pc.addTransceiver(
|
|
16658
|
+
transceiver.replaceTransceiver(this.pc.addTransceiver(mediaTypeToTrackKind(mediaType), {
|
|
16652
16659
|
direction: 'sendrecv',
|
|
16653
16660
|
sendEncodings: getMediaFamily(mediaType) === MediaFamily.Video
|
|
16654
16661
|
? this.getVideoEncodingOptions(mediaContent)
|
|
@@ -16663,7 +16670,7 @@ SCTP Max Message Size: ${maxMessageSize}`);
|
|
|
16663
16670
|
this.recvTransceivers.forEach((transceivers, mediaType) => {
|
|
16664
16671
|
transceivers.forEach((t) => {
|
|
16665
16672
|
const mid = this.midPredictor.getNextMid(mediaType);
|
|
16666
|
-
const rtcRtpTransceiver = this.pc.addTransceiver(
|
|
16673
|
+
const rtcRtpTransceiver = this.pc.addTransceiver(mediaTypeToTrackKind(mediaType), {
|
|
16667
16674
|
direction: 'recvonly',
|
|
16668
16675
|
});
|
|
16669
16676
|
t.replaceTransceiver(rtcRtpTransceiver);
|