@webex/web-client-media-engine 3.34.5 → 3.35.1

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
@@ -7154,6 +7154,29 @@ function isValidActiveSpeakerNotificationMsg(msg) {
7154
7154
  const maybeActiveSpeakerNotificationMsg = msg;
7155
7155
  return Boolean(maybeActiveSpeakerNotificationMsg.seqNum && maybeActiveSpeakerNotificationMsg.csis);
7156
7156
  }
7157
+ function isValidAV1Codec(msg) {
7158
+ if (typeof msg !== 'object' || msg === null) {
7159
+ return false;
7160
+ }
7161
+ const maybeAV1Codec = msg;
7162
+ return (maybeAV1Codec.levelIdx >= 0 &&
7163
+ maybeAV1Codec.tier >= 0 &&
7164
+ Boolean(maybeAV1Codec.maxWidth) &&
7165
+ Boolean(maybeAV1Codec.maxHeight) &&
7166
+ Boolean(maybeAV1Codec.maxPicSize) &&
7167
+ Boolean(maybeAV1Codec.maxDecodeRate));
7168
+ }
7169
+ function areAV1CodecsEqual(left, right) {
7170
+ if (left === undefined || right === undefined) {
7171
+ return left === right;
7172
+ }
7173
+ return (left.levelIdx === right.levelIdx &&
7174
+ left.tier === right.tier &&
7175
+ left.maxWidth === right.maxWidth &&
7176
+ left.maxHeight === right.maxHeight &&
7177
+ left.maxPicSize === right.maxPicSize &&
7178
+ left.maxDecodeRate === right.maxDecodeRate);
7179
+ }
7157
7180
 
