@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/cjs/index.js CHANGED
@@ -7158,6 +7158,29 @@ function isValidActiveSpeakerNotificationMsg(msg) {
7158
7158
  const maybeActiveSpeakerNotificationMsg = msg;
7159
7159
  return Boolean(maybeActiveSpeakerNotificationMsg.seqNum && maybeActiveSpeakerNotificationMsg.csis);
7160
7160
  }
7161
+ function isValidAV1Codec(msg) {
7162
+ if (typeof msg !== 'object' || msg === null) {
7163
+ return false;
7164
+ }
7165
+ const maybeAV1Codec = msg;
7166
+ return (maybeAV1Codec.levelIdx >= 0 &&
7167
+ maybeAV1Codec.tier >= 0 &&
7168
+ Boolean(maybeAV1Codec.maxWidth) &&
7169
+ Boolean(maybeAV1Codec.maxHeight) &&
7170
+ Boolean(maybeAV1Codec.maxPicSize) &&
7171
+ Boolean(maybeAV1Codec.maxDecodeRate));
7172
+ }
7173
+ function areAV1CodecsEqual(left, right) {
7174
+ if (left === undefined || right === undefined) {
7175
+ return left === right;
7176
+ }
7177
+ return (left.levelIdx === right.levelIdx &&
7178
+ left.tier === right.tier &&
7179
+ left.maxWidth === right.maxWidth &&
7180
+ left.maxHeight === right.maxHeight &&
7181
+ left.maxPicSize === right.maxPicSize &&
7182
+ left.maxDecodeRate === right.maxDecodeRate);
7183
+ }
7161
7184
 
