@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/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 applyFormatParameters(mediaDescription, codecs, paramsMap) {
10443
- paramsMap.forEach((value, param) => {
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 && codecs.includes(ci.name))
10464
+ .filter((ci) => ci.name && codecName === ci.name.toLowerCase())
10446
10465
  .forEach((ci) => {
10447
- if (value === null) {
10466
+ if (value === undefined) {
10448
10467
  ci.fmtParams.delete(param);
10449
10468
  }
10450
10469
  else {
10451
- ci.fmtParams.set(param, `${value}`);
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
- applyFormatParameters(mediaDescription, ['H264', 'opus'], this.customCodecParameters);
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
- applyFormatParameters(mediaDescription, ['H264', 'opus'], this.customCodecParameters);
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
- setCodecParameters(parameters) {
10605
- Object.entries(parameters).forEach(([param, value]) => {
10606
- this.customCodecParameters.set(param, value);
10607
- });
10627
+ setCustomCodecParameters(codecMimeType, parameters) {
10628
+ setCodecParameters(this.customCodecParameters, codecMimeType, parameters);
10608
10629
  }
10609
- deleteCodecParameters(parameters) {
10610
- parameters.forEach((param) => {
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
- applyFormatParameters(mediaDescription, ['H264', 'opus'], this.customCodecParameters);
11111
- applyFormatParameters(mediaDescription, ['rtx'], this.customRtxCodecParameters);
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
- applyFormatParameters(mediaDescription, ['rtx'], this.customRtxCodecParameters);
11151
+ const rtxParams = this.customCodecParameters.get(MediaCodecMimeType.RTX);
11152
+ if (rtxParams) {
11153
+ setFormatParameters(mediaDescription, MediaCodecMimeType.RTX, rtxParams);
11154
+ }
11133
11155
  }
11134
- setCodecParameters(parameters) {
11135
- Object.entries(parameters).forEach(([param, value]) => {
11136
- this.customCodecParameters.set(param, value);
11137
- });
11156
+ setCustomCodecParameters(codecMimeType, parameters) {
11157
+ setCodecParameters(this.customCodecParameters, codecMimeType, parameters);
11138
11158
  }
11139
- setRtxCodecParameters(parameters) {
11140
- Object.entries(parameters).forEach(([param, value]) => {
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
- setCodecParameters(parameters) {
11456
- this.munger.setCodecParameters(parameters);
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
- setCodecParameters(parameters) {
15436
- this.munger.setCodecParameters(parameters);
15450
+ setCustomCodecParameters(codecMimeType, codecParams) {
15451
+ this.munger.setCustomCodecParameters(codecMimeType, codecParams);
15437
15452
  this.negotiationNeeded.emit(OfferAnswerType.LocalOnly);
15438
15453
  }
15439
- deleteCodecParameters(parameters) {
15440
- this.munger.deleteCodecParameters(parameters);
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
- this.sendTransceiver.setCodecParameters(parameters);
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
- this.sendTransceiver.deleteCodecParameters(parameters);
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$1(base64, sourcemapArg, enableUnicodeArg) {
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 toMediaStreamTrackKind(mediaType) {
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.createSendTransceiver(MediaType.VideoMain, mainSceneId, videoMainEncodingOptions);
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
- createSendTransceiver(mediaType, sceneId, sendEncodingsOptions) {
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(toMediaStreamTrackKind(mediaType), {
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
- transceiver.setCodecParameters(codecParameters);
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(toMediaStreamTrackKind(mediaType), {
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
- if (getMediaFamily(mediaType) === MediaFamily.Video) {
16248
- recvOnlyTransceiver.setCodecParameters({
16249
- 'sps-pps-idr-in-keyframe': '1',
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(toMediaStreamTrackKind(mediaType), {
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(toMediaStreamTrackKind(mediaType), {
16673
+ const rtcRtpTransceiver = this.pc.addTransceiver(mediaTypeToTrackKind(mediaType), {
16667
16674
  direction: 'recvonly',
16668
16675
  });
16669
16676
  t.replaceTransceiver(rtcRtpTransceiver);