608 lines
23 KiB
JavaScript
608 lines
23 KiB
JavaScript
/******/ (function(modules) { // webpackBootstrap
|
||
/******/ // The module cache
|
||
/******/ var installedModules = {};
|
||
/******/
|
||
/******/ // The require function
|
||
/******/ function __webpack_require__(moduleId) {
|
||
/******/
|
||
/******/ // Check if module is in cache
|
||
/******/ if(installedModules[moduleId]) {
|
||
/******/ return installedModules[moduleId].exports;
|
||
/******/ }
|
||
/******/ // Create a new module (and put it into the cache)
|
||
/******/ var module = installedModules[moduleId] = {
|
||
/******/ i: moduleId,
|
||
/******/ l: false,
|
||
/******/ exports: {}
|
||
/******/ };
|
||
/******/
|
||
/******/ // Execute the module function
|
||
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
||
/******/
|
||
/******/ // Flag the module as loaded
|
||
/******/ module.l = true;
|
||
/******/
|
||
/******/ // Return the exports of the module
|
||
/******/ return module.exports;
|
||
/******/ }
|
||
/******/
|
||
/******/
|
||
/******/ // expose the modules object (__webpack_modules__)
|
||
/******/ __webpack_require__.m = modules;
|
||
/******/
|
||
/******/ // expose the module cache
|
||
/******/ __webpack_require__.c = installedModules;
|
||
/******/
|
||
/******/ // define getter function for harmony exports
|
||
/******/ __webpack_require__.d = function(exports, name, getter) {
|
||
/******/ if(!__webpack_require__.o(exports, name)) {
|
||
/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
|
||
/******/ }
|
||
/******/ };
|
||
/******/
|
||
/******/ // define __esModule on exports
|
||
/******/ __webpack_require__.r = function(exports) {
|
||
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
|
||
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
||
/******/ }
|
||
/******/ Object.defineProperty(exports, '__esModule', { value: true });
|
||
/******/ };
|
||
/******/
|
||
/******/ // create a fake namespace object
|
||
/******/ // mode & 1: value is a module id, require it
|
||
/******/ // mode & 2: merge all properties of value into the ns
|
||
/******/ // mode & 4: return value when already ns object
|
||
/******/ // mode & 8|1: behave like require
|
||
/******/ __webpack_require__.t = function(value, mode) {
|
||
/******/ if(mode & 1) value = __webpack_require__(value);
|
||
/******/ if(mode & 8) return value;
|
||
/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
|
||
/******/ var ns = Object.create(null);
|
||
/******/ __webpack_require__.r(ns);
|
||
/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
|
||
/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
|
||
/******/ return ns;
|
||
/******/ };
|
||
/******/
|
||
/******/ // getDefaultExport function for compatibility with non-harmony modules
|
||
/******/ __webpack_require__.n = function(module) {
|
||
/******/ var getter = module && module.__esModule ?
|
||
/******/ function getDefault() { return module['default']; } :
|
||
/******/ function getModuleExports() { return module; };
|
||
/******/ __webpack_require__.d(getter, 'a', getter);
|
||
/******/ return getter;
|
||
/******/ };
|
||
/******/
|
||
/******/ // Object.prototype.hasOwnProperty.call
|
||
/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
|
||
/******/
|
||
/******/ // __webpack_public_path__
|
||
/******/ __webpack_require__.p = "";
|
||
/******/
|
||
/******/
|
||
/******/ // Load entry module and return exports
|
||
/******/ return __webpack_require__(__webpack_require__.s = 0);
|
||
/******/ })
|
||
/************************************************************************/
|
||
/******/ ([
|
||
/* 0 */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
// ESM COMPAT FLAG
|
||
__webpack_require__.r(__webpack_exports__);
|
||
|
||
// CONCATENATED MODULE: ./modules/e2ee/crypto-utils.js
|
||
/**
|
||
* Derives a set of keys from the master key.
|
||
* @param {CryptoKey} material - master key to derive from
|
||
*
|
||
* See https://tools.ietf.org/html/draft-omara-sframe-00#section-4.3.1
|
||
*/
|
||
async function deriveKeys(material) {
|
||
const info = new ArrayBuffer();
|
||
const textEncoder = new TextEncoder();
|
||
|
||
// https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/deriveKey#HKDF
|
||
// https://developer.mozilla.org/en-US/docs/Web/API/HkdfParams
|
||
const encryptionKey = await crypto.subtle.deriveKey({
|
||
name: 'HKDF',
|
||
salt: textEncoder.encode('JFrameEncryptionKey'),
|
||
hash: 'SHA-256',
|
||
info
|
||
}, material, {
|
||
name: 'AES-CTR',
|
||
length: 128
|
||
}, false, [ 'encrypt', 'decrypt' ]);
|
||
const authenticationKey = await crypto.subtle.deriveKey({
|
||
name: 'HKDF',
|
||
salt: textEncoder.encode('JFrameAuthenticationKey'),
|
||
hash: 'SHA-256',
|
||
info
|
||
}, material, {
|
||
name: 'HMAC',
|
||
hash: 'SHA-256'
|
||
}, false, [ 'sign' ]);
|
||
const saltKey = await crypto.subtle.deriveBits({
|
||
name: 'HKDF',
|
||
salt: textEncoder.encode('JFrameSaltKey'),
|
||
hash: 'SHA-256',
|
||
info
|
||
}, material, 128);
|
||
|
||
return {
|
||
material,
|
||
encryptionKey,
|
||
authenticationKey,
|
||
saltKey
|
||
};
|
||
}
|
||
|
||
/**
|
||
* Ratchets a key. See
|
||
* https://tools.ietf.org/html/draft-omara-sframe-00#section-4.3.5.1
|
||
* @param {CryptoKey} material - base key material
|
||
* @returns {ArrayBuffer} - ratcheted key material
|
||
*/
|
||
async function ratchet(material) {
|
||
const textEncoder = new TextEncoder();
|
||
|
||
// https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/deriveBits
|
||
return crypto.subtle.deriveBits({
|
||
name: 'HKDF',
|
||
salt: textEncoder.encode('JFrameRatchetKey'),
|
||
hash: 'SHA-256',
|
||
info: new ArrayBuffer()
|
||
}, material, 256);
|
||
}
|
||
|
||
/**
|
||
* Converts a raw key into a WebCrypto key object with default options
|
||
* suitable for our usage.
|
||
* @param {ArrayBuffer} keyBytes - raw key
|
||
* @param {Array} keyUsages - key usages, see importKey documentation
|
||
* @returns {CryptoKey} - the WebCrypto key.
|
||
*/
|
||
async function importKey(keyBytes) {
|
||
// https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/importKey
|
||
return crypto.subtle.importKey('raw', keyBytes, 'HKDF', false, [ 'deriveBits', 'deriveKey' ]);
|
||
}
|
||
|
||
// CONCATENATED MODULE: ./modules/e2ee/utils.js
|
||
/**
|
||
* Polyfill RTCEncoded(Audio|Video)Frame.getMetadata() (not available in M83, available M84+).
|
||
* The polyfill can not be done on the prototype since its not exposed in workers. Instead,
|
||
* it is done as another transformation to keep it separate.
|
||
* TODO: remove when we decode to drop M83 support.
|
||
*/
|
||
function polyFillEncodedFrameMetadata(encodedFrame, controller) {
|
||
if (!encodedFrame.getMetadata) {
|
||
encodedFrame.getMetadata = function() {
|
||
return {
|
||
// TODO: provide a more complete polyfill based on additionalData for video.
|
||
synchronizationSource: this.synchronizationSource,
|
||
contributingSources: this.contributingSources
|
||
};
|
||
};
|
||
}
|
||
controller.enqueue(encodedFrame);
|
||
}
|
||
|
||
/**
|
||
* Compares two byteArrays for equality.
|
||
*/
|
||
function isArrayEqual(a1, a2) {
|
||
if (a1.byteLength !== a2.byteLength) {
|
||
return false;
|
||
}
|
||
for (let i = 0; i < a1.byteLength; i++) {
|
||
if (a1[i] !== a2[i]) {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
|
||
// CONCATENATED MODULE: ./modules/e2ee/Context.js
|
||
/* eslint-disable no-bitwise */
|
||
/* global BigInt */
|
||
|
||
|
||
|
||
|
||
// We use a ringbuffer of keys so we can change them and still decode packets that were
|
||
// encrypted with an old key. We use a size of 16 which corresponds to the four bits
|
||
// in the frame trailer.
|
||
const keyRingSize = 16;
|
||
|
||
// We copy the first bytes of the VP8 payload unencrypted.
|
||
// For keyframes this is 10 bytes, for non-keyframes (delta) 3. See
|
||
// https://tools.ietf.org/html/rfc6386#section-9.1
|
||
// This allows the bridge to continue detecting keyframes (only one byte needed in the JVB)
|
||
// and is also a bit easier for the VP8 decoder (i.e. it generates funny garbage pictures
|
||
// instead of being unable to decode).
|
||
// This is a bit for show and we might want to reduce to 1 unconditionally in the final version.
|
||
//
|
||
// For audio (where frame.type is not set) we do not encrypt the opus TOC byte:
|
||
// https://tools.ietf.org/html/rfc6716#section-3.1
|
||
const unencryptedBytes = {
|
||
key: 10,
|
||
delta: 3,
|
||
undefined: 1 // frame.type is not set on audio
|
||
};
|
||
|
||
// Use truncated SHA-256 hashes, 80 bіts for video, 32 bits for audio.
|
||
// This follows the same principles as DTLS-SRTP.
|
||
const authenticationTagOptions = {
|
||
name: 'HMAC',
|
||
hash: 'SHA-256'
|
||
};
|
||
const digestLength = {
|
||
key: 10,
|
||
delta: 10,
|
||
undefined: 4 // frame.type is not set on audio
|
||
};
|
||
|
||
// Maximum number of forward ratchets to attempt when the authentication
|
||
// tag on a remote packet does not match the current key.
|
||
const ratchetWindow = 8;
|
||
|
||
/**
|
||
* Per-participant context holding the cryptographic keys and
|
||
* encode/decode functions
|
||
*/
|
||
class Context_Context {
|
||
/**
|
||
* @param {string} id - local muc resourcepart
|
||
*/
|
||
constructor(id) {
|
||
// An array (ring) of keys that we use for sending and receiving.
|
||
this._cryptoKeyRing = new Array(keyRingSize);
|
||
|
||
// A pointer to the currently used key.
|
||
this._currentKeyIndex = -1;
|
||
|
||
// A per-sender counter that is used create the AES CTR.
|
||
// Must be incremented on every frame that is sent, can be reset on
|
||
// key changes.
|
||
this._sendCount = BigInt(0); // eslint-disable-line new-cap
|
||
|
||
this._id = id;
|
||
}
|
||
|
||
/**
|
||
* Derives the different subkeys and starts using them for encryption or
|
||
* decryption.
|
||
* @param {Uint8Array|false} key bytes. Pass false to disable.
|
||
* @param {Number} keyIndex
|
||
*/
|
||
async setKey(keyBytes, keyIndex) {
|
||
let newKey;
|
||
|
||
if (keyBytes) {
|
||
const material = await importKey(keyBytes);
|
||
|
||
newKey = await deriveKeys(material);
|
||
} else {
|
||
newKey = false;
|
||
}
|
||
this._currentKeyIndex = keyIndex % this._cryptoKeyRing.length;
|
||
this._setKeys(newKey);
|
||
}
|
||
|
||
/**
|
||
* Sets a set of keys and resets the sendCount.
|
||
* decryption.
|
||
* @param {Object} keys set of keys.
|
||
* @private
|
||
*/
|
||
_setKeys(keys) {
|
||
this._cryptoKeyRing[this._currentKeyIndex] = keys;
|
||
this._sendCount = BigInt(0); // eslint-disable-line new-cap
|
||
}
|
||
|
||
/**
|
||
* Function that will be injected in a stream and will encrypt the given encoded frames.
|
||
*
|
||
* @param {RTCEncodedVideoFrame|RTCEncodedAudioFrame} encodedFrame - Encoded video frame.
|
||
* @param {TransformStreamDefaultController} controller - TransportStreamController.
|
||
*
|
||
* The packet format is a variant of
|
||
* https://tools.ietf.org/html/draft-omara-sframe-00
|
||
* using a trailer instead of a header. One of the design goals was to not require
|
||
* changes to the SFU which for video requires not encrypting the keyframe bit of VP8
|
||
* as SFUs need to detect a keyframe (framemarking or the generic frame descriptor will
|
||
* solve this eventually). This also "hides" that a client is using E2EE a bit.
|
||
*
|
||
* Note that this operates on the full frame, i.e. for VP8 the data described in
|
||
* https://tools.ietf.org/html/rfc6386#section-9.1
|
||
*
|
||
* The VP8 payload descriptor described in
|
||
* https://tools.ietf.org/html/rfc7741#section-4.2
|
||
* is part of the RTP packet and not part of the encoded frame and is therefore not
|
||
* controllable by us. This is fine as the SFU keeps having access to it for routing.
|
||
*/
|
||
encodeFunction(encodedFrame, controller) {
|
||
const keyIndex = this._currentKeyIndex;
|
||
|
||
if (this._cryptoKeyRing[keyIndex]) {
|
||
this._sendCount++;
|
||
|
||
// Thіs is not encrypted and contains the VP8 payload descriptor or the Opus TOC byte.
|
||
const frameHeader = new Uint8Array(encodedFrame.data, 0, unencryptedBytes[encodedFrame.type]);
|
||
|
||
// Construct frame trailer. Similar to the frame header described in
|
||
// https://tools.ietf.org/html/draft-omara-sframe-00#section-4.2
|
||
// but we put it at the end.
|
||
// 0 1 2 3 4 5 6 7
|
||
// ---------+---------------------------------+-+-+-+-+-+-+-+-+
|
||
// payload | CTR... (length=LEN) |S|LEN |KID |
|
||
// ---------+---------------------------------+-+-+-+-+-+-+-+-+
|
||
const counter = new Uint8Array(16);
|
||
const counterView = new DataView(counter.buffer);
|
||
|
||
// The counter is encoded as a variable-length field.
|
||
counterView.setBigUint64(8, this._sendCount);
|
||
let counterLength = 8;
|
||
|
||
for (let i = 8; i < counter.byteLength; i++ && counterLength--) {
|
||
if (counterView.getUint8(i) !== 0) {
|
||
break;
|
||
}
|
||
}
|
||
|
||
const frameTrailer = new Uint8Array(counterLength + 1);
|
||
|
||
frameTrailer.set(new Uint8Array(counter.buffer, counter.byteLength - counterLength));
|
||
|
||
// Since we never send a counter of 0 we send counterLength - 1 on the wire.
|
||
// This is different from the sframe draft, increases the key space and lets us
|
||
// ignore the case of a zero-length counter at the receiver.
|
||
frameTrailer[frameTrailer.byteLength - 1] = keyIndex | ((counterLength - 1) << 4);
|
||
|
||
// XOR the counter with the saltKey to construct the AES CTR.
|
||
const saltKey = new DataView(this._cryptoKeyRing[keyIndex].saltKey);
|
||
|
||
for (let i = 0; i < counter.byteLength; i++) {
|
||
counterView.setUint8(i, counterView.getUint8(i) ^ saltKey.getUint8(i));
|
||
}
|
||
|
||
return crypto.subtle.encrypt({
|
||
name: 'AES-CTR',
|
||
counter,
|
||
length: 64
|
||
}, this._cryptoKeyRing[keyIndex].encryptionKey, new Uint8Array(encodedFrame.data,
|
||
unencryptedBytes[encodedFrame.type]))
|
||
.then(cipherText => {
|
||
const newData = new ArrayBuffer(frameHeader.byteLength + cipherText.byteLength
|
||
+ digestLength[encodedFrame.type] + frameTrailer.byteLength);
|
||
const newUint8 = new Uint8Array(newData);
|
||
|
||
newUint8.set(frameHeader); // copy first bytes.
|
||
newUint8.set(new Uint8Array(cipherText), unencryptedBytes[encodedFrame.type]); // add ciphertext.
|
||
// Leave some space for the authentication tag. This is filled with 0s initially, similar to
|
||
// STUN message-integrity described in https://tools.ietf.org/html/rfc5389#section-15.4
|
||
newUint8.set(frameTrailer, frameHeader.byteLength + cipherText.byteLength
|
||
+ digestLength[encodedFrame.type]); // append trailer.
|
||
|
||
return crypto.subtle.sign(authenticationTagOptions, this._cryptoKeyRing[keyIndex].authenticationKey,
|
||
new Uint8Array(newData)).then(authTag => {
|
||
// Set the truncated authentication tag.
|
||
newUint8.set(new Uint8Array(authTag, 0, digestLength[encodedFrame.type]),
|
||
unencryptedBytes[encodedFrame.type] + cipherText.byteLength);
|
||
encodedFrame.data = newData;
|
||
|
||
return controller.enqueue(encodedFrame);
|
||
});
|
||
}, e => {
|
||
// TODO: surface this to the app.
|
||
console.error(e);
|
||
|
||
// We are not enqueuing the frame here on purpose.
|
||
});
|
||
}
|
||
|
||
/* NOTE WELL:
|
||
* This will send unencrypted data (only protected by DTLS transport encryption) when no key is configured.
|
||
* This is ok for demo purposes but should not be done once this becomes more relied upon.
|
||
*/
|
||
controller.enqueue(encodedFrame);
|
||
}
|
||
|
||
/**
|
||
* Function that will be injected in a stream and will decrypt the given encoded frames.
|
||
*
|
||
* @param {RTCEncodedVideoFrame|RTCEncodedAudioFrame} encodedFrame - Encoded video frame.
|
||
* @param {TransformStreamDefaultController} controller - TransportStreamController.
|
||
*/
|
||
async decodeFunction(encodedFrame, controller) {
|
||
const data = new Uint8Array(encodedFrame.data);
|
||
const keyIndex = data[encodedFrame.data.byteLength - 1] & 0xf; // lower four bits.
|
||
|
||
if (this._cryptoKeyRing[keyIndex]) {
|
||
const counterLength = 1 + ((data[encodedFrame.data.byteLength - 1] >> 4) & 0x7);
|
||
const frameHeader = new Uint8Array(encodedFrame.data, 0, unencryptedBytes[encodedFrame.type]);
|
||
|
||
// Extract the truncated authentication tag.
|
||
const authTagOffset = encodedFrame.data.byteLength - (digestLength[encodedFrame.type]
|
||
+ counterLength + 1);
|
||
const authTag = encodedFrame.data.slice(authTagOffset, authTagOffset
|
||
+ digestLength[encodedFrame.type]);
|
||
|
||
// Set authentication tag bytes to 0.
|
||
const zeros = new Uint8Array(digestLength[encodedFrame.type]);
|
||
|
||
data.set(zeros, encodedFrame.data.byteLength - (digestLength[encodedFrame.type] + counterLength + 1));
|
||
|
||
// Do truncated hash comparison. If the hash does not match we might have to advance the
|
||
// ratchet a limited number of times. See (even though the description there is odd)
|
||
// https://tools.ietf.org/html/draft-omara-sframe-00#section-4.3.5.1
|
||
let { authenticationKey, material } = this._cryptoKeyRing[keyIndex];
|
||
let valid = false;
|
||
let newKeys = null;
|
||
|
||
for (let distance = 0; distance < ratchetWindow; distance++) {
|
||
const calculatedTag = await crypto.subtle.sign(authenticationTagOptions,
|
||
authenticationKey, encodedFrame.data);
|
||
|
||
if (isArrayEqual(new Uint8Array(authTag),
|
||
new Uint8Array(calculatedTag.slice(0, digestLength[encodedFrame.type])))) {
|
||
valid = true;
|
||
if (distance > 0) {
|
||
this._setKeys(newKeys);
|
||
}
|
||
break;
|
||
}
|
||
|
||
// Attempt to ratchet and generate the next set of keys.
|
||
material = await importKey(await ratchet(material));
|
||
newKeys = await deriveKeys(material);
|
||
authenticationKey = newKeys.authenticationKey;
|
||
}
|
||
|
||
// Check whether we found a valid signature.
|
||
if (!valid) {
|
||
// TODO: return an error to the app.
|
||
|
||
console.error('Authentication tag mismatch');
|
||
|
||
return;
|
||
}
|
||
|
||
// Extract the counter.
|
||
const counter = new Uint8Array(16);
|
||
|
||
counter.set(data.slice(encodedFrame.data.byteLength - (counterLength + 1),
|
||
encodedFrame.data.byteLength - 1), 16 - counterLength);
|
||
const counterView = new DataView(counter.buffer);
|
||
|
||
// XOR the counter with the saltKey to construct the AES CTR.
|
||
const saltKey = new DataView(this._cryptoKeyRing[keyIndex].saltKey);
|
||
|
||
for (let i = 0; i < counter.byteLength; i++) {
|
||
counterView.setUint8(i,
|
||
counterView.getUint8(i) ^ saltKey.getUint8(i));
|
||
}
|
||
|
||
return crypto.subtle.decrypt({
|
||
name: 'AES-CTR',
|
||
counter,
|
||
length: 64
|
||
}, this._cryptoKeyRing[keyIndex].encryptionKey, new Uint8Array(encodedFrame.data,
|
||
unencryptedBytes[encodedFrame.type],
|
||
encodedFrame.data.byteLength - (unencryptedBytes[encodedFrame.type]
|
||
+ digestLength[encodedFrame.type] + counterLength + 1))
|
||
).then(plainText => {
|
||
const newData = new ArrayBuffer(unencryptedBytes[encodedFrame.type] + plainText.byteLength);
|
||
const newUint8 = new Uint8Array(newData);
|
||
|
||
newUint8.set(frameHeader);
|
||
newUint8.set(new Uint8Array(plainText), unencryptedBytes[encodedFrame.type]);
|
||
encodedFrame.data = newData;
|
||
|
||
return controller.enqueue(encodedFrame);
|
||
}, e => {
|
||
console.error(e);
|
||
|
||
// TODO: notify the application about error status.
|
||
// TODO: For video we need a better strategy since we do not want to based any
|
||
// non-error frames on a garbage keyframe.
|
||
if (encodedFrame.type === undefined) { // audio, replace with silence.
|
||
const newData = new ArrayBuffer(3);
|
||
const newUint8 = new Uint8Array(newData);
|
||
|
||
newUint8.set([ 0xd8, 0xff, 0xfe ]); // opus silence frame.
|
||
encodedFrame.data = newData;
|
||
controller.enqueue(encodedFrame);
|
||
}
|
||
});
|
||
} else if (keyIndex >= this._cryptoKeyRing.length && this._cryptoKeyRing[this._currentKeyIndex]) {
|
||
// If we are encrypting but don't have a key for the remote drop the frame.
|
||
// This is a heuristic since we don't know whether a packet is encrypted,
|
||
// do not have a checksum and do not have signaling for whether a remote participant does
|
||
// encrypt or not.
|
||
return;
|
||
}
|
||
|
||
// TODO: this just passes through to the decoder. Is that ok? If we don't know the key yet
|
||
// we might want to buffer a bit but it is still unclear how to do that (and for how long etc).
|
||
controller.enqueue(encodedFrame);
|
||
}
|
||
}
|
||
|
||
// CONCATENATED MODULE: ./modules/e2ee/Worker.js
|
||
/* global TransformStream */
|
||
/* eslint-disable no-bitwise */
|
||
|
||
// Worker for E2EE/Insertable streams.
|
||
//
|
||
|
||
|
||
|
||
|
||
const contexts = new Map(); // Map participant id => context
|
||
|
||
onmessage = async event => {
|
||
const { operation } = event.data;
|
||
|
||
if (operation === 'encode') {
|
||
const { readableStream, writableStream, participantId } = event.data;
|
||
|
||
if (!contexts.has(participantId)) {
|
||
contexts.set(participantId, new Context_Context(participantId));
|
||
}
|
||
const context = contexts.get(participantId);
|
||
const transformStream = new TransformStream({
|
||
transform: context.encodeFunction.bind(context)
|
||
});
|
||
|
||
readableStream
|
||
.pipeThrough(new TransformStream({
|
||
transform: polyFillEncodedFrameMetadata // M83 polyfill.
|
||
}))
|
||
.pipeThrough(transformStream)
|
||
.pipeTo(writableStream);
|
||
} else if (operation === 'decode') {
|
||
const { readableStream, writableStream, participantId } = event.data;
|
||
|
||
if (!contexts.has(participantId)) {
|
||
contexts.set(participantId, new Context_Context(participantId));
|
||
}
|
||
const context = contexts.get(participantId);
|
||
const transformStream = new TransformStream({
|
||
transform: context.decodeFunction.bind(context)
|
||
});
|
||
|
||
readableStream
|
||
.pipeThrough(new TransformStream({
|
||
transform: polyFillEncodedFrameMetadata // M83 polyfill.
|
||
}))
|
||
.pipeThrough(transformStream)
|
||
.pipeTo(writableStream);
|
||
} else if (operation === 'setKey') {
|
||
const { participantId, key, keyIndex } = event.data;
|
||
|
||
if (!contexts.has(participantId)) {
|
||
contexts.set(participantId, new Context_Context(participantId));
|
||
}
|
||
const context = contexts.get(participantId);
|
||
|
||
if (key) {
|
||
context.setKey(key, keyIndex);
|
||
} else {
|
||
context.setKey(false, keyIndex);
|
||
}
|
||
} else if (operation === 'cleanup') {
|
||
const { participantId } = event.data;
|
||
|
||
contexts.delete(participantId);
|
||
} else {
|
||
console.error('e2ee worker', operation);
|
||
}
|
||
};
|
||
|
||
|
||
/***/ })
|
||
/******/ ]); |