7162
7185
  class H264Codec {
7163
7186
  constructor(maxFs, maxFps, maxMbps, maxWidth, maxHeight) {
@@ -7168,6 +7191,13 @@ class H264Codec {
7168
7191
  this.maxHeight = maxHeight;
7169
7192
  }
7170
7193
  }
7194
+ function isValidH264Codec(msg) {
7195
+ if (typeof msg !== 'object' || msg === null) {
7196
+ return false;
7197
+ }
7198
+ const maybeH264Codec = msg;
7199
+ return Boolean(maybeH264Codec.maxFs && maybeH264Codec.maxFps && maybeH264Codec.maxMbps);
7200
+ }
7171
7201
  function areH264CodecsEqual(left, right) {
7172
7202
  if (left === undefined || right === undefined) {
7173
7203
  return left === right;
@@ -7180,13 +7210,31 @@ function areH264CodecsEqual(left, right) {
7180
7210
  }
7181
7211
 
7182
7212
  class CodecInfo$1 {
7183
- constructor(payloadType, h264) {
7213
+ constructor(payloadType, codec) {
7184
7214
  this.payloadType = payloadType;
7185
- this.h264 = h264;
7215
+ if (isValidAV1Codec(codec))
7216
+ this.av1 = codec;
7217
+ if (isValidH264Codec(codec))
7218
+ this.h264 = codec;
7219
+ }
7220
+ static fromH264(payloadType, h264) {
7221
+ return new CodecInfo$1(payloadType, h264);
7222
+ }
7223
+ static fromAv1(payloadType, av1) {
7224
+ return new CodecInfo$1(payloadType, av1);
7186
7225
  }
7187
7226
  }
7188
7227
  function areCodecInfosEqual(left, right) {
7189
- return left.payloadType === right.payloadType && areH264CodecsEqual(left.h264, right.h264);
7228
+ if (left.payloadType !== right.payloadType) {
7229
+ return false;
7230
+ }
7231
+ if (isValidH264Codec(left.h264) && isValidH264Codec(right.h264)) {
7232
+ return areH264CodecsEqual(left.h264, right.h264);
7233
+ }
7234
+ if (isValidAV1Codec(left.av1) && isValidAV1Codec(right.av1)) {
7235
+ return areAV1CodecsEqual(left.av1, right.av1);
7236
+ }
7237
+ return false;
7190
7238
  }
7191
7239
 
7192
7240
  var JmpMsgType;
@@ -7740,6 +7788,7 @@ exports.WcmeErrorType = void 0;
7740
7788
  WcmeErrorType["SET_SOURCE_STATE_OVERRIDE_FAILED"] = "SET_SOURCE_STATE_OVERRIDE_FAILED";
7741
7789
  WcmeErrorType["DATA_CHANNEL_SEND_FAILED"] = "DATA_CHANNEL_SEND_FAILED";
7742
7790
  WcmeErrorType["RENEW_PEER_CONNECTION_FAILED"] = "RENEW_PEER_CONNECTION_FAILED";
7791
+ WcmeErrorType["UNSUPPORTED_CONFIG"] = "UNSUPPORTED_CONFIG";
7743
7792
  })(exports.WcmeErrorType || (exports.WcmeErrorType = {}));
7744
7793
  class WcmeError {
7745
7794
  constructor(type, message = '') {
@@ -10099,6 +10148,12 @@ class BrowserInfo {
10099
10148
  }
10100
10149
  BrowserInfo.browser = Bowser.getParser(window.navigator.userAgent);
10101
10150
 
10151
+ class CpuInfo {
10152
+ static getNumLogicalCores() {
10153
+ return navigator.hardwareConcurrency;
10154
+ }
10155
+ }
10156
+
10102
10157
  var SystemInfoEvents;
10103
10158
  (function (SystemInfoEvents) {
10104
10159
  SystemInfoEvents["CpuPressureStateChange"] = "cpu-pressure-state-change";
@@ -10136,7 +10191,58 @@ var CapabilityState;
10136
10191
  CapabilityState["NOT_CAPABLE"] = "not capable";
10137
10192
  CapabilityState["CAPABLE"] = "capable";
10138
10193
  CapabilityState["UNKNOWN"] = "unknown";
10139
- })(CapabilityState || (CapabilityState = {}));
10194
+ })(CapabilityState || (CapabilityState = {}));
10195
+ class WebCapabilities {
10196
+ static isCapableOfBackgroundNoiseRemoval() {
10197
+ const numCores = CpuInfo.getNumLogicalCores();
10198
+ if (numCores === undefined) {
10199
+ return CapabilityState.UNKNOWN;
10200
+ }
10201
+ if (numCores < 2) {
10202
+ return CapabilityState.NOT_CAPABLE;
10203
+ }
10204
+ return CapabilityState.CAPABLE;
10205
+ }
10206
+ static isCapableOfVirtualBackground() {
10207
+ const numCores = CpuInfo.getNumLogicalCores();
10208
+ if (numCores === undefined) {
10209
+ return CapabilityState.UNKNOWN;
10210
+ }
10211
+ if (numCores < 2) {
10212
+ return CapabilityState.NOT_CAPABLE;
10213
+ }
10214
+ return CapabilityState.CAPABLE;
10215
+ }
10216
+ static isCapableOfReceiving1080pVideo() {
10217
+ const numCores = CpuInfo.getNumLogicalCores();
10218
+ if (numCores === undefined) {
10219
+ return CapabilityState.UNKNOWN;
10220
+ }
10221
+ if (numCores < 2) {
10222
+ return CapabilityState.NOT_CAPABLE;
10223
+ }
10224
+ return CapabilityState.CAPABLE;
10225
+ }
10226
+ static isCapableOfSending1080pVideo() {
10227
+ const numCores = CpuInfo.getNumLogicalCores();
10228
+ if (numCores === undefined) {
10229
+ return CapabilityState.UNKNOWN;
10230
+ }
10231
+ if (numCores < 8) {
10232
+ return CapabilityState.NOT_CAPABLE;
10233
+ }
10234
+ return CapabilityState.CAPABLE;
10235
+ }
10236
+ static supportsEncodedStreamTransforms() {
10237
+ return window.RTCRtpScriptTransform &&
10238
+ window.RTCRtpSender &&
10239
+ 'transform' in RTCRtpSender.prototype &&
10240
+ window.RTCRtpReceiver &&
10241
+ 'transform' in RTCRtpReceiver.prototype
10242
+ ? CapabilityState.CAPABLE
10243
+ : CapabilityState.NOT_CAPABLE;
10244
+ }
10245
+ }
10140
10246
 
10141
10247
  const simulcastMaxFrameSizes = {
10142
10248
  0: '240',
@@ -11060,6 +11166,21 @@ class HomerMsg {
11060
11166
  }
11061
11167
  }
11062
11168
 
11169
+ var EncodedTransformType;
11170
+ (function (EncodedTransformType) {
11171
+ EncodedTransformType["AudioLevelMonitor"] = "audio-level-monitor";
11172
+ })(EncodedTransformType || (EncodedTransformType = {}));
11173
+ var MainMsgType;
11174
+ (function (MainMsgType) {
11175
+ MainMsgType["GetMetadata"] = "get-metadata";
11176
+ MainMsgType["ClearMetadata"] = "clear-metadata";
11177
+ })(MainMsgType || (MainMsgType = {}));
11178
+ var WorkerMsgType;
11179
+ (function (WorkerMsgType) {
11180
+ WorkerMsgType["Response"] = "response";
11181
+ WorkerMsgType["Log"] = "log";
11182
+ })(WorkerMsgType || (WorkerMsgType = {}));
11183
+
11063
11184
  class MidPredictor {
11064
11185
  constructor() {
11065
11186
  this.currentMid = 0;
@@ -11240,6 +11361,7 @@ class ReceiveOnlyTransceiver extends Transceiver {
11240
11361
  constructor(config) {
11241
11362
  super(config);
11242
11363
  this.metadata = { isActiveSpeaker: false };
11364
+ this.getEncodedStreamMetadataCallback = () => Promise.resolve(undefined);
11243
11365
  this.munger = config.munger;
11244
11366
  this._receiveSlot = new ReceiveSlot(() => {
11245
11367
  if (!this._rtcRtpTransceiver.mid) {
@@ -11248,6 +11370,9 @@ class ReceiveOnlyTransceiver extends Transceiver {
11248
11370
  return this.munger.getReceiverId();
11249
11371
  }, this._rtcRtpTransceiver.receiver.track);
11250
11372
  }
11373
+ setEncodedStreamMetadataCallback(callback) {
11374
+ this.getEncodedStreamMetadataCallback = callback;
11375
+ }
11251
11376
  replaceTransceiver(newRtcRtpTransceiver) {
11252
11377
  super.replaceTransceiver(newRtcRtpTransceiver);
11253
11378
  this._receiveSlot._replaceTrack(newRtcRtpTransceiver.receiver.track);
@@ -11262,7 +11387,10 @@ class ReceiveOnlyTransceiver extends Transceiver {
11262
11387
  getStats() {
11263
11388
  return __awaiter(this, void 0, void 0, function* () {
11264
11389
  const statsMap = new Map();
11265
- const statsReport = yield this.receiver.getStats();
11390
+ const [statsReport, encodedStreamMetadata] = yield Promise.all([
11391
+ this.receiver.getStats(),
11392
+ this.getEncodedStreamMetadataCallback(),
11393
+ ]);
11266
11394
  statsReport.forEach((stats, key) => {
11267
11395
  if (stats.type === 'inbound-rtp') {
11268
11396
  stats.mid = this.mid;
@@ -11276,6 +11404,10 @@ class ReceiveOnlyTransceiver extends Transceiver {
11276
11404
  stats.lastRequestedUpdateTimestamp = this.metadata.lastRequestedUpdateTimestamp;
11277
11405
  stats.isActiveSpeaker = this.metadata.isActiveSpeaker;
11278
11406
  stats.lastActiveSpeakerUpdateTimestamp = this.metadata.lastActiveSpeakerUpdateTimestamp;
11407
+ stats.maxAudioLevelFromRtpHeader =
11408
+ encodedStreamMetadata && 'maxAudioLevel' in encodedStreamMetadata
11409
+ ? encodedStreamMetadata.maxAudioLevel
11410
+ : undefined;
11279
11411
  Object.assign(stats, this.receiverId);
11280
11412
  }
11281
11413
  statsMap.set(key, stats);
@@ -15458,6 +15590,154 @@ const organizeTransceiverStats = (sendTransceivers, recvTransceivers, peerConnec
15458
15590
  return result;
15459
15591
  });
15460
15592
 
15593
+ var WorkerClass = null;
15594
+
15595
+ try {
15596
+ var WorkerThreads =
15597
+ typeof module !== 'undefined' && typeof module.require === 'function' && module.require('worker_threads') ||
15598
+ typeof __non_webpack_require__ === 'function' && __non_webpack_require__('worker_threads') ||
15599
+ typeof require === 'function' && require('worker_threads');
15600
+ WorkerClass = WorkerThreads.Worker;
15601
+ } catch(e) {} // eslint-disable-line
15602
+
15603
+ function decodeBase64$1(base64, enableUnicode) {
15604
+ return Buffer.from(base64, 'base64').toString(enableUnicode ? 'utf16' : 'utf8');
15605
+ }
15606
+
15607
+ function createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg) {
15608
+ var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
15609
+ var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
15610
+ var source = decodeBase64$1(base64, enableUnicode);
15611
+ var start = source.indexOf('\n', 10) + 1;
15612
+ var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
15613
+ return function WorkerFactory(options) {
15614
+ return new WorkerClass(body, Object.assign({}, options, { eval: true }));
15615
+ };
15616
+ }
15617
+
15618
+ function decodeBase64(base64, enableUnicode) {
15619
+ var binaryString = atob(base64);
15620
+ if (enableUnicode) {
15621
+ var binaryView = new Uint8Array(binaryString.length);
15622
+ for (var i = 0, n = binaryString.length; i < n; ++i) {
15623
+ binaryView[i] = binaryString.charCodeAt(i);
15624
+ }
15625
+ const decoder = new TextDecoder("utf-16le");
15626
+ return decoder.decode(new Uint16Array(binaryView.buffer));
15627
+ }
15628
+ return binaryString;
15629
+ }
15630
+
15631
+ function createURL(base64, sourcemapArg, enableUnicodeArg) {
15632
+ var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
15633
+ var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
15634
+ var source = decodeBase64(base64, enableUnicode);
15635
+ var start = source.indexOf('\n', 10) + 1;
15636
+ var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
15637
+ var blob = new Blob([body], { type: 'application/javascript' });
15638
+ return URL.createObjectURL(blob);
15639
+ }
15640
+
15641
+ function createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg) {
15642
+ var url;
15643
+ return function WorkerFactory(options) {
15644
+ url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
15645
+ return new Worker(url, options);
15646
+ };
15647
+ }
15648
+
15649
+ var kIsNodeJS = Object.prototype.toString.call(typeof process !== 'undefined' ? process : 0) === '[object process]';
15650
+
15651
+ function isNodeJS() {
15652
+ return kIsNodeJS;
15653
+ }
15654
+
15655
+ function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
15656
+ if (isNodeJS()) {
15657
+ return createBase64WorkerFactory$2(base64, sourcemapArg, enableUnicodeArg);
15658
+ }
15659
+ return createBase64WorkerFactory$1(base64, sourcemapArg, enableUnicodeArg);
15660
+ }
15661
+
15662
+ 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);
15663
+ /* eslint-enable */
15664
+
15665
+ class WorkerManager {
15666
+ constructor() {
15667
+ this.requestsInProgress = new Map();
15668
+ this.lastRequestId = 0;
15669
+ }
15670
+ startWorker() {
15671
+ if (this.worker) {
15672
+ logger.info('Worker is already started');
15673
+ return;
15674
+ }
15675
+ logger.info('Starting worker...');
15676
+ this.worker = new WorkerFactory();
15677
+ this.worker.onmessage = this.messageHandler.bind(this);
15678
+ }
15679
+ stopWorker() {
15680
+ }
15681
+ getWorker() {
15682
+ return this.worker;
15683
+ }
15684
+ messageHandler(event) {
15685
+ const { type } = event.data;
15686
+ switch (type) {
15687
+ case WorkerMsgType.Response: {
15688
+ const { requestId, responseData } = event.data;
15689
+ const request = this.requestsInProgress.get(requestId);
15690
+ if (request) {
15691
+ request.resolve(responseData);
15692
+ this.requestsInProgress.delete(requestId);
15693
+ }
15694
+ else {
15695
+ logger.warn(`received response for unknown requestId: ${requestId}`);
15696
+ }
15697
+ break;
15698
+ }
15699
+ case WorkerMsgType.Log: {
15700
+ const { messageArgs, level } = event.data;
15701
+ switch (level) {
15702
+ case 'info':
15703
+ logger.log(`worker: `, ...messageArgs);
15704
+ break;
15705
+ case 'warn':
15706
+ logger.warn(`worker: `, ...messageArgs);
15707
+ break;
15708
+ case 'error':
15709
+ logger.error(`worker: `, ...messageArgs);
15710
+ break;
15711
+ default:
15712
+ logger.log(`worker (unsupported log level ${level}): `, ...messageArgs);
15713
+ break;
15714
+ }
15715
+ break;
15716
+ }
15717
+ default:
15718
+ logger.warn(`received unknown message type from worker: ${type}`);
15719
+ }
15720
+ }
15721
+ sendRequestToWorker(payload) {
15722
+ return new Promise((resolve, reject) => {
15723
+ if (!this.worker) {
15724
+ reject(new Error('Worker is not started'));
15725
+ return;
15726
+ }
15727
+ const requestId = this.lastRequestId++;
15728
+ this.requestsInProgress.set(requestId, { resolve, reject });
15729
+ this.worker.postMessage({
15730
+ requestId,
15731
+ payload,
15732
+ });
15733
+ });
15734
+ }
15735
+ }
15736
+ WorkerManager.workerManagerInstance = new WorkerManager();
15737
+ function getWorkerManager() {
15738
+ return WorkerManager.workerManagerInstance;
15739
+ }
15740
+
15461
15741
  function toMediaStreamTrackKind(mediaType) {
15462
15742
  return [exports.MediaType.VideoMain, exports.MediaType.VideoSlides].includes(mediaType)
15463
15743
  ? exports.MediaStreamTrackKind.Video
@@ -15500,7 +15780,9 @@ const defaultMultistreamConnectionOptions = {
15500
15780
  disableAudioMainDtx: true,
15501
15781
  preferredStartingBitrateKbps: defaultStartBitrateKbps,
15502
15782
  metricsCallback: () => { },
15783
+ enableInboundAudioLevelMonitoring: false,
15503
15784
  };
15785
+ let staticIdCounter = 0;
15504
15786
  class MultistreamConnection extends EventEmitter$2 {
15505
15787
  constructor(userOptions = {}) {
15506
15788
  super();
@@ -15513,9 +15795,11 @@ class MultistreamConnection extends EventEmitter$2 {
15513
15795
  this.offerAnswerQueue = new AsyncQueue();
15514
15796
  this.currentCreateOfferId = 0;
15515
15797
  this.metadata = { isMediaBypassEdge: false };
15798
+ this.id = `mc-${staticIdCounter++}`;
15516
15799
  this.options = Object.assign(Object.assign({}, defaultMultistreamConnectionOptions), userOptions);
15517
15800
  logger.info(`Creating multistream connection with options ${JSON.stringify(this.options)}`);
15518
15801
  this.metricsCallback = this.options.metricsCallback;
15802
+ this.startWorkerIfNeeded();
15519
15803
  this.initializePeerConnection();
15520
15804
  this.overuseStateManager = new OveruseStateManager((overuseState) => this.overuseUpdateCallback(overuseState));
15521
15805
  this.overuseStateManager.start();
@@ -15529,6 +15813,15 @@ class MultistreamConnection extends EventEmitter$2 {
15529
15813
  this.createSendTransceiver(exports.MediaType.VideoSlides, slidesSceneId, videoSlidesEncodingOptions);
15530
15814
  this.createSendTransceiver(exports.MediaType.AudioSlides, slidesSceneId);
15531
15815
  }
15816
+ startWorkerIfNeeded() {
15817
+ if (this.options.enableInboundAudioLevelMonitoring) {
15818
+ if (WebCapabilities.supportsEncodedStreamTransforms() !== CapabilityState.CAPABLE) {
15819
+ logger.warn('RTCRtpScriptTransform is not supported in this browser, skipping inbound audio level monitoring.');
15820
+ return;
15821
+ }
15822
+ getWorkerManager().startWorker();
15823
+ }
15824
+ }
15532
15825
  initializePeerConnection() {
15533
15826
  this.pc = new PeerConnection({
15534
15827
  iceServers: this.options.iceServers,
@@ -15852,6 +16145,10 @@ SCTP Max Message Size: ${maxMessageSize}`);
15852
16145
  });
15853
16146
  this.jmpSessions.forEach((jmpSession) => jmpSession.close());
15854
16147
  this.pc.close();
16148
+ getWorkerManager().sendRequestToWorker({
16149
+ type: MainMsgType.ClearMetadata,
16150
+ multistreamConnectionId: this.id,
16151
+ });
15855
16152
  }
15856
16153
  sendMediaRequestStatus(mediaType) {
15857
16154
  var _a;
@@ -15923,6 +16220,7 @@ SCTP Max Message Size: ${maxMessageSize}`);
15923
16220
  direction: 'recvonly',
15924
16221
  });
15925
16222
  const mid = this.midPredictor.getNextMid(mediaType);
16223
+ const doAudioLevelMonitoring = this.options.enableInboundAudioLevelMonitoring && mediaType === exports.MediaType.AudioMain;
15926
16224
  const munger = new IngressSdpMunger();
15927
16225
  const recvOnlyTransceiver = new ReceiveOnlyTransceiver({
15928
16226
  rtcRtpTransceiver,
@@ -15930,6 +16228,23 @@ SCTP Max Message Size: ${maxMessageSize}`);
15930
16228
  mediaType,
15931
16229
  munger,
15932
16230
  });
16231
+ if (doAudioLevelMonitoring) {
16232
+ const encodedTransformId = `INBOUND-${mediaType}-${mid}`;
16233
+ try {
16234
+ this.setupEncodedTransform(rtcRtpTransceiver.receiver, encodedTransformId, EncodedTransformType.AudioLevelMonitor);
16235
+ recvOnlyTransceiver.setEncodedStreamMetadataCallback(() => __awaiter(this, void 0, void 0, function* () {
16236
+ return getWorkerManager().sendRequestToWorker({
16237
+ type: MainMsgType.GetMetadata,
16238
+ mediaType,
16239
+ encodedTransformId,
16240
+ multistreamConnectionId: this.id,
16241
+ });
16242
+ }));
16243
+ }
16244
+ catch (e) {
16245
+ logger.warn(`Failed to setup encoded transform for ${encodedTransformId} audio level monitoring: ${e}`);
16246
+ }
16247
+ }
15933
16248
  if (getMediaFamily(mediaType) === exports.MediaFamily.Video) {
15934
16249
  recvOnlyTransceiver.setCodecParameters({
15935
16250
  'sps-pps-idr-in-keyframe': '1',
@@ -16418,6 +16733,21 @@ SCTP Max Message Size: ${maxMessageSize}`);
16418
16733
  videoSlides: this.getCsiByMediaType(exports.MediaType.VideoSlides),
16419
16734
  };
16420
16735
  }
16736
+ setupEncodedTransform(receiverOrSender, id, type) {
16737
+ if (WebCapabilities.supportsEncodedStreamTransforms() !== CapabilityState.CAPABLE) {
16738
+ throw new Error('encoded stream transport is not supported in this browser.');
16739
+ }
16740
+ if (!getWorkerManager().getWorker()) {
16741
+ throw new Error('no worker available for encoded stream transforms.');
16742
+ }
16743
+ const options = {
16744
+ type,
16745
+ encodedTransformId: id,
16746
+ multistreamConnectionId: this.id,
16747
+ };
16748
+ receiverOrSender.transform = new window.RTCRtpScriptTransform(getWorkerManager().getWorker(), options);
16749
+ logger.log(`started "${type}" transform with id=${id} on connection ${this.id}`);
16750
+ }
16421
16751
  }
16422
16752
 
16423
16753
  class StreamRequest {