7158
7181
  class H264Codec {
7159
7182
  constructor(maxFs, maxFps, maxMbps, maxWidth, maxHeight) {
@@ -7164,6 +7187,13 @@ class H264Codec {
7164
7187
  this.maxHeight = maxHeight;
7165
7188
  }
7166
7189
  }
7190
+ function isValidH264Codec(msg) {
7191
+ if (typeof msg !== 'object' || msg === null) {
7192
+ return false;
7193
+ }
7194
+ const maybeH264Codec = msg;
7195
+ return Boolean(maybeH264Codec.maxFs && maybeH264Codec.maxFps && maybeH264Codec.maxMbps);
7196
+ }
7167
7197
  function areH264CodecsEqual(left, right) {
7168
7198
  if (left === undefined || right === undefined) {
7169
7199
  return left === right;
@@ -7176,13 +7206,31 @@ function areH264CodecsEqual(left, right) {
7176
7206
  }
7177
7207
 
7178
7208
  class CodecInfo$1 {
7179
- constructor(payloadType, h264) {
7209
+ constructor(payloadType, codec) {
7180
7210
  this.payloadType = payloadType;
7181
- this.h264 = h264;
7211
+ if (isValidAV1Codec(codec))
7212
+ this.av1 = codec;
7213
+ if (isValidH264Codec(codec))
7214
+ this.h264 = codec;
7215
+ }
7216
+ static fromH264(payloadType, h264) {
7217
+ return new CodecInfo$1(payloadType, h264);
7218
+ }
7219
+ static fromAv1(payloadType, av1) {
7220
+ return new CodecInfo$1(payloadType, av1);
7182
7221
  }
7183
7222
  }
7184
7223
  function areCodecInfosEqual(left, right) {
7185
- return left.payloadType === right.payloadType && areH264CodecsEqual(left.h264, right.h264);
7224
+ if (left.payloadType !== right.payloadType) {
7225
+ return false;
7226
+ }
7227
+ if (isValidH264Codec(left.h264) && isValidH264Codec(right.h264)) {
7228
+ return areH264CodecsEqual(left.h264, right.h264);
7229
+ }
7230
+ if (isValidAV1Codec(left.av1) && isValidAV1Codec(right.av1)) {
7231
+ return areAV1CodecsEqual(left.av1, right.av1);
7232
+ }
7233
+ return false;
7186
7234
  }
7187
7235
 
7188
7236
  var JmpMsgType;
@@ -7736,6 +7784,7 @@ var WcmeErrorType;
7736
7784
  WcmeErrorType["SET_SOURCE_STATE_OVERRIDE_FAILED"] = "SET_SOURCE_STATE_OVERRIDE_FAILED";
7737
7785
  WcmeErrorType["DATA_CHANNEL_SEND_FAILED"] = "DATA_CHANNEL_SEND_FAILED";
7738
7786
  WcmeErrorType["RENEW_PEER_CONNECTION_FAILED"] = "RENEW_PEER_CONNECTION_FAILED";
7787
+ WcmeErrorType["UNSUPPORTED_CONFIG"] = "UNSUPPORTED_CONFIG";
7739
7788
  })(WcmeErrorType || (WcmeErrorType = {}));
7740
7789
  class WcmeError {
7741
7790
  constructor(type, message = '') {
@@ -10095,6 +10144,12 @@ class BrowserInfo {
10095
10144
  }
10096
10145
  BrowserInfo.browser = Bowser.getParser(window.navigator.userAgent);
10097
10146
 
10147
+ class CpuInfo {
10148
+ static getNumLogicalCores() {
10149
+ return navigator.hardwareConcurrency;
10150
+ }
10151
+ }
10152
+
10098
10153
  var SystemInfoEvents;
10099
10154
  (function (SystemInfoEvents) {
10100
10155
  SystemInfoEvents["CpuPressureStateChange"] = "cpu-pressure-state-change";
@@ -10132,7 +10187,58 @@ var CapabilityState;
10132
10187
  CapabilityState["NOT_CAPABLE"] = "not capable";
10133
10188
  CapabilityState["CAPABLE"] = "capable";
10134
10189
  CapabilityState["UNKNOWN"] = "unknown";
10135
- })(CapabilityState || (CapabilityState = {}));
10190
+ })(CapabilityState || (CapabilityState = {}));
10191
+ class WebCapabilities {
10192
+ static isCapableOfBackgroundNoiseRemoval() {
10193
+ const numCores = CpuInfo.getNumLogicalCores();
10194
+ if (numCores === undefined) {
10195
+ return CapabilityState.UNKNOWN;
10196
+ }
10197
+ if (numCores < 2) {
10198
+ return CapabilityState.NOT_CAPABLE;
10199
+ }
10200
+ return CapabilityState.CAPABLE;
10201
+ }
10202
+ static isCapableOfVirtualBackground() {
10203
+ const numCores = CpuInfo.getNumLogicalCores();
10204
+ if (numCores === undefined) {
10205
+ return CapabilityState.UNKNOWN;
10206
+ }
10207
+ if (numCores < 2) {
10208
+ return CapabilityState.NOT_CAPABLE;
10209
+ }
10210
+ return CapabilityState.CAPABLE;
10211
+ }
10212
+ static isCapableOfReceiving1080pVideo() {
10213
+ const numCores = CpuInfo.getNumLogicalCores();
10214
+ if (numCores === undefined) {
10215
+ return CapabilityState.UNKNOWN;
10216
+ }
10217
+ if (numCores < 2) {
10218
+ return CapabilityState.NOT_CAPABLE;
10219
+ }
10220
+ return CapabilityState.CAPABLE;
10221
+ }
10222
+ static isCapableOfSending1080pVideo() {
10223
+ const numCores = CpuInfo.getNumLogicalCores();
10224
+ if (numCores === undefined) {
10225
+ return CapabilityState.UNKNOWN;
10226
+ }
10227
+ if (numCores < 8) {
10228
+ return CapabilityState.NOT_CAPABLE;
10229
+ }
10230
+ return CapabilityState.CAPABLE;
10231
+ }
10232
+ static supportsEncodedStreamTransforms() {
10233
+ return window.RTCRtpScriptTransform &&
10234
+ window.RTCRtpSender &&
10235
+ 'transform' in RTCRtpSender.prototype &&
10236
+ window.RTCRtpReceiver &&
10237
+ 'transform' in RTCRtpReceiver.prototype
10238
+ ? CapabilityState.CAPABLE
10239
+ : CapabilityState.NOT_CAPABLE;
10240
+ }
10241
+ }
10136
10242
 
10137
10243
  const simulcastMaxFrameSizes = {
10138
10244
  0: '240',
@@ -11056,6 +11162,21 @@ class HomerMsg {
11056
11162
  }
11057
11163
  }
11058
11164
 
11165
+ var EncodedTransformType;
11166
+ (function (EncodedTransformType) {
11167
+ EncodedTransformType["AudioLevelMonitor"] = "audio-level-monitor";
11168
+ })(EncodedTransformType || (EncodedTransformType = {}));
11169
+ var MainMsgType;
11170
+ (function (MainMsgType) {
11171
+ MainMsgType["GetMetadata"] = "get-metadata";
11172
+ MainMsgType["ClearMetadata"] = "clear-metadata";
11173
+ })(MainMsgType || (MainMsgType = {}));
11174
+ var WorkerMsgType;
11175
+ (function (WorkerMsgType) {
11176
+ WorkerMsgType["Response"] = "response";
11177
+ WorkerMsgType["Log"] = "log";
11178
+ })(WorkerMsgType || (WorkerMsgType = {}));
11179
+
11059
11180
  class MidPredictor {
11060
11181
  constructor() {
11061
11182
  this.currentMid = 0;
@@ -11236,6 +11357,7 @@ class ReceiveOnlyTransceiver extends Transceiver {
11236
11357
  constructor(config) {
11237
11358
  super(config);
11238
11359
  this.metadata = { isActiveSpeaker: false };
11360
+ this.getEncodedStreamMetadataCallback = () => Promise.resolve(undefined);
11239
11361
  this.munger = config.munger;
11240
11362
  this._receiveSlot = new ReceiveSlot(() => {
11241
11363
  if (!this._rtcRtpTransceiver.mid) {
@@ -11244,6 +11366,9 @@ class ReceiveOnlyTransceiver extends Transceiver {
11244
11366
  return this.munger.getReceiverId();
11245
11367
  }, this._rtcRtpTransceiver.receiver.track);
11246
11368
  }
11369
+ setEncodedStreamMetadataCallback(callback) {
11370
+ this.getEncodedStreamMetadataCallback = callback;
11371
+ }
11247
11372
  replaceTransceiver(newRtcRtpTransceiver) {
11248
11373
  super.replaceTransceiver(newRtcRtpTransceiver);
11249
11374
  this._receiveSlot._replaceTrack(newRtcRtpTransceiver.receiver.track);
@@ -11258,7 +11383,10 @@ class ReceiveOnlyTransceiver extends Transceiver {
11258
11383
  getStats() {
11259
11384
  return __awaiter(this, void 0, void 0, function* () {
11260
11385
  const statsMap = new Map();
11261
- const statsReport = yield this.receiver.getStats();
11386
+ const [statsReport, encodedStreamMetadata] = yield Promise.all([
11387
+ this.receiver.getStats(),
11388
+ this.getEncodedStreamMetadataCallback(),
11389
+ ]);
11262
11390
  statsReport.forEach((stats, key) => {
11263
11391
  if (stats.type === 'inbound-rtp') {
11264
11392
  stats.mid = this.mid;
@@ -11272,6 +11400,10 @@ class ReceiveOnlyTransceiver extends Transceiver {
11272
11400
  stats.lastRequestedUpdateTimestamp = this.metadata.lastRequestedUpdateTimestamp;
11273
11401
  stats.isActiveSpeaker = this.metadata.isActiveSpeaker;
11274
11402
  stats.lastActiveSpeakerUpdateTimestamp = this.metadata.lastActiveSpeakerUpdateTimestamp;
11403
+ stats.maxAudioLevelFromRtpHeader =
11404
+ encodedStreamMetadata && 'maxAudioLevel' in encodedStreamMetadata
11405
+ ? encodedStreamMetadata.maxAudioLevel
11406
+ : undefined;
11275
11407
  Object.assign(stats, this.receiverId);
11276
11408
  }
11277
11409
  statsMap.set(key, stats);
@@ -15454,6 +15586,154 @@ const organizeTransceiverStats = (sendTransceivers, recvTransceivers, peerConnec
15454
15586
  return result;
15455
15587
  });
15456
15588
 
15589
+ var WorkerClass = null;
15590
+
15591
+ try {
15592
+ var WorkerThreads =
15593
+ typeof module !== 'undefined' && typeof module.require === 'function' && module.require('worker_threads') ||
15594
+ typeof __non_webpack_require__ === 'function' && __non_webpack_require__('worker_threads') ||
15595
+ typeof require === 'function' && require('worker_threads');
15596
+ WorkerClass = WorkerThreads.Worker;
15597
+ } catch(e) {} // eslint-disable-line
15598
+
15599
+ function decodeBase64$1(base64, enableUnicode) {
15600
+ return Buffer.from(base64, 'base64').toString(enableUnicode ? 'utf16' : 'utf8');
15601
+ }
15602
+
15603
+ function createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg) {
15604
+ var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
15605
+ var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
15606
+ var source = decodeBase64$1(base64, enableUnicode);
15607
+ var start = source.indexOf('\n', 10) + 1;
15608
+ var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
15609
+ return function WorkerFactory(options) {
15610
+ return new WorkerClass(body, Object.assign({}, options, { eval: true }));
15611
+ };
15612
+ }
15613
+
15614
+ function decodeBase64(base64, enableUnicode) {
15615
+ var binaryString = atob(base64);
15616
+ if (enableUnicode) {
15617
+ var binaryView = new Uint8Array(binaryString.length);
15618
+ for (var i = 0, n = binaryString.length; i < n; ++i) {
15619
+ binaryView[i] = binaryString.charCodeAt(i);
15620
+ }
15621
+ const decoder = new TextDecoder("utf-16le");
15622
+ return decoder.decode(new Uint16Array(binaryView.buffer));
15623
+ }
15624
+ return binaryString;
15625
+ }
15626
+
15627
+ function createURL(base64, sourcemapArg, enableUnicodeArg) {
15628
+ var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
15629
+ var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
15630
+ var source = decodeBase64(base64, enableUnicode);
15631
+ var start = source.indexOf('\n', 10) + 1;
15632
+ var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
15633
+ var blob = new Blob([body], { type: 'application/javascript' });
15634
+ return URL.createObjectURL(blob);
15635
+ }
15636
+
15637
+ function createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg) {
15638
+ var url;
15639
+ return function WorkerFactory(options) {
15640
+ url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
15641
+ return new Worker(url, options);
15642
+ };
15643
+ }
15644
+
15645
+ var kIsNodeJS = Object.prototype.toString.call(typeof process !== 'undefined' ? process : 0) === '[object process]';
15646
+
15647
+ function isNodeJS() {
15648
+ return kIsNodeJS;
15649
+ }
15650
+
15651
+ function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
15652
+ if (isNodeJS()) {
15653
+ return createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg);
15654
+ }
15655
+ return createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg);
15656
+ }
15657
+
15658
+ 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);
15659
+ /* eslint-enable */
15660
+
15661
+ class WorkerManager {
15662
+ constructor() {
15663
+ this.requestsInProgress = new Map();
15664
+ this.lastRequestId = 0;
15665
+ }
15666
+ startWorker() {
15667
+ if (this.worker) {
15668
+ logger.info('Worker is already started');
15669
+ return;
15670
+ }
15671
+ logger.info('Starting worker...');
15672
+ this.worker = new WorkerFactory();
15673
+ this.worker.onmessage = this.messageHandler.bind(this);
15674
+ }
15675
+ stopWorker() {
15676
+ }
15677
+ getWorker() {
15678
+ return this.worker;
15679
+ }
15680
+ messageHandler(event) {
15681
+ const { type } = event.data;
15682
+ switch (type) {
15683
+ case WorkerMsgType.Response: {
15684
+ const { requestId, responseData } = event.data;
15685
+ const request = this.requestsInProgress.get(requestId);
15686
+ if (request) {
15687
+ request.resolve(responseData);
15688
+ this.requestsInProgress.delete(requestId);
15689
+ }
15690
+ else {
15691
+ logger.warn(`received response for unknown requestId: ${requestId}`);
15692
+ }
15693
+ break;
15694
+ }
15695
+ case WorkerMsgType.Log: {
15696
+ const { messageArgs, level } = event.data;
15697
+ switch (level) {
15698
+ case 'info':
15699
+ logger.log(`worker: `, ...messageArgs);
15700
+ break;
15701
+ case 'warn':
15702
+ logger.warn(`worker: `, ...messageArgs);
15703
+ break;
15704
+ case 'error':
15705
+ logger.error(`worker: `, ...messageArgs);
15706
+ break;
15707
+ default:
15708
+ logger.log(`worker (unsupported log level ${level}): `, ...messageArgs);
15709
+ break;
15710
+ }
15711
+ break;
15712
+ }
15713
+ default:
15714
+ logger.warn(`received unknown message type from worker: ${type}`);
15715
+ }
15716
+ }
15717
+ sendRequestToWorker(payload) {
15718
+ return new Promise((resolve, reject) => {
15719
+ if (!this.worker) {
15720
+ reject(new Error('Worker is not started'));
15721
+ return;
15722
+ }
15723
+ const requestId = this.lastRequestId++;
15724
+ this.requestsInProgress.set(requestId, { resolve, reject });
15725
+ this.worker.postMessage({
15726
+ requestId,
15727
+ payload,
15728
+ });
15729
+ });
15730
+ }
15731
+ }
15732
+ WorkerManager.workerManagerInstance = new WorkerManager();
15733
+ function getWorkerManager() {
15734
+ return WorkerManager.workerManagerInstance;
15735
+ }
15736
+
15457
15737
  function toMediaStreamTrackKind(mediaType) {
15458
15738
  return [MediaType.VideoMain, MediaType.VideoSlides].includes(mediaType)
15459
15739
  ? MediaStreamTrackKind.Video
@@ -15496,7 +15776,9 @@ const defaultMultistreamConnectionOptions = {
15496
15776
  disableAudioMainDtx: true,
15497
15777
  preferredStartingBitrateKbps: defaultStartBitrateKbps,
15498
15778
  metricsCallback: () => { },
15779
+ enableInboundAudioLevelMonitoring: false,
15499
15780
  };
15781
+ let staticIdCounter = 0;
15500
15782
  class MultistreamConnection extends EventEmitter$2 {
15501
15783
  constructor(userOptions = {}) {
15502
15784
  super();
@@ -15509,9 +15791,11 @@ class MultistreamConnection extends EventEmitter$2 {
15509
15791
  this.offerAnswerQueue = new AsyncQueue();
15510
15792
  this.currentCreateOfferId = 0;
15511
15793
  this.metadata = { isMediaBypassEdge: false };
15794
+ this.id = `mc-${staticIdCounter++}`;
15512
15795
  this.options = Object.assign(Object.assign({}, defaultMultistreamConnectionOptions), userOptions);
15513
15796
  logger.info(`Creating multistream connection with options ${JSON.stringify(this.options)}`);
15514
15797
  this.metricsCallback = this.options.metricsCallback;
15798
+ this.startWorkerIfNeeded();
15515
15799
  this.initializePeerConnection();
15516
15800
  this.overuseStateManager = new OveruseStateManager((overuseState) => this.overuseUpdateCallback(overuseState));
15517
15801
  this.overuseStateManager.start();
@@ -15525,6 +15809,15 @@ class MultistreamConnection extends EventEmitter$2 {
15525
15809
  this.createSendTransceiver(MediaType.VideoSlides, slidesSceneId, videoSlidesEncodingOptions);
15526
15810
  this.createSendTransceiver(MediaType.AudioSlides, slidesSceneId);
15527
15811
  }
15812
+ startWorkerIfNeeded() {
15813
+ if (this.options.enableInboundAudioLevelMonitoring) {
15814
+ if (WebCapabilities.supportsEncodedStreamTransforms() !== CapabilityState.CAPABLE) {
15815
+ logger.warn('RTCRtpScriptTransform is not supported in this browser, skipping inbound audio level monitoring.');
15816
+ return;
15817
+ }
15818
+ getWorkerManager().startWorker();
15819
+ }
15820
+ }
15528
15821
  initializePeerConnection() {
15529
15822
  this.pc = new PeerConnection({
15530
15823
  iceServers: this.options.iceServers,
@@ -15848,6 +16141,10 @@ SCTP Max Message Size: ${maxMessageSize}`);
15848
16141
  });
15849
16142
  this.jmpSessions.forEach((jmpSession) => jmpSession.close());
15850
16143
  this.pc.close();
16144
+ getWorkerManager().sendRequestToWorker({
16145
+ type: MainMsgType.ClearMetadata,
16146
+ multistreamConnectionId: this.id,
16147
+ });
15851
16148
  }
15852
16149
  sendMediaRequestStatus(mediaType) {
15853
16150
  var _a;
@@ -15919,6 +16216,7 @@ SCTP Max Message Size: ${maxMessageSize}`);
15919
16216
  direction: 'recvonly',
15920
16217
  });
15921
16218
  const mid = this.midPredictor.getNextMid(mediaType);
16219
+ const doAudioLevelMonitoring = this.options.enableInboundAudioLevelMonitoring && mediaType === MediaType.AudioMain;
15922
16220
  const munger = new IngressSdpMunger();
15923
16221
  const recvOnlyTransceiver = new ReceiveOnlyTransceiver({
15924
16222
  rtcRtpTransceiver,
@@ -15926,6 +16224,23 @@ SCTP Max Message Size: ${maxMessageSize}`);
15926
16224
  mediaType,
15927
16225
  munger,
15928
16226
  });
16227
+ if (doAudioLevelMonitoring) {
16228
+ const encodedTransformId = `INBOUND-${mediaType}-${mid}`;
16229
+ try {
16230
+ this.setupEncodedTransform(rtcRtpTransceiver.receiver, encodedTransformId, EncodedTransformType.AudioLevelMonitor);
16231
+ recvOnlyTransceiver.setEncodedStreamMetadataCallback(() => __awaiter(this, void 0, void 0, function* () {
16232
+ return getWorkerManager().sendRequestToWorker({
16233
+ type: MainMsgType.GetMetadata,
16234
+ mediaType,
16235
+ encodedTransformId,
16236
+ multistreamConnectionId: this.id,
16237
+ });
16238
+ }));
16239
+ }
16240
+ catch (e) {
16241
+ logger.warn(`Failed to setup encoded transform for ${encodedTransformId} audio level monitoring: ${e}`);
16242
+ }
16243
+ }
15929
16244
  if (getMediaFamily(mediaType) === MediaFamily.Video) {
15930
16245
  recvOnlyTransceiver.setCodecParameters({
15931
16246
  'sps-pps-idr-in-keyframe': '1',
@@ -16414,6 +16729,21 @@ SCTP Max Message Size: ${maxMessageSize}`);
16414
16729
  videoSlides: this.getCsiByMediaType(MediaType.VideoSlides),
16415
16730
  };
16416
16731
  }
16732
+ setupEncodedTransform(receiverOrSender, id, type) {
16733
+ if (WebCapabilities.supportsEncodedStreamTransforms() !== CapabilityState.CAPABLE) {
16734
+ throw new Error('encoded stream transport is not supported in this browser.');
16735
+ }
16736
+ if (!getWorkerManager().getWorker()) {
16737
+ throw new Error('no worker available for encoded stream transforms.');
16738
+ }
16739
+ const options = {
16740
+ type,
16741
+ encodedTransformId: id,
16742
+ multistreamConnectionId: this.id,
16743
+ };
16744
+ receiverOrSender.transform = new window.RTCRtpScriptTransform(getWorkerManager().getWorker(), options);
16745
+ logger.log(`started "${type}" transform with id=${id} on connection ${this.id}`);
16746
+ }
16417
16747
  }
16418
16748
 
16419
16749
  class StreamRequest {