summaryrefslogtreecommitdiffstats
path: root/testing/web-platform/tests/webrtc-encoded-transform
diff options
context:
space:
mode:
Diffstat (limited to 'testing/web-platform/tests/webrtc-encoded-transform')
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/META.yml6
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/helper.js26
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/idlharness.https.window.js18
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/routines.js93
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform-worker.js30
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform.https.html69
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-change-transform-worker.js39
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-change-transform.https.html60
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-late-transform.https.html94
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform-worker.js45
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform.https.html300
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame-simulcast.https.html136
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.https.html229
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.js70
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.https.html110
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.js63
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform-worker.js33
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-transform.https.html60
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform-worker.js22
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform.https.html62
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/set-metadata.https.html53
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-keys.https.html69
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-buffer-source.html50
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-in-worker.https.html61
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-readable.html19
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-worker.js7
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/sframe-transform.html141
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-clone.https.html51
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-receive-cloned.https.html52
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-send-incoming.https.html63
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-serviceworker-failure.https.html60
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-clone.https.html61
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-serviceworker-failure.https.html60
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-audio.https.html228
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-errors.https.html46
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-simulcast.https.html89
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video-frames.https.html80
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video.https.html182
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-worker.https.html196
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams.js262
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-sender-worker-single-frame.js19
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-worker-transform.js22
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/blank.html2
-rw-r--r--testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/serviceworker-failure.js30
44 files changed, 3468 insertions, 0 deletions
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/META.yml b/testing/web-platform/tests/webrtc-encoded-transform/META.yml
new file mode 100644
index 0000000000..8947732b6f
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/META.yml
@@ -0,0 +1,6 @@
+spec: https://w3c.github.io/webrtc-encoded-transform/
+suggested_reviewers:
+ - alvestrand
+ - guidou
+ - youennf
+ - jan-ivar
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/helper.js b/testing/web-platform/tests/webrtc-encoded-transform/helper.js
new file mode 100644
index 0000000000..d4cec39ffc
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/helper.js
@@ -0,0 +1,26 @@
+"use strict";
+
+async function setupLoopbackWithCodecAndGetReader(t, codec) {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const transceiver = caller.addTransceiver(videoTrack);
+ const codecCapability =
+ RTCRtpSender.getCapabilities('video').codecs.find(capability => {
+ return capability.mimeType.includes(codec);
+ });
+ assert_not_equals(codecCapability, undefined);
+ transceiver.setCodecPreferences([codecCapability]);
+
+ const senderStreams = transceiver.sender.createEncodedStreams();
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+ return senderStreams.readable.getReader();
+}
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/idlharness.https.window.js b/testing/web-platform/tests/webrtc-encoded-transform/idlharness.https.window.js
new file mode 100644
index 0000000000..2c6ef19ca8
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/idlharness.https.window.js
@@ -0,0 +1,18 @@
+// META: script=/resources/WebIDLParser.js
+// META: script=/resources/idlharness.js
+// META: script=./RTCPeerConnection-helper.js
+
+'use strict';
+
+idl_test(
+ ['webrtc-encoded-transform'],
+ ['webrtc', 'streams', 'html', 'dom'],
+ async idlArray => {
+ idlArray.add_objects({
+ // TODO: RTCEncodedVideoFrame
+ // TODO: RTCEncodedAudioFrame
+ RTCRtpSender: [`new RTCPeerConnection().addTransceiver('audio').sender`],
+ RTCRtpReceiver: [`new RTCPeerConnection().addTransceiver('audio').receiver`],
+ });
+ }
+);
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/routines.js b/testing/web-platform/tests/webrtc-encoded-transform/routines.js
new file mode 100644
index 0000000000..0d3e2b9b28
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/routines.js
@@ -0,0 +1,93 @@
+async function getNextMessage(portOrWorker) {
+ return new Promise(resolve => {
+ const resolveWithData = event => resolve(event.data);
+ const rejectWithData = event => reject(event.data);
+ portOrWorker.addEventListener('message', resolveWithData, {once: true});
+ portOrWorker.addEventListener('messageerror', rejectWithData, {once: true});
+ });
+}
+
+
+async function postMethod(port, method, options) {
+ port.postMessage(Object.assign({method}, options));
+ return await getNextMessage(port);
+}
+
+async function createWorker(script) {
+ const worker = new Worker(script);
+ const data = await getNextMessage(worker);
+ assert_equals(data, "registered");
+ return worker;
+}
+
+async function createTransform(worker) {
+ const channel = new MessageChannel;
+ const transform = new RTCRtpScriptTransform(worker, {name:'MockRTCRtpTransform', port: channel.port2}, [channel.port2]);
+ transform.port = channel.port1;
+ channel.port1.start();
+ assert_equals(await getNextMessage(channel.port1), "started");
+ return transform;
+}
+
+async function createTransforms(script) {
+ const worker = await createWorker(script)
+ return Promise.all([createTransform(worker), createTransform(worker)]);
+}
+
+async function createConnectionWithTransform(test, script, gumOptions) {
+ const [senderTransform, receiverTransform] = await createTransforms(script);
+
+ const localStream = await navigator.mediaDevices.getUserMedia(gumOptions);
+
+ let senderPc, receiverPc, sender, receiver;
+
+ await createConnections(test, (firstConnection) => {
+ senderPc = firstConnection;
+ sender = firstConnection.addTrack(localStream.getTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ receiverPc = secondConnection;
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ };
+ });
+
+ assert_true(!!sender, "sender should be set");
+ assert_true(!!receiver, "receiver should be set");
+
+ return {sender, receiver, senderPc, receiverPc};
+}
+
+async function createConnections(test, setupLocalConnection, setupRemoteConnection, doNotCloseAutmoatically) {
+ const localConnection = new RTCPeerConnection();
+ const remoteConnection = new RTCPeerConnection();
+
+ remoteConnection.onicecandidate = (event) => { localConnection.addIceCandidate(event.candidate); };
+ localConnection.onicecandidate = (event) => { remoteConnection.addIceCandidate(event.candidate); };
+
+ await setupLocalConnection(localConnection);
+ await setupRemoteConnection(remoteConnection);
+
+ const offer = await localConnection.createOffer();
+ await localConnection.setLocalDescription(offer);
+ await remoteConnection.setRemoteDescription(offer);
+
+ const answer = await remoteConnection.createAnswer();
+ await remoteConnection.setLocalDescription(answer);
+ await localConnection.setRemoteDescription(answer);
+
+ if (!doNotCloseAutmoatically) {
+ test.add_cleanup(() => {
+ localConnection.close();
+ remoteConnection.close();
+ });
+ }
+
+ return [localConnection, remoteConnection];
+}
+
+function waitFor(test, duration)
+{
+ return new Promise((resolve) => test.step_timeout(resolve, duration));
+}
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform-worker.js
new file mode 100644
index 0000000000..7cb43713d3
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform-worker.js
@@ -0,0 +1,30 @@
+class MockRTCRtpTransformer {
+ constructor(transformer) {
+ this.context = transformer;
+ this.start();
+ }
+ start()
+ {
+ this.reader = this.context.readable.getReader();
+ this.writer = this.context.writable.getWriter();
+ this.process();
+ this.context.options.port.postMessage("started " + this.context.options.mediaType + " " + this.context.options.side);
+ }
+
+ process()
+ {
+ this.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+
+ this.writer.write(chunk.value);
+ this.process();
+ });
+ }
+};
+
+onrtctransform = (event) => {
+ new MockRTCRtpTransformer(event.transformer);
+};
+
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform.https.html
new file mode 100644
index 0000000000..0d7f401582
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-audio-transform.https.html
@@ -0,0 +1,69 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="video" autoplay playsInline></video>
+ <script src="routines.js"></script>
+ <script>
+function waitForMessage(test, port, data)
+{
+ let gotMessage;
+ const promise = new Promise((resolve, reject) => {
+ gotMessage = resolve;
+ test.step_timeout(() => { reject("did not get " + data) }, 5000);
+ });
+ port.onmessage = event => {
+ if (event.data === data)
+ gotMessage();
+ };
+ return promise;
+}
+
+promise_test(async (test) => {
+ worker = new Worker("script-audio-transform-worker.js");
+ const data = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ assert_equals(data, "registered");
+
+ await setMediaPermission("granted", ["microphone"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({audio: true});
+
+ const senderChannel = new MessageChannel;
+ const receiverChannel = new MessageChannel;
+ const senderTransform = new RTCRtpScriptTransform(worker, {name:'MockRTCRtpTransform', mediaType:'audio', side:'sender', port:senderChannel.port2}, [senderChannel.port2]);
+ const receiverTransform = new RTCRtpScriptTransform(worker, {name:'MockRTCRtpTransform', mediaType:'audio', side:'receiver', port:receiverChannel.port2}, [receiverChannel.port2]);
+ senderTransform.port = senderChannel.port1;
+ receiverTransform.port = receiverChannel.port1;
+
+ promise1 = waitForMessage(test, senderTransform.port, "started audio sender");
+ promise2 = waitForMessage(test, receiverTransform.port, "started audio receiver");
+
+ const stream = await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ sender = firstConnection.addTrack(localStream.getAudioTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ await promise1;
+ await promise2;
+
+ video.srcObject = stream;
+ return video.play();
+});
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform-worker.js
new file mode 100644
index 0000000000..84a7aaac18
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform-worker.js
@@ -0,0 +1,39 @@
+function appendToBuffer(buffer, value) {
+ const result = new ArrayBuffer(buffer.byteLength + 1);
+ const byteResult = new Uint8Array(result);
+ byteResult.set(new Uint8Array(buffer), 0);
+ byteResult[buffer.byteLength] = value;
+ return result;
+}
+
+onrtctransform = (event) => {
+ const transformer = event.transformer;
+
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+ if (transformer.options.name === 'sender1')
+ chunk.value.data = appendToBuffer(chunk.value.data, 1);
+ else if (transformer.options.name === 'sender2')
+ chunk.value.data = appendToBuffer(chunk.value.data, 2);
+ else {
+ const value = new Uint8Array(chunk.value.data)[chunk.value.data.byteLength - 1];
+ if (value !== 1 && value !== 2)
+ self.postMessage("unexpected value: " + value);
+ else if (value === 2)
+ self.postMessage("got value 2");
+ chunk.value.data = chunk.value.data.slice(0, chunk.value.data.byteLength - 1);
+ }
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+
+ process(transformer);
+};
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform.https.html
new file mode 100644
index 0000000000..1bb0398dc5
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-change-transform.https.html
@@ -0,0 +1,60 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="video1" autoplay controls playsinline></video>
+ <script src ="routines.js"></script>
+ <script>
+async function waitForMessage(worker, data)
+{
+ while (true) {
+ const received = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ if (data === received)
+ return;
+ }
+}
+
+promise_test(async (test) => {
+ worker = new Worker('script-change-transform-worker.js');
+ const data = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ assert_equals(data, "registered");
+
+ await setMediaPermission("granted", ["camera"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({video: true});
+
+ let sender, receiver;
+ const senderTransform1 = new RTCRtpScriptTransform(worker, {name:'sender1'});
+ const senderTransform2 = new RTCRtpScriptTransform(worker, {name:'sender2'});
+ const receiverTransform = new RTCRtpScriptTransform(worker, {name:'receiver'});
+
+ const stream = await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ sender = firstConnection.addTrack(localStream.getVideoTracks()[0], localStream);
+ sender.transform = senderTransform1;
+ }, (secondConnection) => {
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ video1.srcObject = stream;
+ await video1.play();
+
+ const updatePromise = new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ sender.transform = senderTransform2;
+ assert_equals(await updatePromise, "got value 2");
+}, "change sender transform");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-late-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-late-transform.https.html
new file mode 100644
index 0000000000..726852bad9
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-late-transform.https.html
@@ -0,0 +1,94 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video controls id="video" autoplay></video>
+ <canvas id="canvas" width="640" height="480"></canvas>
+ <script src ="routines.js"></script>
+ <script>
+function grabFrameData(x, y, w, h)
+{
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+
+ canvas.getContext('2d').drawImage(video, x, y, w, h, x, y, w, h);
+ return canvas.getContext('2d').getImageData(x, y, w, h).data;
+}
+
+function getCircleImageData()
+{
+ return grabFrameData(450, 100, 150, 100);
+}
+
+async function checkVideoIsUpdated(test, shouldBeUpdated, count, referenceData)
+{
+ if (count === undefined)
+ count = 0;
+ else if (count >= 20)
+ return Promise.reject("checkVideoIsUpdated timed out :" + shouldBeUpdated + " " + count);
+
+ if (referenceData === undefined)
+ referenceData = getCircleImageData();
+
+ await waitFor(test, 200);
+ const newData = getCircleImageData();
+
+ if (shouldBeUpdated === (JSON.stringify(referenceData) !== JSON.stringify(newData)))
+ return;
+
+ await checkVideoIsUpdated(test, shouldBeUpdated, ++count, newData);
+}
+
+promise_test(async (test) => {
+ await setMediaPermission("granted", ["camera"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({video: true});
+ const senderTransform = new SFrameTransform({ compatibilityMode: "H264" });
+ const receiverTransform = new SFrameTransform({ compatibilityMode: "H264" });
+ await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]).then(key => {
+ senderTransform.setEncryptionKey(key);
+ receiverTransform.setEncryptionKey(key);
+ });
+
+ let sender, receiver;
+ const stream = await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ pc1 = firstConnection;
+ sender = firstConnection.addTrack(localStream.getVideoTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ pc2 = secondConnection;
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ // we do not set the receiver transform here;
+ resolve(trackEvent.streams[0]);
+ };
+ }, {
+ observeOffer : (offer) => {
+ const lines = offer.sdp.split('\r\n');
+ const h264Lines = lines.filter(line => line.indexOf("a=fmtp") === 0 && line.indexOf("42e01f") !== -1);
+ const baselineNumber = h264Lines[0].substring(6).split(' ')[0];
+ offer.sdp = lines.filter(line => {
+ return (line.indexOf('a=fmtp') === -1 && line.indexOf('a=rtcp-fb') === -1 && line.indexOf('a=rtpmap') === -1) || line.indexOf(baselineNumber) !== -1;
+ }).join('\r\n');
+ }
+ });
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ video.srcObject = stream;
+ video.play();
+
+ // We set the receiver transform here so that the decoder probably tried to decode sframe content.
+ test.step_timeout(() => receiver.transform = receiverTransform, 50);
+ await checkVideoIsUpdated(test, true);
+}, "video exchange with late receiver transform");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform-worker.js
new file mode 100644
index 0000000000..40f7e547d7
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform-worker.js
@@ -0,0 +1,45 @@
+onrtctransform = (event) => {
+ const transformer = event.transformer;
+
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+
+ async function waitForDetachAndPostMetadata(frame) {
+ while (true) {
+ if (frame.data.byteLength == 0) {
+ // frame.data has been detached! Verify metadata is still there.
+ self.postMessage({
+ name: `${transformer.options.name} after write`,
+ timestamp: frame.timestamp, type: frame.type,
+ metadata: frame.getMetadata()
+ });
+ return;
+ }
+ await new Promise(r => setTimeout(r, 100));
+ }
+ }
+
+ let isFirstFrame = true;
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+
+ if (isFirstFrame) {
+ isFirstFrame = false;
+ self.postMessage({
+ name: transformer.options.name,
+ timestamp: chunk.value.timestamp,
+ type: chunk.value.type,
+ metadata: chunk.value.getMetadata()
+ });
+ waitForDetachAndPostMetadata(chunk.value);
+ }
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+ process(transformer);
+};
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform.https.html
new file mode 100644
index 0000000000..11c88b4693
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-metadata-transform.https.html
@@ -0,0 +1,300 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset='utf-8'>
+ <meta name='timeout' content='long'>
+<script src='/resources/testharness.js'></script>
+ <script src='/resources/testharnessreport.js'></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id='video1' autoplay></video>
+ <script src ='routines.js'></script>
+ <script>
+async function waitForMessage(worker, data)
+{
+ while (true) {
+ const received = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ if (data === received)
+ return;
+ }
+}
+
+async function gatherMetadata(test, kind)
+{
+ worker = new Worker('script-metadata-transform-worker.js');
+ const data = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ assert_equals(data, 'registered');
+
+ // Both audio and vido are needed at one time or another
+ // so asking for both permissions
+ await setMediaPermission();
+ const localStream = await navigator.mediaDevices.getUserMedia({[kind]: true});
+
+ let sender, receiver;
+ const senderTransform = new RTCRtpScriptTransform(worker, {name:'sender'});
+ const receiverTransform = new RTCRtpScriptTransform(worker, {name:'receiver'});
+
+ await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ pc1 = firstConnection;
+ sender = firstConnection.addTrack(localStream.getTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ pc2 = secondConnection;
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+ test.step_timeout(() => reject('Test timed out'), 5000);
+ });
+
+ let senderBeforeWrite, senderAfterWrite, receiverBeforeWrite, receiverAfterWrite;
+ while (true) {
+ const {data} = await new Promise(r => worker.onmessage = r);
+ if (data.name == 'sender') {
+ senderBeforeWrite = data;
+ } else if (data.name == 'receiver') {
+ receiverBeforeWrite = data;
+ } else if (data.name == 'sender after write') {
+ senderAfterWrite = data;
+ } else if (data.name == 'receiver after write') {
+ receiverAfterWrite = data;
+ }
+ if (senderBeforeWrite &&
+ senderAfterWrite &&
+ receiverBeforeWrite &&
+ receiverAfterWrite) {
+ return {
+ senderBeforeWrite,
+ senderAfterWrite,
+ receiverBeforeWrite,
+ receiverAfterWrite
+ };
+ }
+ }
+}
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'audio');
+
+ assert_equals(typeof data.senderBeforeWrite.timestamp, 'number');
+ assert_not_equals(data.senderBeforeWrite.timestamp, 0);
+ assert_equals(data.senderBeforeWrite.timestamp,
+ data.senderAfterWrite.timestamp,
+ 'timestamp matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.timestamp,
+ data.receiverBeforeWrite.timestamp,
+ 'timestamp matches (for sender and receiver)');
+ assert_equals(data.receiverBeforeWrite.timestamp,
+ data.receiverAfterWrite.timestamp,
+ 'timestamp matches (for receiver before and after write)');
+}, 'audio metadata: timestamp');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'audio');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.synchronizationSource, 'number');
+ assert_not_equals(data.senderBeforeWrite.metadata.synchronizationSource, 0);
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.senderAfterWrite.metadata.synchronizationSource,
+ 'ssrc matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.receiverBeforeWrite.metadata.synchronizationSource,
+ 'ssrc matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.receiverAfterWrite.metadata.synchronizationSource,
+ 'ssrc matches (for receiver before and after write)');
+}, 'audio metadata: synchronizationSource');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'audio');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.payloadType, 'number');
+ assert_equals(data.senderBeforeWrite.metadata.payloadType,
+ data.senderAfterWrite.metadata.payloadType,
+ 'payload type matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.payloadType,
+ data.receiverBeforeWrite.metadata.payloadType,
+ 'payload type matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.payloadType,
+ data.receiverAfterWrite.metadata.payloadType,
+ 'payload type matches (for receiver before and after write)');
+}, 'audio metadata: payloadType');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'audio');
+
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.senderAfterWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match (for sender before and after write)');
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.receiverBeforeWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match');
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.receiverAfterWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match (for receiver before and after write)');
+}, 'audio metadata: contributingSources');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'audio');
+
+ assert_equals(typeof data.receiverBeforeWrite.metadata.sequenceNumber,
+ 'number');
+ assert_equals(data.receiverBeforeWrite.metadata.sequenceNumber,
+ data.receiverAfterWrite.metadata.sequenceNumber,
+ 'sequenceNumber matches (for receiver before and after write)');
+ // spec says sequenceNumber exists only for incoming audio frames
+ assert_equals(data.senderBeforeWrite.metadata.sequenceNumber, undefined);
+ assert_equals(data.senderAfterWrite.metadata.sequenceNumber, undefined);
+}, 'audio metadata: sequenceNumber');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.timestamp, 'number');
+ assert_equals(data.senderBeforeWrite.timestamp,
+ data.senderAfterWrite.timestamp,
+ 'timestamp matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.timestamp,
+ data.receiverBeforeWrite.timestamp,
+ 'timestamp matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.timestamp,
+ data.receiverAfterWrite.timestamp,
+ 'timestamp matches (for receiver before and after write)');
+}, 'video metadata: timestamp');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.synchronizationSource,
+ 'number');
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.senderAfterWrite.metadata.synchronizationSource,
+ 'ssrc matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.receiverBeforeWrite.metadata.synchronizationSource,
+ 'ssrc matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.synchronizationSource,
+ data.receiverAfterWrite.metadata.synchronizationSource,
+ 'ssrc matches (for receiver before and after write)');
+}, 'video metadata: ssrc');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.senderAfterWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match (for sender before and after write)');
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.receiverBeforeWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match');
+ assert_array_equals(data.senderBeforeWrite.metadata.contributingSources,
+ data.receiverAfterWrite.metadata.contributingSources,
+ 'csrcs are arrays, and match (for receiver before and after write)');
+}, 'video metadata: csrcs');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.height, 'number');
+ assert_equals(data.senderBeforeWrite.metadata.height,
+ data.senderAfterWrite.metadata.height,
+ 'height matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.height,
+ data.receiverBeforeWrite.metadata.height,
+ 'height matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.height,
+ data.receiverAfterWrite.metadata.height,
+ 'height matches (for receiver before and after write)');
+ assert_equals(typeof data.senderBeforeWrite.metadata.width, 'number');
+ assert_equals(data.senderBeforeWrite.metadata.width,
+ data.senderAfterWrite.metadata.width,
+ 'width matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.width,
+ data.receiverBeforeWrite.metadata.width,
+ 'width matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.width,
+ data.receiverAfterWrite.metadata.width,
+ 'width matches (for receiver before and after write)');
+}, 'video metadata: width and height');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.spatialIndex,
+ 'number');
+ assert_equals(data.senderBeforeWrite.metadata.spatialIndex,
+ data.senderAfterWrite.metadata.spatialIndex,
+ 'spatialIndex matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.spatialIndex,
+ data.receiverBeforeWrite.metadata.spatialIndex,
+ 'spatialIndex matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.spatialIndex,
+ data.receiverAfterWrite.metadata.spatialIndex,
+ 'spatialIndex matches (for receiver before and after write)');
+ assert_equals(typeof data.senderBeforeWrite.metadata.temporalIndex,
+ 'number');
+ assert_equals(data.senderBeforeWrite.metadata.temporalIndex,
+ data.senderAfterWrite.metadata.temporalIndex,
+ 'temporalIndex matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.temporalIndex,
+ data.receiverBeforeWrite.metadata.temporalIndex,
+ 'temporalIndex matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.temporalIndex,
+ data.receiverAfterWrite.metadata.temporalIndex,
+ 'temporalIndex matches (for receiver before and after write)');
+}, 'video metadata: spatial and temporal index');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_array_equals(data.senderBeforeWrite.metadata.dependencies,
+ data.senderAfterWrite.metadata.dependencies,
+ 'dependencies are arrays, and match (for sender before and after write)');
+ assert_array_equals(data.senderBeforeWrite.metadata.dependencies,
+ data.receiverBeforeWrite.metadata.dependencies,
+ 'dependencies are arrays, and match (for sender and receiver)');
+ assert_array_equals(data.senderBeforeWrite.metadata.dependencies,
+ data.receiverAfterWrite.metadata.dependencies,
+ 'dependencies are arrays, and match (for receiver before and after write)');
+}, 'video metadata: dependencies');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.metadata.frameId, 'number');
+ assert_equals(data.senderBeforeWrite.metadata.frameId,
+ data.senderAfterWrite.metadata.frameId,
+ 'frameId matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.metadata.frameId,
+ data.receiverBeforeWrite.metadata.frameId,
+ 'frameId matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.metadata.frameId,
+ data.receiverAfterWrite.metadata.frameId,
+ 'frameId matches (for receiver before and after write)');
+}, 'video metadata: frameId');
+
+promise_test(async (test) => {
+ const data = await gatherMetadata(test, 'video');
+
+ assert_equals(typeof data.senderBeforeWrite.type, 'string');
+ assert_true(data.senderBeforeWrite.type == 'key' || data.senderBeforeWrite.type == 'delta');
+ assert_equals(data.senderBeforeWrite.type,
+ data.senderAfterWrite.type,
+ 'type matches (for sender before and after write)');
+ assert_equals(data.senderBeforeWrite.type,
+ data.receiverBeforeWrite.type,
+ 'type matches (for sender and receiver)');
+ assert_equals(data.senderBeforeWrite.type,
+ data.receiverAfterWrite.type,
+ 'type matches (for receiver before and after write)');
+}, 'video metadata: type');
+
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame-simulcast.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame-simulcast.https.html
new file mode 100644
index 0000000000..4174aaf24a
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame-simulcast.https.html
@@ -0,0 +1,136 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset=utf-8>
+ <title>RTCRtpScriptTransformer.generateKeyFrame simulcast tests</title>
+ <meta name='timeout' content='long'>
+ <script src='/resources/testharness.js'></script>
+ <script src='/resources/testharnessreport.js'></script>
+ <script src=/resources/testdriver.js></script>
+ <script src=/resources/testdriver-vendor.js></script>
+ <script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id='video1' autoplay></video>
+ <video id='video2' autoplay></video>
+ <script src ='routines.js'></script>
+ <script src ='../webrtc/simulcast/simulcast.js'></script>
+ <script src ='../webrtc/RTCPeerConnection-helper.js'></script>
+ <script src='../webrtc/third_party/sdp/sdp.js'></script>
+ <script>
+
+const generateKeyFrame = (port, opts) => postMethod(port, 'generateKeyFrame', opts);
+const waitForFrame = port => postMethod(port, 'waitForFrame');
+
+promise_test(async (test) => {
+ const worker = await createWorker('script-transform-generateKeyFrame.js');
+ const transform = await createTransform(worker);
+ const senderPc = new RTCPeerConnection();
+ const receiverPc = new RTCPeerConnection();
+ // This will only work if first rid is 0
+ exchangeIceCandidates(senderPc, receiverPc);
+ const stream = await navigator.mediaDevices.getUserMedia({video: true});
+ const {sender} = senderPc.addTransceiver(stream.getTracks()[0], {sendEncodings: [{rid: '0'}, {rid: '1'}, {rid: '2'}]});
+ sender.transform = transform;
+ await doOfferToSendSimulcastAndAnswer(senderPc, receiverPc, ['0', '1', '2']);
+
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // Spec says ridless generateKeyFrame selects the first stream, so should work
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '0'});
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '1'});
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '2'});
+ assert_equals(message.result, 'success');
+}, 'generateKeyFrame works with simulcast rids');
+
+promise_test(async (test) => {
+ const worker = await createWorker('script-transform-generateKeyFrame.js');
+ const transform = await createTransform(worker);
+ const senderPc = new RTCPeerConnection();
+ const receiverPc = new RTCPeerConnection();
+ // This will only work if first rid is 0
+ exchangeIceCandidates(senderPc, receiverPc);
+ const stream = await navigator.mediaDevices.getUserMedia({video: true});
+ const {sender} = senderPc.addTransceiver(stream.getTracks()[0], {sendEncodings: [{rid: '0'}, {rid: '1'}, {rid: '2'}]});
+ sender.transform = transform;
+ await doOfferToSendSimulcastAndAnswer(senderPc, receiverPc, ['0', '1', '2']);
+
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // Remove the '1' encoding
+ await doOfferToSendSimulcastAndAnswer(senderPc, receiverPc, ['0', '2']);
+
+ // Spec says ridless generateKeyFrame selects the first stream, so should work
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '0'});
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '1'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotFoundError', `Message: ${message.message}`);
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '2'});
+ assert_equals(message.result, 'success');
+}, 'generateKeyFrame for rid that was negotiated away fails');
+
+promise_test(async (test) => {
+ const worker = await createWorker('script-transform-generateKeyFrame.js');
+ const transform = await createTransform(worker);
+ const senderPc = new RTCPeerConnection();
+ const receiverPc = new RTCPeerConnection();
+ // This will only work if first rid is 0
+ exchangeIceCandidates(senderPc, receiverPc);
+ const stream = await navigator.mediaDevices.getUserMedia({video: true});
+ const {sender} = senderPc.addTransceiver(stream.getTracks()[0], {sendEncodings: [{rid: '0'}, {rid: '1'}, {rid: '2'}]});
+ sender.transform = transform;
+ await doOfferToSendSimulcastAndAnswer(senderPc, receiverPc, ['0', '1', '2']);
+
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // Drop to unicast
+ await doOfferToSendSimulcastAndAnswer(senderPc, receiverPc, []);
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ // This is really lame, but there could be frames with rids in flight, and
+ // there's not really any good way to know when they've been flushed out.
+ // If RTCEncodedVideoFrame had a rid field, we might be able to watch for a
+ // frame without a rid. We can't just use generateKeyFrame(null) either,
+ // because a keyframe in flight with the first rid can resolve it. However,
+ // it's reasonable to expect that if we wait for a _second_
+ // generateKeyFrame(null), that should not be resolved with a keyframe for
+ // '0'
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '0'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotFoundError', `Message: ${message.message}`);
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '1'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotFoundError', `Message: ${message.message}`);
+
+ message = await generateKeyFrame(sender.transform.port, {rid: '2'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotFoundError', `Message: ${message.message}`);
+}, 'generateKeyFrame with rid after simulcast->unicast negotiation fails');
+
+ </script>
+ </body>
+</html>
+
+
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.https.html
new file mode 100644
index 0000000000..348902ea36
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.https.html
@@ -0,0 +1,229 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset=utf-8>
+ <title>RTCRtpScriptTransformer.generateKeyFrame tests</title>
+ <meta name='timeout' content='long'>
+ <script src='/resources/testharness.js'></script>
+ <script src='/resources/testharnessreport.js'></script>
+ <script src=/resources/testdriver.js></script>
+ <script src=/resources/testdriver-vendor.js></script>
+ <script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id='video1' autoplay></video>
+ <video id='video2' autoplay></video>
+ <script src ='routines.js'></script>
+ <script src ='../webrtc/simulcast/simulcast.js'></script>
+ <script src ='../webrtc/RTCPeerConnection-helper.js'></script>
+ <script src='../webrtc/third_party/sdp/sdp.js'></script>
+ <script>
+
+const generateKeyFrame = (port, opts) => postMethod(port, 'generateKeyFrame', opts);
+const waitForFrame = port => postMethod(port, 'waitForFrame');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {audio: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // No rids
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+
+ message = await waitForFrame(receiver.transform.port);
+ assert_equals(message, 'got frame');
+
+ // No rids
+ message = await generateKeyFrame(receiver.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+}, 'generateKeyFrame() throws for audio');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // No rids
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+ // value should be a timestamp
+ assert_equals(typeof message.value, 'number');
+ assert_greater_than(message.value, 0);
+
+ // No rids
+ message = await generateKeyFrame(receiver.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+
+ video1.srcObject = new MediaStream([receiver.track]);
+ await video1.play();
+}, 'generateKeyFrame(null) resolves for video sender, and throws for video receiver');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ // Invalid rid, empty string
+ message = await generateKeyFrame(sender.transform.port, {rid: ''});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+
+ // Invalid rid, bad ASCII characters
+ message = await generateKeyFrame(sender.transform.port, {rid: '!?'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+
+ // Invalid rid, bad ASCII characters (according to RFC 8852, but not RFC 8851)
+ message = await generateKeyFrame(sender.transform.port, {rid: 'foo-bar'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+
+ // Invalid rid, bad ASCII characters (according to RFC 8852, but not RFC 8851)
+ message = await generateKeyFrame(sender.transform.port, {rid: 'foo_bar'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+
+ // Invalid rid, bad non-ASCII characters
+ message = await generateKeyFrame(sender.transform.port, {rid: '(╯°□°)╯︵ ┻━┻'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+
+ // Invalid rid, too long
+ message = await generateKeyFrame(sender.transform.port, {rid: 'a'.repeat(256)});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotAllowedError', `Message: ${message.message}`);
+}, 'generateKeyFrame throws NotAllowedError for invalid rid');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port, {rid: 'foo'});
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'NotFoundError', `Message: ${message.message}`);
+}, 'generateKeyFrame throws NotFoundError for unknown rid');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ const senderTransform = sender.transform;
+ sender.transform = null;
+
+ message = await generateKeyFrame(senderTransform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+}, 'generateKeyFrame throws for unset transforms');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+ // value should be a timestamp
+ assert_equals(typeof message.value, 'number');
+ assert_greater_than(message.value, 0);
+ const timestamp = message.value;
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+ // value should be a timestamp
+ assert_equals(typeof message.value, 'number');
+ assert_greater_than(message.value, timestamp);
+}, 'generateKeyFrame timestamp should advance');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+ const count = message.count;
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+ assert_greater_than(message.count, count);
+}, 'await generateKeyFrame, await generateKeyFrame should see an increase in count of keyframes');
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ senderPc.getTransceivers()[0].direction = 'inactive';
+ await senderPc.setLocalDescription();
+ await receiverPc.setRemoteDescription(senderPc.localDescription);
+ await receiverPc.setLocalDescription();
+ await senderPc.setRemoteDescription(receiverPc.localDescription);
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+
+ senderPc.getTransceivers()[0].direction = 'sendonly';
+ await senderPc.setLocalDescription();
+ await receiverPc.setRemoteDescription(senderPc.localDescription);
+ await receiverPc.setLocalDescription();
+ await senderPc.setRemoteDescription(receiverPc.localDescription);
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+}, 'generateKeyFrame rejects when the sender is negotiated inactive, and resumes succeeding when negotiated back to active');
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ senderPc.getTransceivers()[0].stop();
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+}, 'generateKeyFrame rejects when the sender is stopped, even without negotiation');
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-generateKeyFrame.js', {video: true});
+ let message = await waitForFrame(sender.transform.port);
+ assert_equals(message, 'got frame');
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'success');
+
+ await senderPc.getTransceivers()[0].sender.replaceTrack(null);
+
+ message = await generateKeyFrame(sender.transform.port);
+ assert_equals(message.result, 'failure');
+ assert_equals(message.value, 'InvalidStateError', `Message: ${message.message}`);
+}, 'generateKeyFrame rejects with a null track');
+
+// TODO: It would be nice to be able to test that pending generateKeyFrame
+// promises are _rejected_ when the transform is unset, or the sender stops
+// sending. However, getting the timing on this right is going to be very hard.
+// While we could stop the processing of frames before calling
+// generateKeyFrame, this would not necessarily help, because generateKeyFrame
+// promises are resolved _before_ enqueueing the frame into |readable|, and
+// right now the spec does not have a high water mark/backpressure on
+// |readable|, so pausing would not necessarily prevent the enqueue.
+ </script>
+ </body>
+</html>
+
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.js b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.js
new file mode 100644
index 0000000000..5e68ee1fb9
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-generateKeyFrame.js
@@ -0,0 +1,70 @@
+onrtctransform = event => {
+ const transformer = event.transformer;
+ let keyFrameCount = 0;
+ let gotFrame;
+
+ transformer.options.port.onmessage = event => {
+ const {method, rid} = event.data;
+ // Maybe refactor to have transaction ids?
+ if (method == 'generateKeyFrame') {
+ generateKeyFrame(rid);
+ } else if (method == 'waitForFrame') {
+ waitForFrame();
+ }
+ }
+
+ async function rejectInMs(timeout) {
+ return new Promise((_, reject) => {
+ const rejectWithTimeout = () => {
+ reject(new DOMException(`Timed out after waiting for ${timeout} ms`,
+ 'TimeoutError'));
+ };
+ setTimeout(rejectWithTimeout, timeout);
+ });
+ }
+
+ async function generateKeyFrame(rid) {
+ try {
+ const timestamp = await Promise.race([transformer.generateKeyFrame(rid), rejectInMs(8000)]);
+ transformer.options.port.postMessage({result: 'success', value: timestamp, count: keyFrameCount});
+ } catch (e) {
+ // TODO: This does not work if we send e.name, why?
+ transformer.options.port.postMessage({result: 'failure', value: `${e.name}`, message: `${e.message}`});
+ }
+ }
+
+ async function waitForFrame() {
+ try {
+ await Promise.race([new Promise(r => gotFrameCallback = r), rejectInMs(8000)]);
+ transformer.options.port.postMessage('got frame');
+ } catch (e) {
+ // TODO: This does not work if we send e.name, why?
+ transformer.options.port.postMessage({result: 'failure', value: `${e.name}`, message: `${e.message}`});
+ }
+ }
+
+ transformer.options.port.postMessage('started');
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+ if (chunk.value instanceof RTCEncodedVideoFrame) {
+ if (chunk.value.type == 'key') {
+ keyFrameCount++;
+ }
+ }
+ if (gotFrameCallback) {
+ gotFrameCallback();
+ }
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+
+ process(transformer);
+};
+self.postMessage('registered');
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.https.html
new file mode 100644
index 0000000000..51b797eb68
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.https.html
@@ -0,0 +1,110 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset=utf-8>
+ <title>RTCRtpScriptTransformer.sendKeyFrameRequest tests</title>
+ <meta name='timeout' content='long'>
+ <script src='/resources/testharness.js'></script>
+ <script src='/resources/testharnessreport.js'></script>
+ <script src=/resources/testdriver.js></script>
+ <script src=/resources/testdriver-vendor.js></script>
+ <script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id='video1' autoplay></video>
+ <video id='video2' autoplay></video>
+ <script src ='routines.js'></script>
+ <script>
+
+const sendKeyFrameRequest = (port, opts) => postMethod(port, 'sendKeyFrameRequest', opts);
+const waitForFrame = port => postMethod(port, 'waitForFrame');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-sendKeyFrameRequest.js', {video: true});
+ assert_equals(await waitForFrame(receiver.transform.port), 'got frame');
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'success');
+
+ assert_equals(await sendKeyFrameRequest(sender.transform.port), 'failure: InvalidStateError');
+
+ video1.srcObject = new MediaStream([receiver.track]);
+ await video1.play();
+}, 'sendKeyFrameRequest resolves for video receiver, and throws for video sender');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-sendKeyFrameRequest.js', {audio: true});
+ assert_equals(await waitForFrame(receiver.transform.port), 'got frame');
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'failure: InvalidStateError');
+
+ assert_equals(await waitForFrame(sender.transform.port), 'got frame');
+
+ assert_equals(await sendKeyFrameRequest(sender.transform.port), 'failure: InvalidStateError');
+
+ video1.srcObject = new MediaStream([receiver.track]);
+ await video1.play();
+}, 'sendKeyFrameRequest throws for audio sender/receiver');
+
+promise_test(async (test) => {
+ const [senderTransform] = await createTransforms('script-transform-sendKeyFrameRequest.js');
+ assert_equals(await sendKeyFrameRequest(senderTransform.port), 'failure: InvalidStateError');
+}, 'sendKeyFrameRequest throws for unused transforms');
+
+promise_test(async (test) => {
+ const {sender, receiver} = await createConnectionWithTransform(test, 'script-transform-sendKeyFrameRequest.js', {video: true});
+ assert_equals(await waitForFrame(receiver.transform.port), 'got frame');
+
+ const receiverTransform = receiver.transform;
+ assert_equals(await sendKeyFrameRequest(receiverTransform.port), 'success');
+
+ // TODO: Spec currently says that this will immediately cause the transformer
+ // to stop working. This may change.
+ receiver.transform = null;
+
+ assert_equals(await sendKeyFrameRequest(receiverTransform.port), 'failure: InvalidStateError');
+}, 'sendKeyFrameRequest throws for unset transforms');
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-sendKeyFrameRequest.js', {video: true});
+ assert_equals(await waitForFrame(receiver.transform.port), 'got frame');
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'success');
+
+ senderPc.getTransceivers()[0].direction = 'inactive';
+ await senderPc.setLocalDescription();
+ await receiverPc.setRemoteDescription(senderPc.localDescription);
+ await receiverPc.setLocalDescription();
+ await senderPc.setRemoteDescription(receiverPc.localDescription);
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'failure: InvalidStateError');
+
+ senderPc.getTransceivers()[0].direction = 'sendonly';
+ await senderPc.setLocalDescription();
+ await receiverPc.setRemoteDescription(senderPc.localDescription);
+ await receiverPc.setLocalDescription();
+ await senderPc.setRemoteDescription(receiverPc.localDescription);
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'success');
+}, 'sendKeyFrameRequest rejects when the receiver is negotiated inactive, and resumes succeeding when negotiated back to active');
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-sendKeyFrameRequest.js', {video: true});
+ assert_equals(await waitForFrame(receiver.transform.port), 'got frame');
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'success');
+
+ receiverPc.getTransceivers()[0].stop();
+
+ assert_equals(await sendKeyFrameRequest(receiver.transform.port), 'failure: InvalidStateError');
+}, 'sendKeyFrameRequest rejects when the receiver is stopped');
+
+// Testing that sendKeyFrameRequest actually results in the sending of keyframe
+// requests is effectively impossible, because there is no API to expose the
+// reception of a keyframe request, keyframes are sent regularly anyway, and
+// the spec allows the receiver to ignore these calls if sending a keyframe
+// request is 'not deemed appropriate'! sendKeyFrameRequest is at most a
+// suggestion.
+
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.js b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.js
new file mode 100644
index 0000000000..361d7ce023
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-sendKeyFrameRequest.js
@@ -0,0 +1,63 @@
+onrtctransform = event => {
+ const transformer = event.transformer;
+ let gotFrame;
+
+ transformer.options.port.onmessage = event => {
+ const {method} = event.data;
+ if (method == 'sendKeyFrameRequest') {
+ sendKeyFrameRequest();
+ } else if (method == 'waitForFrame') {
+ waitForFrame();
+ }
+ }
+
+ async function rejectInMs(timeout) {
+ return new Promise((_, reject) => {
+ const rejectWithTimeout = () => {
+ reject(new DOMException(`Timed out after waiting for ${timeout} ms`,
+ 'TimeoutError'));
+ };
+ setTimeout(rejectWithTimeout, timeout);
+ });
+ }
+
+ async function sendKeyFrameRequest() {
+ try {
+ await Promise.race([transformer.sendKeyFrameRequest(), rejectInMs(8000)]);;
+ transformer.options.port.postMessage('success');
+ } catch (e) {
+ // TODO: This does not work if we send e.name, why?
+ transformer.options.port.postMessage(`failure: ${e.name}`);
+ }
+ }
+
+ async function waitForFrame() {
+ try {
+ await Promise.race([new Promise(r => gotFrameCallback = r), rejectInMs(8000)]);
+ transformer.options.port.postMessage('got frame');
+ } catch (e) {
+ // TODO: This does not work if we send e.name, why?
+ transformer.options.port.postMessage({result: 'failure', value: `${e.name}`, message: `${e.message}`});
+ }
+ }
+
+ transformer.options.port.postMessage('started');
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+ if (gotFrameCallback) {
+ gotFrameCallback();
+ }
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+
+ process(transformer);
+};
+self.postMessage('registered');
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-worker.js
new file mode 100644
index 0000000000..88efb9c6a3
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform-worker.js
@@ -0,0 +1,33 @@
+onrtctransform = (event) => {
+ const transformer = event.transformer;
+ transformer.options.port.onmessage = (event) => {
+ if (event.data == "ping") {
+ transformer.options.port.postMessage("pong");
+ }
+ };
+
+ transformer.options.port.postMessage("started");
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+ if (chunk.value instanceof RTCEncodedVideoFrame) {
+ transformer.options.port.postMessage("video chunk");
+ if (chunk.value.type == "key") {
+ transformer.options.port.postMessage("video keyframe");
+ }
+ }
+ else if (chunk.value instanceof RTCEncodedAudioFrame)
+ transformer.options.port.postMessage("audio chunk");
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+
+ process(transformer);
+};
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-transform.https.html
new file mode 100644
index 0000000000..491e917e86
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-transform.https.html
@@ -0,0 +1,60 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="video1" autoplay></video>
+ <video id="video2" autoplay></video>
+ <script src ="routines.js"></script>
+ <script>
+
+promise_test(async (test) => {
+ const worker = await createWorker('script-transform-worker.js');
+ const transform = await createTransform(worker);
+ transform.port.postMessage("ping");
+ assert_equals(await getNextMessage(transform.port), "pong");
+}, "transform messaging");
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-worker.js', {audio: true});
+
+ const sender2 = senderPc.addTransceiver('video').sender;
+ const receiver2 = senderPc.getReceivers()[1];
+
+ assert_throws_dom("InvalidStateError", () => sender2.transform = sender.transform);
+ assert_throws_dom("InvalidStateError", () => receiver2.transform = receiver.transform);
+
+ sender.transform = sender.transform;
+ receiver.transform = receiver.transform;
+
+ sender.transform = null;
+ receiver.transform = null;
+}, "Cannot reuse attached transforms");
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-worker.js', {audio: true});
+ assert_equals(await getNextMessage(sender.transform.port), "audio chunk");
+
+ video1.srcObject = new MediaStream([receiver.track]);
+ await video1.play();
+}, "audio exchange with transform");
+
+promise_test(async (test) => {
+ const {sender, receiver, senderPc, receiverPc} = await createConnectionWithTransform(test, 'script-transform-worker.js', {video: true});
+
+ assert_equals(await getNextMessage(sender.transform.port), "video chunk");
+ // First frame should be a keyframe
+ assert_equals(await getNextMessage(sender.transform.port), "video keyframe");
+
+ video1.srcObject = new MediaStream([receiver.track]);
+ await video1.play();
+}, "video exchange with transform");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform-worker.js
new file mode 100644
index 0000000000..5d428c81b3
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform-worker.js
@@ -0,0 +1,22 @@
+onrtctransform = (event) => {
+ const transformer = event.transformer;
+
+ self.postMessage("started");
+
+ transformer.reader = transformer.readable.getReader();
+ transformer.writer = transformer.writable.getWriter();
+ function process(transformer)
+ {
+ transformer.reader.read().then(chunk => {
+ if (chunk.done)
+ return;
+
+ transformer.writer.write(chunk.value);
+ transformer.writer.write(chunk.value);
+ process(transformer);
+ });
+ }
+
+ process(transformer);
+};
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform.https.html b/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform.https.html
new file mode 100644
index 0000000000..c4a49af7ac
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/script-write-twice-transform.https.html
@@ -0,0 +1,62 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="video1" autoplay></video>
+ <video id="video2" autoplay></video>
+ <script src ="routines.js"></script>
+ <script>
+async function waitForMessage(worker, data)
+{
+ while (true) {
+ const received = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ if (data === received)
+ return;
+ }
+}
+
+promise_test(async (test) => {
+ worker = new Worker('script-write-twice-transform-worker.js');
+ const data = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ assert_equals(data, "registered");
+
+ await setMediaPermission("granted", ["camera"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({video: true});
+
+ let sender, receiver;
+ const senderTransform = new RTCRtpScriptTransform(worker, {name:'MockRTCRtpTransform', side:'sender', role:'encrypt'});
+ const receiverTransform = new RTCRtpScriptTransform(worker, {name:'MockRTCRtpTransform', side:'receiver', role:'decrypt'});
+
+ const startedPromise = new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+
+ const stream = await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ pc1 = firstConnection;
+ sender = firstConnection.addTrack(localStream.getVideoTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ pc2 = secondConnection;
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ assert_equals(await startedPromise, "started");
+
+ video1.srcObject = stream;
+ await video1.play();
+}, "video exchange with write twice transform");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/set-metadata.https.html b/testing/web-platform/tests/webrtc-encoded-transform/set-metadata.https.html
new file mode 100644
index 0000000000..c4699598cf
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/set-metadata.https.html
@@ -0,0 +1,53 @@
+<!DOCTYPE html>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+<script src="../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+<script src='./helper.js'></script>
+<script>
+"use strict";
+
+promise_test(async t => {
+ const senderReader = await setupLoopbackWithCodecAndGetReader(t, 'VP8');
+ const result = await senderReader.read();
+ const metadata = result.value.getMetadata();
+ // TODO(https://crbug.com/webrtc/14709): When RTCEncodedVideoFrame has a
+ // constructor, create a new frame from scratch instead of cloning it to
+ // ensure that none of the metadata was carried over via structuredClone().
+ // This would allow us to be confident that setMetadata() is doing all the
+ // work.
+ //
+ // At that point, we can refactor the structuredClone() implementation to be
+ // the same as constructor() + set data + setMetadata() to ensure that
+ // structuredClone() cannot do things that are not already exposed in
+ // JavaScript (no secret steps!).
+ const clone = structuredClone(result.value);
+ clone.setMetadata(metadata);
+ const cloneMetadata = clone.getMetadata();
+ // Encoding related metadata.
+ assert_equals(cloneMetadata.frameId, metadata.frameId, 'frameId');
+ assert_array_equals(cloneMetadata.dependencies, metadata.dependencies,
+ 'dependencies');
+ assert_equals(cloneMetadata.width, metadata.width, 'width');
+ assert_equals(cloneMetadata.height, metadata.height, 'height');
+ assert_equals(cloneMetadata.spatialIndex, metadata.spatialIndex,
+ 'spatialIndex');
+ assert_equals(cloneMetadata.temporalIndex, metadata.temporalIndex,
+ 'temporalIndex');
+ // RTP related metadata.
+ // TODO(https://crbug.com/webrtc/14709): This information also needs to be
+ // settable but isn't - the assertions only pass because structuredClone()
+ // copies them for us. It would be great if different layers didn't consider
+ // different subset of the struct as "the metadata". Can we consolidate into a
+ // single webrtc struct for all metadata?
+ assert_equals(cloneMetadata.synchronizationSource,
+ metadata.synchronizationSource, 'synchronizationSource');
+ assert_array_equals(cloneMetadata.contributingSources,
+ metadata.contributingSources, 'contributingSources');
+ assert_equals(cloneMetadata.payloadType, metadata.payloadType, 'payloadType');
+}, "[VP8] setMetadata() carries over codec-specific properties");
+
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-keys.https.html b/testing/web-platform/tests/webrtc-encoded-transform/sframe-keys.https.html
new file mode 100644
index 0000000000..c87ac12e29
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-keys.https.html
@@ -0,0 +1,69 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="audio" autoplay playsInline></video>
+ <script src ="routines.js"></script>
+ <script>
+let sender, receiver;
+let key1, key2, key3, key4;
+
+promise_test(async (test) => {
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ const transform = new SFrameTransform;
+
+ await transform.setEncryptionKey(key);
+ await transform.setEncryptionKey(key, 1);
+
+ await transform.setEncryptionKey(key, BigInt('18446744073709551613'));
+ await transform.setEncryptionKey(key, BigInt('18446744073709551614'));
+ await transform.setEncryptionKey(key, BigInt('18446744073709551615'));
+ await transform.setEncryptionKey(key, BigInt('18446744073709551616')).then(assert_unreached, (e) => {
+ assert_true(e instanceof RangeError);
+ assert_equals(e.message, "Not a 64 bits integer");
+ });
+}, "Passing various key IDs");
+
+promise_test(async (test) => {
+ key1 = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ key2 = await crypto.subtle.importKey("raw", new Uint8Array([144, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ key3 = await crypto.subtle.importKey("raw", new Uint8Array([145, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ key4 = await crypto.subtle.importKey("raw", new Uint8Array([146, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+
+ await setMediaPermission("granted", ["microphone"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({audio: true});
+ const stream = await new Promise((resolve, reject) => {
+ const connections = createConnections(test, (firstConnection) => {
+ sender = firstConnection.addTrack(localStream.getAudioTracks()[0], localStream);
+ let transform = new SFrameTransform;
+ transform.setEncryptionKey(key1);
+ sender.transform = transform;
+ }, (secondConnection) => {
+ secondConnection.ontrack = (trackEvent) => {
+ let transform = new SFrameTransform;
+ transform.setEncryptionKey(key1);
+ transform.setEncryptionKey(key2);
+ transform.setEncryptionKey(key3, 1000);
+ transform.setEncryptionKey(key4, BigInt('18446744073709551615'));
+ receiver = trackEvent.receiver;
+ receiver.transform = transform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ audio.srcObject = stream;
+ await audio.play();
+}, "Audio exchange with SFrame setup");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-buffer-source.html b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-buffer-source.html
new file mode 100644
index 0000000000..99b45f22c9
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-buffer-source.html
@@ -0,0 +1,50 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+ <script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ </head>
+ <body>
+ <script>
+
+async function getEncryptedData(transform)
+{
+ const chunk = await transform.readable.getReader().read();
+ const value = new Uint8Array(chunk.value);
+ return [...value];
+}
+
+promise_test(async (test) => {
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ const transform1 = new SFrameTransform;
+ const transform2 = new SFrameTransform;
+ const transform3 = new SFrameTransform;
+
+ await transform1.setEncryptionKey(key);
+ await transform2.setEncryptionKey(key);
+ await transform3.setEncryptionKey(key);
+
+ const buffer1 = new ArrayBuffer(10);
+ const buffer2 = new ArrayBuffer(11);
+ const view1 = new Uint8Array(buffer1);
+ const view2 = new Uint8Array(buffer2, 1);
+ for (let i = 0 ; i < buffer1.byteLength; ++i) {
+ view1[i] = i;
+ view2[i] = i;
+ }
+
+ transform1.writable.getWriter().write(buffer1);
+ transform2.writable.getWriter().write(view1);
+ transform3.writable.getWriter().write(view2);
+
+ const result1 = await getEncryptedData(transform1);
+ const result2 = await getEncryptedData(transform2);
+ const result3 = await getEncryptedData(transform3);
+
+ assert_array_equals(result1, result2, "result2");
+ assert_array_equals(result1, result3, "result3");
+}, "Uint8Array as input to SFrameTransform");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-in-worker.https.html b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-in-worker.https.html
new file mode 100644
index 0000000000..f5d7b5a930
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-in-worker.https.html
@@ -0,0 +1,61 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+<script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ <script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../mediacapture-streams/permission-helper.js'></script>
+ </head>
+ <body>
+ <video id="video1" controls autoplay></video>
+ <script src ="routines.js"></script>
+ <script>
+async function waitForMessage(worker, data)
+{
+ while (true) {
+ const received = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ if (data === received)
+ return;
+ }
+}
+
+promise_test(async (test) => {
+ worker = new Worker('sframe-transform-worker.js');
+ const data = await new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+ assert_equals(data, "registered");
+ await setMediaPermission("granted", ["camera"]);
+ const localStream = await navigator.mediaDevices.getUserMedia({ video: true });
+
+ let sender, receiver;
+ const senderTransform = new SFrameTransform({ compatibilityMode: "H264" });
+ const receiverTransform = new RTCRtpScriptTransform(worker, "SFrameRTCRtpTransform");
+
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+ senderTransform.setEncryptionKey(key);
+
+ const startedPromise = new Promise(resolve => worker.onmessage = (event) => resolve(event.data));
+
+ const stream = await new Promise((resolve, reject) => {
+ createConnections(test, (firstConnection) => {
+ pc1 = firstConnection;
+ sender = firstConnection.addTrack(localStream.getTracks()[0], localStream);
+ sender.transform = senderTransform;
+ }, (secondConnection) => {
+ pc2 = secondConnection;
+ secondConnection.ontrack = (trackEvent) => {
+ receiver = trackEvent.receiver;
+ receiver.transform = receiverTransform;
+ resolve(trackEvent.streams[0]);
+ };
+ });
+ test.step_timeout(() => reject("Test timed out"), 5000);
+ });
+
+ video1.srcObject = stream;
+ await video1.play();
+}, "video exchange with SFrame transform in worker");
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-readable.html b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-readable.html
new file mode 100644
index 0000000000..2afce1b271
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-readable.html
@@ -0,0 +1,19 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+ <script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ </head>
+ <body>
+ <iframe src="." id="frame"></iframe>
+ <script>
+promise_test(async (test) => {
+ const frameDOMException = frame.contentWindow.DOMException;
+ const transform = new frame.contentWindow.SFrameTransform;
+ frame.remove();
+ assert_throws_dom("InvalidStateError", frameDOMException, () => transform.readable);
+});
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-worker.js b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-worker.js
new file mode 100644
index 0000000000..617cf0a38a
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform-worker.js
@@ -0,0 +1,7 @@
+onrtctransform = (event) => {
+ const sframeTransform = new SFrameTransform({ role : "decrypt", authenticationSize: "10", compatibilityMode: "H264" });
+ crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]).then(key => sframeTransform.setEncryptionKey(key));
+ const transformer = event.transformer;
+ transformer.readable.pipeThrough(sframeTransform).pipeTo(transformer.writable);
+}
+self.postMessage("registered");
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform.html b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform.html
new file mode 100644
index 0000000000..2e40135b04
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/sframe-transform.html
@@ -0,0 +1,141 @@
+<!doctype html>
+<html>
+ <head>
+ <meta charset="utf-8">
+ <script src="/resources/testharness.js"></script>
+ <script src="/resources/testharnessreport.js"></script>
+ </head>
+ <body>
+ <script>
+
+promise_test(async (test) => {
+ const pc = new RTCPeerConnection();
+ const senderTransform = new SFrameTransform();
+ const receiverTransform = new SFrameTransform();
+ const sender1 = pc.addTransceiver('audio').sender;
+ const sender2 = pc.addTransceiver('video').sender;
+ const receiver1 = pc.getReceivers()[0];
+ const receiver2 = pc.getReceivers()[1];
+
+ sender1.transform = senderTransform;
+ receiver1.transform = receiverTransform;
+ assert_throws_dom("InvalidStateError", () => sender2.transform = senderTransform);
+ assert_throws_dom("InvalidStateError", () => receiver2.transform = receiverTransform);
+
+ sender1.transform = senderTransform;
+ receiver1.transform = receiverTransform;
+
+ sender1.transform = null;
+ receiver1.transform = null;
+}, "Cannot reuse attached transforms");
+
+test(() => {
+ const senderTransform = new SFrameTransform();
+
+ assert_true(senderTransform.readable instanceof ReadableStream);
+ assert_true(senderTransform.writable instanceof WritableStream);
+}, "SFrameTransform exposes readable and writable");
+
+promise_test(async (test) => {
+ const pc = new RTCPeerConnection();
+ const senderTransform = new SFrameTransform();
+ const receiverTransform = new SFrameTransform();
+ const sender1 = pc.addTransceiver('audio').sender;
+ const sender2 = pc.addTransceiver('video').sender;
+ const receiver1 = pc.getReceivers()[0];
+ const receiver2 = pc.getReceivers()[1];
+
+ assert_false(senderTransform.readable.locked, "sender readable before");
+ assert_false(senderTransform.writable.locked, "sender writable before");
+ assert_false(receiverTransform.readable.locked, "receiver readable before");
+ assert_false(receiverTransform.writable.locked, "receiver writable before");
+
+ sender1.transform = senderTransform;
+ receiver1.transform = receiverTransform;
+
+ assert_true(senderTransform.readable.locked, "sender readable during");
+ assert_true(senderTransform.writable.locked, "sender writable during");
+ assert_true(receiverTransform.readable.locked, "receiver readable during");
+ assert_true(receiverTransform.writable.locked, "receiver writable during");
+
+ sender1.transform = null;
+ receiver1.transform = null;
+
+ assert_true(senderTransform.readable.locked, "sender readable after");
+ assert_true(senderTransform.writable.locked, "sender writable after");
+ assert_true(receiverTransform.readable.locked, "receiver readable after");
+ assert_true(receiverTransform.writable.locked, "receiver writable after");
+}, "readable/writable are locked when attached and after being attached");
+
+promise_test(async (test) => {
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+
+ const senderTransform = new SFrameTransform({ role : 'encrypt', authenticationSize: 10 });
+ senderTransform.setEncryptionKey(key);
+
+ const receiverTransform = new SFrameTransform({ role : 'decrypt', authenticationSize: 10 });
+ receiverTransform.setEncryptionKey(key);
+
+ const writer = senderTransform.writable.getWriter();
+ const reader = receiverTransform.readable.getReader();
+
+ senderTransform.readable.pipeTo(receiverTransform.writable);
+
+ const sent = new ArrayBuffer(8);
+ const view = new Int8Array(sent);
+ for (let cptr = 0; cptr < sent.byteLength; ++cptr)
+ view[cptr] = cptr;
+
+ writer.write(sent);
+ const received = await reader.read();
+
+ assert_equals(received.value.byteLength, 8);
+ const view2 = new Int8Array(received.value);
+ for (let cptr = 0; cptr < sent.byteLength; ++cptr)
+ assert_equals(view2[cptr], view[cptr]);
+}, "SFrame with array buffer - authentication size 10");
+
+promise_test(async (test) => {
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+
+ const senderTransform = new SFrameTransform({ role : 'encrypt', authenticationSize: 10 });
+ const senderWriter = senderTransform.writable.getWriter();
+ const senderReader = senderTransform.readable.getReader();
+
+ const receiverTransform = new SFrameTransform({ role : 'decrypt', authenticationSize: 10 });
+ const receiverWriter = receiverTransform.writable.getWriter();
+ const receiverReader = receiverTransform.readable.getReader();
+
+ senderTransform.setEncryptionKey(key);
+ receiverTransform.setEncryptionKey(key);
+
+ const chunk = new ArrayBuffer(8);
+
+ // decryption should fail, leading to an empty array buffer.
+ await receiverWriter.write(chunk);
+ let received = await receiverReader.read();
+ assert_equals(received.value.byteLength, 0);
+
+ // We write again but this time with a chunk we can decrypt.
+ await senderWriter.write(chunk);
+ const encrypted = await senderReader.read();
+ await receiverWriter.write(encrypted.value);
+ received = await receiverReader.read();
+ assert_equals(received.value.byteLength, 8);
+}, "SFrame decryption with array buffer that is too small");
+
+promise_test(async (test) => {
+ const key = await crypto.subtle.importKey("raw", new Uint8Array([143, 77, 43, 10, 72, 19, 37, 67, 236, 219, 24, 93, 26, 165, 91, 178]), "HKDF", false, ["deriveBits", "deriveKey"]);
+
+ const receiverTransform = new SFrameTransform({ role : 'decrypt', authenticationSize: 10 });
+ const receiverWriter = receiverTransform.writable.getWriter();
+ receiverTransform.setEncryptionKey(key);
+
+ // decryption should fail, leading to erroring the transform.
+ await promise_rejects_js(test, TypeError, receiverWriter.write({ }));
+ await promise_rejects_js(test, TypeError, receiverWriter.closed);
+}, "SFrame transform gets errored if trying to process unexpected value types");
+
+ </script>
+ </body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-clone.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-clone.https.html
new file mode 100644
index 0000000000..9f07713d44
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-clone.https.html
@@ -0,0 +1,51 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>RTCEncodedAudioFrame can be cloned and distributed</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller1 = new RTCPeerConnection();
+ t.add_cleanup(() => caller1.close());
+ const callee1 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee1.close());
+ await setMediaPermission("granted", ["microphone"]);
+ const inputStream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const inputTrack = inputStream.getAudioTracks()[0];
+ t.add_cleanup(() => inputTrack.stop());
+ caller1.addTrack(inputTrack)
+ exchangeIceCandidates(caller1, callee1);
+
+ const caller2 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller2.close());
+ const sender2 = caller2.addTransceiver("audio").sender;
+ const writer2 = sender2.createEncodedStreams().writable.getWriter();
+ sender2.replaceTrack(new MediaStreamTrackGenerator({ kind: 'audio' }));
+
+ const framesReceivedCorrectly = new Promise((resolve, reject) => {
+ callee1.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+ const result = await receiverReader.read();
+ const original = result.value;
+ let clone = structuredClone(original);
+ assert_equals(original.timestamp, clone.timestamp);
+ assert_equals(original.getMetadata().absCaptureTime, clone.getMetadata().absCaptureTime);
+ assert_array_equals(Array.from(original.data), Array.from(clone.data));
+ await writer2.write(clone);
+ resolve();
+ }
+ });
+
+ await exchangeOfferAnswer(caller1, callee1);
+
+ return framesReceivedCorrectly;
+}, "Cloning before sending works");
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-receive-cloned.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-receive-cloned.https.html
new file mode 100644
index 0000000000..3077632a3b
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-receive-cloned.https.html
@@ -0,0 +1,52 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>RTCEncodedAudioFrame can be cloned and distributed</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller1 = new RTCPeerConnection();
+ t.add_cleanup(() => caller1.close());
+ const callee1 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee1.close());
+ await setMediaPermission("granted", ["microphone"]);
+ const inputStream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const inputTrack = inputStream.getAudioTracks()[0];
+ t.add_cleanup(() => inputTrack.stop());
+ caller1.addTrack(inputTrack)
+ exchangeIceCandidates(caller1, callee1);
+
+ const caller2 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller2.close());
+ const sender2 = caller2.addTransceiver("audio").sender;
+ const writer2 = sender2.createEncodedStreams().writable.getWriter();
+ sender2.replaceTrack(new MediaStreamTrackGenerator({ kind: 'audio' }));
+
+ const framesReceivedCorrectly = new Promise((resolve, reject) => {
+ callee1.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+ const receiverWriter = receiverStreams.writable.getWriter();
+ const result = await receiverReader.read();
+ const originalAudioFrame = result.value;
+ let clonedAudioFrame = structuredClone(originalAudioFrame);
+ assert_equals(originalAudioFrame.timestamp, clonedAudioFrame.timestamp);
+ assert_array_equals(Array.from(originalAudioFrame.data), Array.from(clonedAudioFrame.data));
+ await writer2.write(clonedAudioFrame);
+ await receiverWriter.write(structuredClone(originalAudioFrame));
+ resolve();
+ }
+ });
+
+ await exchangeOfferAnswer(caller1, callee1);
+
+ return framesReceivedCorrectly;
+}, "Cloning before sending works");
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-send-incoming.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-send-incoming.https.html
new file mode 100644
index 0000000000..02f3b17e0c
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-send-incoming.https.html
@@ -0,0 +1,63 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>RTCEncodedAudioFrame can be cloned and distributed</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller1 = new RTCPeerConnection();
+ t.add_cleanup(() => caller1.close());
+ const callee1 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee1.close());
+ await setMediaPermission("granted", ["microphone"]);
+ const inputStream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const inputTrack = inputStream.getAudioTracks()[0];
+ t.add_cleanup(() => inputTrack.stop());
+ caller1.addTrack(inputTrack)
+ exchangeIceCandidates(caller1, callee1);
+
+ const caller2 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller2.close());
+ const callee2 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee2.close());
+ const sender2 = caller2.addTransceiver("audio").sender;
+ const writer2 = sender2.createEncodedStreams().writable.getWriter();
+ sender2.replaceTrack(new MediaStreamTrackGenerator({ kind: 'audio' }));
+ exchangeIceCandidates(caller2, callee2);
+
+ const IncomingframesSentAndReceivedCorrectly = new Promise((resolve, reject) => {
+ // Write the received incoming frames on callee1 to caller2.
+ callee1.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+ const result = await receiverReader.read();
+ const original = result.value;
+ await writer2.write(original);
+ resolve();
+ }
+
+ // callee2 receives frames over the PC from caller2.
+ callee2.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+ const receiverWriter = receiverStreams.writable.getWriter();
+ const result = await receiverReader.read();
+ receiverWriter.write(result.value);
+ resolve();
+ }
+
+ });
+
+ await exchangeOfferAnswer(caller1, callee1);
+ await exchangeOfferAnswer(caller2, callee2);
+
+ return IncomingframesSentAndReceivedCorrectly;
+}, "Send endoded incoming frame");
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-serviceworker-failure.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-serviceworker-failure.https.html
new file mode 100644
index 0000000000..b2f5f5e94c
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedAudioFrame-serviceworker-failure.https.html
@@ -0,0 +1,60 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<!-- Based on similar tests in html/infrastructure/safe-passing-of-structured-data/shared-array-buffers/ -->
+<title>RTCEncodedVideoFrame cannot cross agent clusters, service worker edition</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["microphone"]);
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+
+ const sender = caller.addTrack(track)
+ const streams = sender.createEncodedStreams();
+ const reader = streams.readable.getReader();
+ const writer = streams.writable.getWriter();
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ const result = await reader.read();
+ const scope = "resources/blank.html";
+ let reg = await service_worker_unregister_and_register(t, "resources/serviceworker-failure.js", scope);
+ t.add_cleanup(() => service_worker_unregister(t, scope));
+ await wait_for_state(t, reg.installing, "activated");
+ let iframe = await with_iframe(scope);
+ t.add_cleanup(() => iframe.remove());
+ const sw = iframe.contentWindow.navigator.serviceWorker;
+ let state = "start in window";
+ return new Promise(resolve => {
+ sw.onmessage = t.step_func(e => {
+ if (e.data === "start in worker") {
+ assert_equals(state, "start in window");
+ sw.controller.postMessage(result.value);
+ state = "we are expecting confirmation of an onmessageerror in the worker";
+ } else if (e.data === "onmessageerror was received in worker") {
+ assert_equals(state, "we are expecting confirmation of an onmessageerror in the worker");
+ resolve();
+ } else {
+ assert_unreached("Got an unexpected message from the service worker: " + e.data);
+ }
+ });
+
+ sw.controller.postMessage(state);
+ });
+});
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-clone.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-clone.https.html
new file mode 100644
index 0000000000..324c44f193
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-clone.https.html
@@ -0,0 +1,61 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>RTCEncodedVideoFrame can be cloned and distributed</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const videoSender = caller.addTrack(videoTrack)
+ const senderStreams = videoSender.createEncodedStreams();
+ const senderReader = senderStreams.readable.getReader();
+ const senderWriter = senderStreams.writable.getWriter();
+
+ exchangeIceCandidates(caller, callee);
+
+ // Send 10 frames and stop
+ const numFramesToSend = 10;
+
+ const framesReceivedCorrectly = new Promise((resolve, reject) => {
+ callee.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+ const receiverWriter = receiverStreams.writable.getWriter();
+
+ // This should all be able to happen in 5 seconds.
+ // For fast failures, uncomment this line.
+ // t.step_timeout(reject, 5000);
+ for (let i = 0; i < numFramesToSend; i++) {
+ const result = await receiverReader.read();
+ // Write upstream, purely to avoid "no frame received" error messages
+ receiverWriter.write(result.value);
+ }
+ resolve();
+ }
+ });
+
+ await exchangeOfferAnswer(caller, callee);
+
+ for (let i = 0; i < numFramesToSend; i++) {
+ const result = await senderReader.read();
+ senderWriter.write(structuredClone(result.value));
+ }
+ return framesReceivedCorrectly;
+}, "Cloning before sending works");
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-serviceworker-failure.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-serviceworker-failure.https.html
new file mode 100644
index 0000000000..e725e5ce12
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCEncodedVideoFrame-serviceworker-failure.https.html
@@ -0,0 +1,60 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<!-- Based on similar tests in html/infrastructure/safe-passing-of-structured-data/shared-array-buffers/ -->
+<title>RTCEncodedVideoFrame cannot cross agent clusters, service worker edition</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../service-workers/service-worker/resources/test-helpers.sub.js"></script>
+
+<script>
+"use strict";
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const videoSender = caller.addTrack(videoTrack)
+ const senderStreams = videoSender.createEncodedStreams();
+ const senderReader = senderStreams.readable.getReader();
+ const senderWriter = senderStreams.writable.getWriter();
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ const result = await senderReader.read();
+ const scope = "resources/blank.html";
+ const reg = await service_worker_unregister_and_register(t, "resources/serviceworker-failure.js", scope)
+ t.add_cleanup(() => service_worker_unregister(t, scope));
+ await wait_for_state(t, reg.installing, "activated");
+ const iframe = await with_iframe(scope);
+ t.add_cleanup(() => iframe.remove());
+ const sw = iframe.contentWindow.navigator.serviceWorker;
+ let state = "start in window";
+ return new Promise(resolve => {
+ sw.onmessage = t.step_func(e => {
+ if (e.data === "start in worker") {
+ assert_equals(state, "start in window");
+ sw.controller.postMessage(result.value);
+ state = "we are expecting confirmation of an onmessageerror in the worker";
+ } else if (e.data === "onmessageerror was received in worker") {
+ assert_equals(state, "we are expecting confirmation of an onmessageerror in the worker");
+ resolve();
+ } else {
+ assert_unreached("Got an unexpected message from the service worker: " + e.data);
+ }
+ });
+
+ sw.controller.postMessage(state);
+ });
+});
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-audio.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-audio.https.html
new file mode 100644
index 0000000000..83d284146a
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-audio.https.html
@@ -0,0 +1,228 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>RTCPeerConnection Insertable Streams Audio</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="./RTCPeerConnection-insertable-streams.js"></script>
+</head>
+<body>
+<script>
+async function testAudioFlow(t, negotiationFunction, setConstructorParam, perFrameCallback = () => {}) {
+ const caller = new RTCPeerConnection(setConstructorParam ? {encodedInsertableStreams:true} : {});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection(setConstructorParam ? {encodedInsertableStreams:true} : {});
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["microphone"]);
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const audioTrack = stream.getAudioTracks()[0];
+ t.add_cleanup(() => audioTrack.stop());
+
+ const audioSender = caller.addTrack(audioTrack)
+ const senderStreams = audioSender.createEncodedStreams();
+ const senderReader = senderStreams.readable.getReader();
+ const senderWriter = senderStreams.writable.getWriter();
+
+ const frameInfos = [];
+ const numFramesPassthrough = 5;
+ const numFramesReplaceData = 5;
+ const numFramesModifyData = 5;
+ const numFramesToSend = numFramesPassthrough + numFramesReplaceData + numFramesModifyData;
+
+ let streamsCreatedAtNegotiation;
+
+ const ontrackPromise = new Promise(resolve => {
+ callee.ontrack = t.step_func(() => {
+ const audioReceiver = callee.getReceivers().find(r => r.track.kind === 'audio');
+ assert_not_equals(audioReceiver, undefined);
+
+ let receiverReader;
+ let receiverWriter;
+ if (streamsCreatedAtNegotiation) {
+ const audioStreams = streamsCreatedAtNegotiation.find(r => r.kind === 'audio');
+ assert_true(!!audioStreams);
+ receiverReader = audioStreams.streams.readable.getReader();
+ receiverWriter = audioStreams.streams.writable.getWriter();
+ } else {
+ const receiverStreams =
+ audioReceiver.createEncodedStreams();
+ receiverReader = receiverStreams.readable.getReader();
+ receiverWriter = receiverStreams.writable.getWriter();
+ }
+
+ const maxFramesToReceive = numFramesToSend;
+ let numVerifiedFrames = 0;
+ for (let i = 0; i < maxFramesToReceive; i++) {
+ receiverReader.read().then(t.step_func(result => {
+ if (frameInfos[numVerifiedFrames] &&
+ areFrameInfosEqual(result.value, frameInfos[numVerifiedFrames])) {
+ numVerifiedFrames++;
+ } else {
+ // Receiving unexpected frames is an indication that
+ // frames are not passed correctly between sender and receiver.
+ assert_unreached("Incorrect frame received");
+ }
+ assert_not_equals(result.value.getMetadata().sequenceNumber, undefined);
+
+ if (numVerifiedFrames == numFramesToSend)
+ resolve();
+ }));
+ }
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await negotiationFunction(caller, callee, (streams) => {streamsCreatedAtNegotiation = streams;});
+
+ // Pass frames as they come from the encoder.
+ for (let i = 0; i < numFramesPassthrough; i++) {
+ const result = await senderReader.read();
+ const frame = result.value;
+ perFrameCallback(frame);
+ frameInfos.push({
+ data: frame.data,
+ timestamp: frame.timestamp,
+ type: frame.type,
+ metadata: frame.getMetadata(),
+ getMetadata() { return this.metadata; }
+ });
+ senderWriter.write(frame);
+ }
+
+ // Replace frame data with arbitrary buffers.
+ for (let i = 0; i < numFramesReplaceData; i++) {
+ const result = await senderReader.read()
+ const frame = result.value;
+
+ const buffer = new ArrayBuffer(100);
+ const int8View = new Int8Array(buffer);
+ int8View.fill(i);
+
+ frame.data = buffer;
+ perFrameCallback(frame);
+ frameInfos.push({
+ data: frame.data,
+ timestamp: frame.timestamp,
+ type: frame.type,
+ metadata: frame.getMetadata(),
+ getMetadata() { return this.metadata; }
+ });
+ senderWriter.write(frame);
+ }
+
+ // Modify frame data.
+ for (let i = 0; i < numFramesReplaceData; i++) {
+ const result = await senderReader.read()
+ const frame = result.value;
+ const int8View = new Int8Array(frame.data);
+ int8View.fill(i);
+
+ perFrameCallback(frame);
+ frameInfos.push({
+ data: frame.data,
+ timestamp: frame.timestamp,
+ type: frame.type,
+ metadata: frame.getMetadata(),
+ getMetadata() { return this.metadata; }
+ });
+ senderWriter.write(frame);
+ }
+
+ return ontrackPromise;
+}
+
+for (const setConstructorParam of [false, true]) {
+ promise_test(async t => {
+ return testAudioFlow(t, exchangeOfferAnswer, setConstructorParam);
+ }, 'Frames flow correctly using insertable streams' + (setConstructorParam ? ' with param' : ''));
+
+ promise_test(async t => {
+ return testAudioFlow(t, exchangeOfferAnswerReverse, setConstructorParam);
+ }, 'Frames flow correctly using insertable streams when receiver starts negotiation' + (setConstructorParam ? ' with param' : ''));
+}
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+
+ const sender = caller.addTrack(track)
+ const streams = sender.createEncodedStreams();
+ const transformer = new TransformStream({
+ transform(frame, controller) {
+ // Inserting the same frame twice will result in failure since the frame
+ // will be neutered after the first insertion is processed.
+ controller.enqueue(frame);
+ controller.enqueue(frame);
+ }
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ await promise_rejects_dom(
+ t, 'OperationError',
+ streams.readable.pipeThrough(transformer).pipeTo(streams.writable));
+}, 'Enqueuing the same frame twice fails');
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+ const sender = caller.addTrack(track)
+ sender.createEncodedStreams();
+ assert_throws_dom("InvalidStateError", () => sender.createEncodedStreams());
+}, 'Creating streams twice throws');
+
+promise_test(async t => {
+ let clonedFrames = [];
+ function verifyFramesSerializeAndDeserialize(frame) {
+ // Clone encoded frames using structedClone (ie serialize + deserialize) and
+ // keep a reference.
+ const clone = structuredClone(frame);
+ clonedFrames.push(clone);
+ };
+
+ await testAudioFlow(
+ t, exchangeOfferAnswer, /*setConstructorParam=*/false, verifyFramesSerializeAndDeserialize);
+
+ // Ensure all of our cloned frames are still alive and well, despite the
+ // originals having been sent through the PeerConnection.
+ clonedFrames.forEach((clonedFrame) => {
+ assert_not_equals(clonedFrame.data.size, 0);
+ assert_not_equals(clonedFrame.timestamp, 0);
+ });
+}, 'Encoded frames serialize and deserialize into a deep clone');
+
+promise_test(async t => {
+ let clonedFrames = [];
+ function rewriteFrameTimestamps(frame) {
+ // Add 1 to the rtp timestamp of the frame.
+ const metadata = frame.getMetadata();
+ metadata.rtpTimestamp += 1;
+ frame.setMetadata(metadata);
+
+ assert_equals(frame.getMetadata().rtpTimestamp, metadata.rtpTimestamp)
+ };
+
+ // Run audio flows which will assert that the frames received have the
+ // rtp timestamp set by our modification.
+ await testAudioFlow(
+ t, exchangeOfferAnswer, /*setConstructorParam=*/false, rewriteFrameTimestamps);
+}, 'Modifying rtp timestamp');
+
+</script>
+</body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-errors.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-errors.https.html
new file mode 100644
index 0000000000..a0c68c400a
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-errors.https.html
@@ -0,0 +1,46 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>RTCPeerConnection Insertable Streams - Errors</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="./RTCPeerConnection-insertable-streams.js"></script>
+</head>
+<body>
+<script>
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+
+ const sender = caller.addTrack(track)
+ const streams = sender.createEncodedStreams();
+ const transformer = new TransformStream({
+ transform(frame, controller) {
+ // Inserting the same frame twice will result in failure since the frame
+ // will be neutered after the first insertion is processed.
+ controller.enqueue(frame);
+ controller.enqueue(frame);
+ }
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ await promise_rejects_dom(
+ t, 'OperationError',
+ streams.readable.pipeThrough(transformer).pipeTo(streams.writable));
+}, 'Enqueuing the same frame twice fails');
+
+</script>
+</body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-simulcast.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-simulcast.https.html
new file mode 100644
index 0000000000..834644674e
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-simulcast.https.html
@@ -0,0 +1,89 @@
+<!doctype html>
+<meta charset=utf-8>
+<title>RTCPeerConnection Insertable Streams Simulcast</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="../../webrtc/third_party/sdp/sdp.js"></script>
+<script src="../../webrtc/simulcast/simulcast.js"></script>
+<script>
+// Test based on wpt/webrtc/simulcast/basic.https.html
+promise_test(async t => {
+ const rids = [0, 1, 2];
+ const pc1 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => pc1.close());
+ const pc2 = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => pc2.close());
+
+ exchangeIceCandidates(pc1, pc2);
+
+ const metadataToBeLoaded = [];
+ let receiverSSRCs = []
+ pc2.ontrack = t.step_func(e => {
+ const receiverTransformer = new TransformStream({
+ async transform(encodedFrame, controller) {
+ let ssrc = encodedFrame.getMetadata().synchronizationSource;
+ if (receiverSSRCs.indexOf(ssrc) == -1)
+ receiverSSRCs.push(ssrc);
+ controller.enqueue(encodedFrame);
+ }
+ });
+ const receiverStreams = e.receiver.createEncodedStreams();
+ receiverStreams.readable
+ .pipeThrough(receiverTransformer)
+ .pipeTo(receiverStreams.writable);
+
+ const stream = e.streams[0];
+ const v = document.createElement('video');
+ v.autoplay = true;
+ v.srcObject = stream;
+ v.id = stream.id
+ metadataToBeLoaded.push(new Promise((resolve) => {
+ v.addEventListener('loadedmetadata', () => {
+ resolve();
+ });
+ }));
+ });
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video: {width: 1280, height: 720}});
+ t.add_cleanup(() => stream.getTracks().forEach(track => track.stop()));
+ const transceiver = pc1.addTransceiver(stream.getVideoTracks()[0], {
+ streams: [stream],
+ sendEncodings: rids.map(rid => {rid}),
+ });
+ const senderStreams = transceiver.sender.createEncodedStreams();
+ let senderSSRCs = [];
+ const senderTransformer = new TransformStream({
+ async transform(encodedFrame, controller) {
+ if (senderSSRCs.indexOf(encodedFrame.getMetadata().synchronizationSource) == -1)
+ senderSSRCs.push(encodedFrame.getMetadata().synchronizationSource);
+ controller.enqueue(encodedFrame);
+ }
+ });
+ senderStreams.readable
+ .pipeThrough(senderTransformer)
+ .pipeTo(senderStreams.writable);
+
+ const offer = await pc1.createOffer();
+ await pc1.setLocalDescription(offer),
+ await pc2.setRemoteDescription({
+ type: 'offer',
+ sdp: swapRidAndMidExtensionsInSimulcastOffer(offer, rids),
+ });
+ const answer = await pc2.createAnswer();
+ await pc2.setLocalDescription(answer);
+ await pc1.setRemoteDescription({
+ type: 'answer',
+ sdp: swapRidAndMidExtensionsInSimulcastAnswer(answer, pc1.localDescription, rids),
+ });
+ assert_equals(metadataToBeLoaded.length, 3);
+ await Promise.all(metadataToBeLoaded);
+ // Ensure that frames from the 3 simulcast layers are exposed.
+ assert_equals(senderSSRCs.length, 3);
+ assert_equals(receiverSSRCs.length, 3);
+}, 'Basic simulcast setup with three spatial layers');
+</script>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video-frames.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video-frames.https.html
new file mode 100644
index 0000000000..d3db116ff6
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video-frames.https.html
@@ -0,0 +1,80 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>RTCPeerConnection Insertable Streams - Video Frames</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="./RTCPeerConnection-insertable-streams.js"></script>
+</head>
+<body>
+<script>
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+
+ const sender = caller.addTrack(track)
+ const senderStreams = sender.createEncodedStreams();
+ const senderReader = senderStreams.readable.getReader();
+ const senderWriter = senderStreams.writable.getWriter();
+ const numFramesToSend = 20;
+
+ const ontrackPromise = new Promise((resolve, reject) => {
+ callee.ontrack = async e => {
+ const receiverStreams = e.receiver.createEncodedStreams();
+ const receiverReader = receiverStreams.readable.getReader();
+
+ let numReceivedKeyFrames = 0;
+ let numReceivedDeltaFrames = 0;
+ for (let i = 0; i < numFramesToSend; i++) {
+ const result = await receiverReader.read();
+ if (result.value.type == 'key')
+ numReceivedKeyFrames++;
+ else if (result.value.type == 'delta')
+ numReceivedDeltaFrames++;
+
+ if (numReceivedKeyFrames > 0 && numReceivedDeltaFrames > 0)
+ resolve();
+ else if (numReceivedKeyFrames + numReceivedDeltaFrames >= numFramesToSend)
+ reject();
+ }
+ }
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ let numSentKeyFrames = 0;
+ let numSentDeltaFrames = 0;
+ // Pass frames as they come from the encoder.
+ for (let i = 0; i < numFramesToSend; i++) {
+ const result = await senderReader.read();
+ verifyNonstandardAdditionalDataIfPresent(result.value);
+ if (result.value.type == 'key') {
+ numSentKeyFrames++;
+ } else {
+ numSentDeltaFrames++;
+ }
+
+ senderWriter.write(result.value);
+ }
+
+ assert_greater_than(numSentKeyFrames, 0);
+ assert_greater_than(numSentDeltaFrames, 0);
+
+ return ontrackPromise;
+}, 'Key and Delta frames are sent and received');
+
+</script>
+</body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video.https.html
new file mode 100644
index 0000000000..5334b8d1f9
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-video.https.html
@@ -0,0 +1,182 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>RTCPeerConnection Insertable Streams - Video</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="./RTCPeerConnection-insertable-streams.js"></script>
+</head>
+<body>
+<script>
+async function testVideoFlow(t, negotiationFunction, setConstructorParam, frameCallback = () => {}) {
+ const caller = new RTCPeerConnection(setConstructorParam ? {encodedInsertableStreams:true} : {});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection(setConstructorParam ? {encodedInsertableStreams:true} : {});
+ t.add_cleanup(() => callee.close());
+
+ await setMediaPermission("granted", ["camera"]);
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const videoSender = caller.addTrack(videoTrack)
+ const senderStreams = videoSender.createEncodedStreams();
+ const senderReader = senderStreams.readable.getReader();
+ const senderWriter = senderStreams.writable.getWriter();
+
+ const frameInfos = [];
+ const numFramesPassthrough = 5;
+ const numFramesReplaceData = 5;
+ const numFramesModifyData = 5;
+ const numFramesToSend = numFramesPassthrough + numFramesReplaceData + numFramesModifyData;
+
+ let streamsCreatedAtNegotiation;
+ const ontrackPromise = new Promise(resolve => {
+ callee.ontrack = t.step_func(() => {
+ const videoReceiver = callee.getReceivers().find(r => r.track.kind === 'video');
+ assert_not_equals(videoReceiver, undefined);
+
+ let receiverReader;
+ let receiverWriter;
+ if (streamsCreatedAtNegotiation) {
+ const videoStreams = streamsCreatedAtNegotiation.find(r => r.kind === 'video');
+ assert_true(!!videoStreams);
+ receiverReader = videoStreams.streams.readable.getReader();
+ receiverWriter = videoStreams.streams.writable.getWriter();
+ } else {
+ const receiverStreams =
+ videoReceiver.createEncodedStreams();
+ receiverReader = receiverStreams.readable.getReader();
+ receiverWriter = receiverStreams.writable.getWriter();
+ }
+
+ const maxFramesToReceive = numFramesToSend;
+ let numVerifiedFrames = 0;
+ for (let i = 0; i < maxFramesToReceive; i++) {
+ receiverReader.read().then(t.step_func(result => {
+ verifyNonstandardAdditionalDataIfPresent(result.value);
+ if (frameInfos[numVerifiedFrames] &&
+ areFrameInfosEqual(result.value, frameInfos[numVerifiedFrames])) {
+ numVerifiedFrames++;
+ } else {
+ // Receiving unexpected frames is an indication that
+ // frames are not passed correctly between sender and receiver.
+ assert_unreached("Incorrect frame received");
+ }
+
+ if (numVerifiedFrames == numFramesToSend)
+ resolve();
+ }));
+ }
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await negotiationFunction(caller, callee, (streams) => {streamsCreatedAtNegotiation = streams;});
+
+ // Pass frames as they come from the encoder.
+ for (let i = 0; i < numFramesPassthrough; i++) {
+ const result = await senderReader.read();
+ const frame = result.value;
+ const metadata = frame.getMetadata();
+ assert_true(containsVideoMetadata(metadata));
+ verifyNonstandardAdditionalDataIfPresent(frame);
+ frameInfos.push({
+ timestamp: frame.timestamp,
+ type: frame.type,
+ data: frame.data,
+ metadata: metadata,
+ getMetadata() { return this.metadata; }
+ });
+ frameCallback(frame);
+ senderWriter.write(frame);
+ }
+
+ // Replace frame data with arbitrary buffers.
+ for (let i = 0; i < numFramesReplaceData; i++) {
+ const result = await senderReader.read();
+ const metadata = result.value.getMetadata();
+ assert_true(containsVideoMetadata(metadata));
+ const buffer = new ArrayBuffer(100);
+ const int8View = new Int8Array(buffer);
+ int8View.fill(i);
+
+ result.value.data = buffer;
+ frameInfos.push({
+ timestamp: result.value.timestamp,
+ type: result.value.type,
+ data: result.value.data,
+ metadata: metadata,
+ getMetadata() { return this.metadata; }
+ });
+ senderWriter.write(result.value);
+ }
+
+ // Modify frame data.
+ for (let i = 0; i < numFramesReplaceData; i++) {
+ const result = await senderReader.read();
+ const metadata = result.value.getMetadata();
+ assert_true(containsVideoMetadata(metadata));
+ const int8View = new Int8Array(result.value.data);
+ int8View.fill(i);
+
+ frameInfos.push({
+ timestamp: result.value.timestamp,
+ type: result.value.type,
+ data: result.value.data,
+ metadata: metadata,
+ getMetadata() { return this.metadata; }
+ });
+ senderWriter.write(result.value);
+ }
+
+ return ontrackPromise;
+}
+
+for (const setConstructorParam of [false, true]) {
+ promise_test(async t => {
+ return testVideoFlow(t, exchangeOfferAnswer, setConstructorParam);
+ }, 'Frames flow correctly using insertable streams' + (setConstructorParam ? ' with param' : ''));
+
+ promise_test(async t => {
+ return testVideoFlow(t, exchangeOfferAnswerReverse, setConstructorParam);
+ }, 'Frames flow correctly using insertable streams when receiver starts negotiation' + (setConstructorParam ? ' with param' : ''));
+}
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection();
+ t.add_cleanup(() => caller.close());
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+ const sender = caller.addTrack(track)
+ sender.createEncodedStreams();
+ assert_throws_dom("InvalidStateError", () => sender.createEncodedStreams());
+}, 'Creating streams twice throws');
+
+promise_test(async t => {
+ let clonedFrames = [];
+ function verifyFramesSerializeAndDeserialize(frame) {
+ // Clone encoded frames using structedClone (ie serialize + deserialize) and
+ // keep a reference.
+ const clone = structuredClone(frame);
+ clonedFrames.push(clone);
+ };
+
+ await testVideoFlow(t, exchangeOfferAnswer, /*setConstructorParam=*/false, verifyFramesSerializeAndDeserialize);
+
+ // Ensure all of our cloned frames are still alive and well, despite the
+ // originals having been sent through the PeerConnection.
+ clonedFrames.forEach((clonedFrame) => {
+ assert_not_equals(clonedFrame.data.size, 0);
+ assert_not_equals(clonedFrame.type, "empty")
+ });
+}, 'Encoded frames serialize and deserialize into a deep clone');
+
+</script>
+</body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-worker.https.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-worker.https.html
new file mode 100644
index 0000000000..94943f8b69
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams-worker.https.html
@@ -0,0 +1,196 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>RTCPeerConnection Insertable Streams - Worker</title>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+<script src=/resources/testdriver.js></script>
+<script src=/resources/testdriver-vendor.js></script>
+<script src='../../mediacapture-streams/permission-helper.js'></script>
+<script src="../../webrtc/RTCPeerConnection-helper.js"></script>
+<script src="./RTCPeerConnection-insertable-streams.js"></script>
+</head>
+<body>
+<script>
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ // Video is used in a later test, so we ask for both permissions
+ await setMediaPermission();
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const track = stream.getTracks()[0];
+ t.add_cleanup(() => track.stop());
+
+ const sender = caller.addTrack(track)
+ const senderStreams = sender.createEncodedStreams();
+
+ const senderWorker = new Worker('RTCPeerConnection-sender-worker-single-frame.js')
+ t.add_cleanup(() => senderWorker.terminate());
+ senderWorker.postMessage(
+ {readableStream: senderStreams.readable},
+ [senderStreams.readable]);
+
+ let expectedFrameData = null;
+ let verifiedFrameData = false;
+ let numVerifiedFrames = 0;
+ const onmessagePromise = new Promise(resolve => {
+ senderWorker.onmessage = t.step_func(message => {
+ if (!(message.data instanceof RTCEncodedAudioFrame)) {
+ // This is the first message sent from the Worker to the test.
+ // It contains an object (not an RTCEncodedAudioFrame) with the same
+ // fields as the RTCEncodedAudioFrame to be sent in follow-up messages.
+ // These serve as expected values to validate that the
+ // RTCEncodedAudioFrame is sent correctly back to the test in the next
+ // message.
+ expectedFrameData = message.data;
+ } else {
+ // This is the frame sent by the Worker after reading it from the
+ // readable stream. The Worker sends it twice after sending the
+ // verification message.
+ assert_equals(message.data.type, expectedFrameData.type);
+ assert_equals(message.data.timestamp, expectedFrameData.timestamp);
+ assert_true(areArrayBuffersEqual(message.data.data, expectedFrameData.data));
+ if (++numVerifiedFrames == 2)
+ resolve();
+ }
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ return onmessagePromise;
+}, 'RTCRtpSender readable stream transferred to a Worker and the Worker sends an RTCEncodedAudioFrame back');
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const videoSender = caller.addTrack(videoTrack)
+ const senderStreams = videoSender.createEncodedStreams();
+
+ const senderWorker = new Worker('RTCPeerConnection-sender-worker-single-frame.js')
+ t.add_cleanup(() => senderWorker.terminate());
+ senderWorker.postMessage(
+ {readableStream: senderStreams.readable},
+ [senderStreams.readable]);
+
+ let expectedFrameData = null;
+ let verifiedFrameData = false;
+ let numVerifiedFrames = 0;
+ const onmessagePromise = new Promise(resolve => {
+ senderWorker.onmessage = t.step_func(message => {
+ if (!(message.data instanceof RTCEncodedVideoFrame)) {
+ // This is the first message sent from the Worker to the test.
+ // It contains an object (not an RTCEncodedVideoFrame) with the same
+ // fields as the RTCEncodedVideoFrame to be sent in follow-up messages.
+ // These serve as expected values to validate that the
+ // RTCEncodedVideoFrame is sent correctly back to the test in the next
+ // message.
+ expectedFrameData = message.data;
+ } else {
+ // This is the frame sent by the Worker after reading it from the
+ // readable stream. The Worker sends it twice after sending the
+ // verification message.
+ assert_equals(message.data.type, expectedFrameData.type);
+ assert_equals(message.data.timestamp, expectedFrameData.timestamp);
+ assert_true(areArrayBuffersEqual(message.data.data, expectedFrameData.data));
+ assert_equals(message.data.getMetadata().synchronizationSource, expectedFrameData.metadata.synchronizationSource);
+ if (++numVerifiedFrames == 2)
+ resolve();
+ }
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ return onmessagePromise;
+}, 'RTCRtpSender readable stream transferred to a Worker and the Worker sends an RTCEncodedVideoFrame back');
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ const stream = await navigator.mediaDevices.getUserMedia({video:true});
+ const videoTrack = stream.getVideoTracks()[0];
+ t.add_cleanup(() => videoTrack.stop());
+
+ const videoSender = caller.addTrack(videoTrack)
+ const senderStreams = videoSender.createEncodedStreams();
+
+ const senderWorker = new Worker('RTCPeerConnection-worker-transform.js')
+ t.add_cleanup(() => senderWorker.terminate());
+ senderWorker.postMessage(
+ {
+ readableStream: senderStreams.readable,
+ writableStream: senderStreams.writable,
+ insertError: true
+ },
+ [senderStreams.readable, senderStreams.writable]);
+
+ const onmessagePromise = new Promise(resolve => {
+ senderWorker.onmessage = t.step_func(message => {
+ assert_false(message.data.success);
+ assert_true(message.data.error instanceof TypeError);
+ resolve();
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ return onmessagePromise;
+}, 'Video RTCRtpSender insertable streams transferred to a worker, which tries to write an invalid frame');
+
+promise_test(async t => {
+ const caller = new RTCPeerConnection({encodedInsertableStreams:true});
+ t.add_cleanup(() => caller.close());
+ const callee = new RTCPeerConnection();
+ t.add_cleanup(() => callee.close());
+
+ const stream = await navigator.mediaDevices.getUserMedia({audio:true});
+ const audioTrack = stream.getAudioTracks()[0];
+ t.add_cleanup(() => audioTrack.stop());
+
+ const audioSender = caller.addTrack(audioTrack)
+ const senderStreams = audioSender.createEncodedStreams();
+
+ const senderWorker = new Worker('RTCPeerConnection-worker-transform.js')
+ t.add_cleanup(() => senderWorker.terminate());
+ senderWorker.postMessage(
+ {
+ readableStream: senderStreams.readable,
+ writableStream: senderStreams.writable,
+ insertError: true
+ },
+ [senderStreams.readable, senderStreams.writable]);
+
+ const onmessagePromise = new Promise(resolve => {
+ senderWorker.onmessage = t.step_func(message => {
+ assert_false(message.data.success);
+ assert_true(message.data.error instanceof TypeError);
+ resolve();
+ });
+ });
+
+ exchangeIceCandidates(caller, callee);
+ await exchangeOfferAnswer(caller, callee);
+
+ return onmessagePromise;
+}, 'Audio RTCRtpSender insertable streams transferred to a worker, which tries to write an invalid frame');
+
+</script>
+</body>
+</html>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams.js b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams.js
new file mode 100644
index 0000000000..0bf820acde
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-insertable-streams.js
@@ -0,0 +1,262 @@
+function areArrayBuffersEqual(buffer1, buffer2)
+{
+ if (buffer1.byteLength !== buffer2.byteLength) {
+ return false;
+ }
+ let array1 = new Int8Array(buffer1);
+ var array2 = new Int8Array(buffer2);
+ for (let i = 0 ; i < buffer1.byteLength ; ++i) {
+ if (array1[i] !== array2[i]) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function areArraysEqual(a1, a2) {
+ if (a1 === a1)
+ return true;
+ if (a1.length != a2.length)
+ return false;
+ for (let i = 0; i < a1.length; i++) {
+ if (a1[i] != a2[i])
+ return false;
+ }
+ return true;
+}
+
+function areMetadataEqual(metadata1, metadata2, type) {
+ return metadata1.synchronizationSource === metadata2.synchronizationSource &&
+ metadata1.payloadType == metadata2.payloadType &&
+ areArraysEqual(
+ metadata1.contributingSources, metadata2.contributingSources) &&
+ metadata1.absCaptureTime == metadata2.absCaptureTime &&
+ metadata1.frameId === metadata2.frameId &&
+ areArraysEqual(metadata1.dependencies, metadata2.dependencies) &&
+ metadata1.spatialIndex === metadata2.spatialIndex &&
+ metadata1.temporalIndex === metadata2.temporalIndex &&
+ // Width and height are reported only for key frames on the receiver
+ // side.
+ type == 'key' ?
+ metadata1.width === metadata2.width &&
+ metadata1.height === metadata2.height :
+ true;
+}
+
+function areFrameInfosEqual(frame1, frame2) {
+ return frame1.timestamp === frame2.timestamp &&
+ frame1.type === frame2.type &&
+ areMetadataEqual(frame1.getMetadata(), frame2.getMetadata(), frame1.type) &&
+ areArrayBuffersEqual(frame1.data, frame2.data);
+}
+
+function containsVideoMetadata(metadata) {
+ return metadata.synchronizationSource !== undefined &&
+ metadata.width !== undefined &&
+ metadata.height !== undefined &&
+ metadata.spatialIndex !== undefined &&
+ metadata.temporalIndex !== undefined &&
+ metadata.dependencies !== undefined;
+}
+
+function enableExtension(sdp, extension) {
+ if (sdp.indexOf(extension) !== -1)
+ return sdp;
+
+ const extensionIds = sdp.trim().split('\n')
+ .map(line => line.trim())
+ .filter(line => line.startsWith('a=extmap:'))
+ .map(line => line.split(' ')[0].substr(9))
+ .map(id => parseInt(id, 10))
+ .sort((a, b) => a - b);
+ for (let newId = 1; newId <= 15; newId++) {
+ if (!extensionIds.includes(newId)) {
+ return sdp += 'a=extmap:' + newId + ' ' + extension + '\r\n';
+ }
+ }
+ if (sdp.indexOf('a=extmap-allow-mixed') !== -1) { // Pick the next highest one.
+ const newId = extensionIds[extensionIds.length - 1] + 1;
+ return sdp += 'a=extmap:' + newId + ' ' + extension + '\r\n';
+ }
+ throw 'Could not find free extension id to use for ' + extension;
+}
+
+const GFD_V00_EXTENSION =
+ 'http://www.webrtc.org/experiments/rtp-hdrext/generic-frame-descriptor-00';
+const ABS_V00_EXTENSION =
+ 'http://www.webrtc.org/experiments/rtp-hdrext/abs-capture-time';
+
+async function exchangeOfferAnswer(pc1, pc2) {
+ const offer = await pc1.createOffer();
+ // Munge the SDP to enable the GFD and ACT extension in order to get correct
+ // metadata.
+ const sdpABS = enableExtension(offer.sdp, ABS_V00_EXTENSION);
+ const sdpGFD = enableExtension(sdpABS, GFD_V00_EXTENSION);
+ await pc1.setLocalDescription({type: offer.type, sdp: sdpGFD});
+ // Munge the SDP to disable bandwidth probing via RTX.
+ // TODO(crbug.com/1066819): remove this hack when we do not receive duplicates from RTX
+ // anymore.
+ const sdpRTX = sdpGFD.replace(new RegExp('rtx', 'g'), 'invalid');
+ await pc2.setRemoteDescription({type: 'offer', sdp: sdpRTX});
+
+ const answer = await pc2.createAnswer();
+ await pc2.setLocalDescription(answer);
+ await pc1.setRemoteDescription(answer);
+}
+
+async function exchangeOfferAnswerReverse(pc1, pc2, encodedStreamsCallback) {
+ const offer = await pc2.createOffer({offerToReceiveAudio: true, offerToReceiveVideo: true});
+ if (encodedStreamsCallback) {
+ // RTCRtpReceivers will have been created during the above createOffer call, so if the caller
+ // wants to createEncodedStreams synchronously after creation to ensure all frames pass
+ // through the transform, it will have to be done now.
+ encodedStreamsCallback(
+ pc2.getReceivers().map(r => {
+ return {kind: r.track.kind, streams: r.createEncodedStreams()};
+ }));
+ }
+
+ // Munge the SDP to enable the GFD extension in order to get correct metadata.
+ const sdpABS = enableExtension(offer.sdp, ABS_V00_EXTENSION);
+ const sdpGFD = enableExtension(sdpABS, GFD_V00_EXTENSION);
+ // Munge the SDP to disable bandwidth probing via RTX.
+ // TODO(crbug.com/1066819): remove this hack when we do not receive duplicates from RTX
+ // anymore.
+ const sdpRTX = sdpGFD.replace(new RegExp('rtx', 'g'), 'invalid');
+ await pc1.setRemoteDescription({type: 'offer', sdp: sdpRTX});
+ await pc2.setLocalDescription({type: 'offer', sdp: sdpGFD});
+
+ const answer = await pc1.createAnswer();
+ await pc2.setRemoteDescription(answer);
+ await pc1.setLocalDescription(answer);
+}
+
+function createFrameDescriptor(videoFrame) {
+ const kMaxSpatialLayers = 8;
+ const kMaxTemporalLayers = 8;
+ const kMaxNumFrameDependencies = 8;
+
+ const metadata = videoFrame.getMetadata();
+ let frameDescriptor = {
+ beginningOfSubFrame: true,
+ endOfSubframe: false,
+ frameId: metadata.frameId & 0xFFFF,
+ spatialLayers: 1 << metadata.spatialIndex,
+ temporalLayer: metadata.temporalLayer,
+ frameDependenciesDiffs: [],
+ width: 0,
+ height: 0
+ };
+
+ for (const dependency of metadata.dependencies) {
+ frameDescriptor.frameDependenciesDiffs.push(metadata.frameId - dependency);
+ }
+ if (metadata.dependencies.length == 0) {
+ frameDescriptor.width = metadata.width;
+ frameDescriptor.height = metadata.height;
+ }
+ return frameDescriptor;
+}
+
+function additionalDataSize(descriptor) {
+ if (!descriptor.beginningOfSubFrame) {
+ return 1;
+ }
+
+ let size = 4;
+ for (const fdiff of descriptor.frameDependenciesDiffs) {
+ size += (fdiff >= (1 << 6)) ? 2 : 1;
+ }
+ if (descriptor.beginningOfSubFrame &&
+ descriptor.frameDependenciesDiffs.length == 0 &&
+ descriptor.width > 0 &&
+ descriptor.height > 0) {
+ size += 4;
+ }
+
+ return size;
+}
+
+// Compute the buffer reported in the additionalData field using the metadata
+// provided by a video frame.
+// Based on the webrtc::RtpDescriptorAuthentication() C++ function at
+// https://source.chromium.org/chromium/chromium/src/+/main:third_party/webrtc/modules/rtp_rtcp/source/rtp_descriptor_authentication.cc
+function computeAdditionalData(videoFrame) {
+ const kMaxSpatialLayers = 8;
+ const kMaxTemporalLayers = 8;
+ const kMaxNumFrameDependencies = 8;
+
+ const metadata = videoFrame.getMetadata();
+ if (metadata.spatialIndex < 0 ||
+ metadata.temporalIndex < 0 ||
+ metadata.spatialIndex >= kMaxSpatialLayers ||
+ metadata.temporalIndex >= kMaxTemporalLayers ||
+ metadata.dependencies.length > kMaxNumFrameDependencies) {
+ return new ArrayBuffer(0);
+ }
+
+ const descriptor = createFrameDescriptor(videoFrame);
+ const size = additionalDataSize(descriptor);
+ const additionalData = new ArrayBuffer(size);
+ const data = new Uint8Array(additionalData);
+
+ const kFlagBeginOfSubframe = 0x80;
+ const kFlagEndOfSubframe = 0x40;
+ const kFlagFirstSubframeV00 = 0x20;
+ const kFlagLastSubframeV00 = 0x10;
+
+ const kFlagDependencies = 0x08;
+ const kFlagMoreDependencies = 0x01;
+ const kFlageXtendedOffset = 0x02;
+
+ let baseHeader =
+ (descriptor.beginningOfSubFrame ? kFlagBeginOfSubframe : 0) |
+ (descriptor.endOfSubFrame ? kFlagEndOfSubframe : 0);
+ baseHeader |= kFlagFirstSubframeV00;
+ baseHeader |= kFlagLastSubframeV00;
+
+ if (!descriptor.beginningOfSubFrame) {
+ data[0] = baseHeader;
+ return additionalData;
+ }
+
+ data[0] =
+ baseHeader |
+ (descriptor.frameDependenciesDiffs.length == 0 ? 0 : kFlagDependencies) |
+ descriptor.temporalLayer;
+ data[1] = descriptor.spatialLayers;
+ data[2] = descriptor.frameId & 0xFF;
+ data[3] = descriptor.frameId >> 8;
+
+ const fdiffs = descriptor.frameDependenciesDiffs;
+ let offset = 4;
+ if (descriptor.beginningOfSubFrame &&
+ fdiffs.length == 0 &&
+ descriptor.width > 0 &&
+ descriptor.height > 0) {
+ data[offset++] = (descriptor.width >> 8);
+ data[offset++] = (descriptor.width & 0xFF);
+ data[offset++] = (descriptor.height >> 8);
+ data[offset++] = (descriptor.height & 0xFF);
+ }
+ for (let i = 0; i < fdiffs.length; i++) {
+ const extended = fdiffs[i] >= (1 << 6);
+ const more = i < fdiffs.length - 1;
+ data[offset++] = ((fdiffs[i] & 0x3f) << 2) |
+ (extended ? kFlageXtendedOffset : 0) |
+ (more ? kFlagMoreDependencies : 0);
+ if (extended) {
+ data[offset++] = fdiffs[i] >> 6;
+ }
+ }
+ return additionalData;
+}
+
+function verifyNonstandardAdditionalDataIfPresent(videoFrame) {
+ if (videoFrame.additionalData === undefined)
+ return;
+
+ const computedData = computeAdditionalData(videoFrame);
+ assert_true(areArrayBuffersEqual(videoFrame.additionalData, computedData));
+}
+
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-sender-worker-single-frame.js b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-sender-worker-single-frame.js
new file mode 100644
index 0000000000..c943dafe5b
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-sender-worker-single-frame.js
@@ -0,0 +1,19 @@
+onmessage = async (event) => {
+ const readableStream = event.data.readableStream;
+ const reader = readableStream.getReader();
+ const result = await reader.read();
+
+ // Post an object with individual fields so that the test side has
+ // values to verify the serialization of the RTCEncodedVideoFrame.
+ postMessage({
+ type: result.value.type,
+ timestamp: result.value.timestamp,
+ data: result.value.data,
+ metadata: result.value.getMetadata(),
+ });
+
+ // Send the frame twice to verify that the frame does not change after the
+ // first serialization.
+ postMessage(result.value);
+ postMessage(result.value);
+}
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-worker-transform.js b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-worker-transform.js
new file mode 100644
index 0000000000..36e3949e4d
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/RTCPeerConnection-worker-transform.js
@@ -0,0 +1,22 @@
+onmessage = async (event) => {
+ const readableStream = event.data.readableStream;
+ const writableStream = event.data.writableStream;
+ const insertError = event.data.insertError;
+
+ try {
+ await readableStream.pipeThrough(new TransformStream({
+ transform: (chunk, controller) => {
+ if (insertError) {
+ controller.enqueue("This is not a valid frame");
+ } else {
+ controller.enqueue(chunk);
+ }
+ }
+ })).pipeTo(writableStream);
+
+ postMessage({success:true});
+ } catch(e) {
+ postMessage({success:false, error: e});
+ }
+
+}
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/blank.html b/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/blank.html
new file mode 100644
index 0000000000..a3c3a4689a
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/blank.html
@@ -0,0 +1,2 @@
+<!DOCTYPE html>
+<title>Empty doc</title>
diff --git a/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/serviceworker-failure.js b/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/serviceworker-failure.js
new file mode 100644
index 0000000000..e7aa8e11be
--- /dev/null
+++ b/testing/web-platform/tests/webrtc-encoded-transform/tentative/resources/serviceworker-failure.js
@@ -0,0 +1,30 @@
+// Based on similar tests in html/infrastructure/safe-passing-of-structured-data/shared-array-buffers/.
+"use strict";
+self.importScripts("/resources/testharness.js");
+
+let state = "start in worker";
+
+self.onmessage = e => {
+ if (e.data === "start in window") {
+ assert_equals(state, "start in worker");
+ e.source.postMessage(state);
+ state = "we are expecting a messageerror due to the window sending us an RTCEncodedVideoFrame or RTCEncodedAudioFrame";
+ } else {
+ e.source.postMessage(`worker onmessage was reached when in state "${state}" and data ${e.data}`);
+ }
+};
+
+self.onmessageerror = e => {
+ if (state === "we are expecting a messageerror due to the window sending us an RTCEncodedVideoFrame or RTCEncodedAudioFrame") {
+ assert_equals(e.constructor.name, "ExtendableMessageEvent", "type");
+ assert_equals(e.data, null, "data");
+ assert_equals(e.origin, self.origin, "origin");
+ assert_not_equals(e.source, null, "source");
+ assert_equals(e.ports.length, 0, "ports length");
+
+ state = "onmessageerror was received in worker";
+ e.source.postMessage(state);
+ } else {
+ e.source.postMessage(`worker onmessageerror was reached when in state "${state}" and data ${e.data}`);
+ }
+};