diff options
Diffstat (limited to 'dom/media/test')
1140 files changed, 30988 insertions, 0 deletions
diff --git a/dom/media/test/.eslintrc.js b/dom/media/test/.eslintrc.js new file mode 100644 index 0000000000..845ed3f013 --- /dev/null +++ b/dom/media/test/.eslintrc.js @@ -0,0 +1,5 @@ +"use strict"; + +module.exports = { + extends: ["plugin:mozilla/mochitest-test"], +}; diff --git a/dom/media/test/16bit_wave_extrametadata.wav b/dom/media/test/16bit_wave_extrametadata.wav Binary files differnew file mode 100644 index 0000000000..443ec73a3d --- /dev/null +++ b/dom/media/test/16bit_wave_extrametadata.wav diff --git a/dom/media/test/16bit_wave_extrametadata.wav^headers^ b/dom/media/test/16bit_wave_extrametadata.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/16bit_wave_extrametadata.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/320x240.ogv b/dom/media/test/320x240.ogv Binary files differnew file mode 100644 index 0000000000..093158432a --- /dev/null +++ b/dom/media/test/320x240.ogv diff --git a/dom/media/test/320x240.ogv^headers^ b/dom/media/test/320x240.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/320x240.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/448636.ogv b/dom/media/test/448636.ogv Binary files differnew file mode 100644 index 0000000000..628df924f8 --- /dev/null +++ b/dom/media/test/448636.ogv diff --git a/dom/media/test/448636.ogv^headers^ b/dom/media/test/448636.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/448636.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/A4.ogv b/dom/media/test/A4.ogv Binary files differnew file mode 100644 index 0000000000..de99616ece --- /dev/null +++ b/dom/media/test/A4.ogv diff --git a/dom/media/test/A4.ogv^headers^ b/dom/media/test/A4.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/A4.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/VID_0001.ogg b/dom/media/test/VID_0001.ogg Binary files differnew file mode 100644 index 0000000000..0068b9af85 --- /dev/null +++ b/dom/media/test/VID_0001.ogg diff --git a/dom/media/test/VID_0001.ogg^headers^ b/dom/media/test/VID_0001.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/VID_0001.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/allowed.sjs b/dom/media/test/allowed.sjs new file mode 100644 index 0000000000..176a1a24a9 --- /dev/null +++ b/dom/media/test/allowed.sjs @@ -0,0 +1,56 @@ +function parseQuery(request, key) { + var params = request.queryString.split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +var types = { + js: "text/javascript", + m4s: "video/mp4", + mp4: "video/mp4", + ogg: "video/ogg", + ogv: "video/ogg", + oga: "audio/ogg", + webm: "video/webm", + wav: "audio/x-wav" +}; + +// Return file with name as per the query string with access control +// allow headers. +function handleRequest(request, response) +{ + var resource = parseQuery(request, ""); + + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/" + resource; + var split = paths.split("/"); + for(var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + dump("file=" + file + "\n"); + bis.setInputStream(fis); + var bytes = bis.readBytes(bis.available()); + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + response.setHeader("Content-Range", "bytes 0-" + (bytes.length - 1) + "/" + bytes.length); + response.setHeader("Content-Length", ""+bytes.length, false); + var ext = resource.substring(resource.lastIndexOf(".")+1); + response.setHeader("Content-Type", types[ext], false); + response.setHeader("Access-Control-Allow-Origin", "*"); + response.write(bytes, bytes.length); + bis.close(); +} diff --git a/dom/media/test/ambisonics.mp4 b/dom/media/test/ambisonics.mp4 Binary files differnew file mode 100644 index 0000000000..4f5bcdfd26 --- /dev/null +++ b/dom/media/test/ambisonics.mp4 diff --git a/dom/media/test/ambisonics.mp4^headers^ b/dom/media/test/ambisonics.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/ambisonics.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/audio-gaps-short.ogg b/dom/media/test/audio-gaps-short.ogg Binary files differnew file mode 100644 index 0000000000..e01a24bfda --- /dev/null +++ b/dom/media/test/audio-gaps-short.ogg diff --git a/dom/media/test/audio-gaps-short.ogg^headers^ b/dom/media/test/audio-gaps-short.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/audio-gaps-short.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/audio-gaps.ogg b/dom/media/test/audio-gaps.ogg Binary files differnew file mode 100644 index 0000000000..ce96748ccd --- /dev/null +++ b/dom/media/test/audio-gaps.ogg diff --git a/dom/media/test/audio-gaps.ogg^headers^ b/dom/media/test/audio-gaps.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/audio-gaps.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/audio-overhang.ogg b/dom/media/test/audio-overhang.ogg Binary files differnew file mode 100644 index 0000000000..c07986e7a1 --- /dev/null +++ b/dom/media/test/audio-overhang.ogg diff --git a/dom/media/test/audio-overhang.ogg^headers^ b/dom/media/test/audio-overhang.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/audio-overhang.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/audio.wav b/dom/media/test/audio.wav Binary files differnew file mode 100644 index 0000000000..c6fd5cb869 --- /dev/null +++ b/dom/media/test/audio.wav diff --git a/dom/media/test/audio.wav^headers^ b/dom/media/test/audio.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/audio.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/av1.mp4 b/dom/media/test/av1.mp4 Binary files differnew file mode 100644 index 0000000000..28de929a29 --- /dev/null +++ b/dom/media/test/av1.mp4 diff --git a/dom/media/test/av1.mp4^headers^ b/dom/media/test/av1.mp4^headers^ new file mode 100644 index 0000000000..2567dc2fe5 --- /dev/null +++ b/dom/media/test/av1.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store
\ No newline at end of file diff --git a/dom/media/test/background_video.js b/dom/media/test/background_video.js new file mode 100644 index 0000000000..508f8fd89a --- /dev/null +++ b/dom/media/test/background_video.js @@ -0,0 +1,224 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +// This file expects manager to be defined in the global scope. +/* global manager */ +/* import-globals-from manifest.js */ + +"use strict"; + +function startTest(test) { + info(test.desc); + SimpleTest.waitForExplicitFinish(); + SpecialPowers.pushPrefEnv({ set: test.prefs }, () => { + manager.runTests(test.tests, test.runTest); + }); +} + +function nextVideoEnded(video) { + return nextEvent(video, "ended"); +} + +function nextVideoPlaying(video) { + return nextEvent(video, "playing"); +} + +function nextVideoResumes(video) { + return nextEvent(video, "mozexitvideosuspend"); +} + +function nextVideoSuspends(video) { + return nextEvent(video, "mozentervideosuspend"); +} + +/** + * @param {string} url video src. + * @returns {HTMLMediaElement} The created video element. + */ +function appendVideoToDoc(url, token, width, height) { + // Default size of (160, 120) is used by other media tests. + if (width === undefined) { + width = 160; + } + if (height === undefined) { + height = (3 * width) / 4; + } + + let v = document.createElement("video"); + v.token = token; + v.width = width; + v.height = height; + v.src = url; + document.body.appendChild(v); + return v; +} + +function appendVideoToDocWithoutLoad(token, width, height) { + // Default size of (160, 120) is used by other media tests. + if (width === undefined) { + width = 160; + } + if (height === undefined) { + height = (3 * width) / 4; + } + + let v = document.createElement("video"); + v.token = token; + document.body.appendChild(v); + v.width = width; + v.height = height; + return v; +} + +function loadAndWaitUntilLoadedmetadata(video, url, preloadType = "metadata") { + return new Promise((resolve, reject) => { + video.preload = preloadType; + video.addEventListener( + "loadedmetadata", + () => { + resolve(); + }, + true + ); + video.src = url; + }); +} + +/** + * @param {HTMLMediaElement} video Video element with under test. + * @returns {Promise} Promise that is resolved when video 'visibilitychanged' event fires. + */ +function waitUntilVisible(video) { + let videoChrome = SpecialPowers.wrap(video); + if (videoChrome.isInViewPort) { + return Promise.resolve(); + } + + return new Promise(resolve => { + videoChrome.addEventListener("visibilitychanged", () => { + if (videoChrome.isInViewPort) { + ok(true, `${video.token} is visible.`); + videoChrome.removeEventListener("visibilitychanged", this); + resolve(); + } + }); + }); +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video 'playing' event fires. + */ +function waitUntilPlaying(video) { + var p = once(video, "playing", () => { + ok(true, `${video.token} played.`); + }); + Log(video.token, "Start playing"); + video.play(); + return p; +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise which is resolved when video 'ended' event fires. + */ +function waitUntilEnded(video) { + Log(video.token, "Waiting for ended"); + if (video.ended) { + ok(true, video.token + " already ended"); + return Promise.resolve(); + } + + return once(video, "ended", () => { + ok(true, `${video.token} ended`); + }); +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode starts + * suspend timer. + */ +function testSuspendTimerStartedWhenHidden(video) { + var p = once(video, "mozstartvideosuspendtimer").then(() => { + ok(true, `${video.token} suspend begins`); + }); + Log(video.token, "Set Hidden"); + video.setVisible(false); + return p; +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode suspends. + */ +function testVideoSuspendsWhenHidden(video) { + let p = once(video, "mozentervideosuspend").then(() => { + ok(true, `${video.token} suspends`); + }); + Log(video.token, "Set hidden"); + video.setVisible(false); + return p; +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode resumes. + */ +function testVideoResumesWhenShown(video) { + var p = once(video, "mozexitvideosuspend").then(() => { + ok(true, `${video.token} resumes`); + }); + Log(video.token, "Set visible"); + video.setVisible(true); + return p; +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode resumes. + */ +function testVideoOnlySeekCompletedWhenShown(video) { + var p = once(video, "mozvideoonlyseekcompleted").then(() => { + ok(true, `${video.token} resumes`); + }); + Log(video.token, "Set visible"); + video.setVisible(true); + return p; +} + +/** + * @param {HTMLVideoElement} video Video element under test. + * @returns {Promise} Promise that is resolved if video ends and rejects if video suspends. + */ +function checkVideoDoesntSuspend(video) { + let p = Promise.race([ + waitUntilEnded(video).then(() => { + ok(true, `${video.token} ended before decode was suspended`); + }), + once(video, "mozentervideosuspend", () => { + Promise.reject(new Error(`${video.token} suspended`)); + }), + ]); + Log(video.token, "Set hidden."); + video.setVisible(false); + return p; +} + +/** + * @param {HTMLMediaElement} video Video element under test. + * @param {number} time video current time to wait til. + * @returns {Promise} Promise that is resolved once currentTime passes time. + */ +function waitTil(video, time) { + Log(video.token, `Waiting for time to reach ${time}s`); + return new Promise(resolve => { + video.addEventListener("timeupdate", function timeUpdateEvent() { + if (video.currentTime > time) { + video.removeEventListener(name, timeUpdateEvent); + resolve(); + } + }); + }); +} diff --git a/dom/media/test/badtags.ogg b/dom/media/test/badtags.ogg Binary files differnew file mode 100644 index 0000000000..12d8358730 --- /dev/null +++ b/dom/media/test/badtags.ogg diff --git a/dom/media/test/badtags.ogg^headers^ b/dom/media/test/badtags.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/badtags.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4 b/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4 Binary files differnew file mode 100644 index 0000000000..dc4f197ffa --- /dev/null +++ b/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4 diff --git a/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4 b/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4 Binary files differnew file mode 100644 index 0000000000..916c64e9ee --- /dev/null +++ b/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4 diff --git a/dom/media/test/beta-phrasebook.ogg b/dom/media/test/beta-phrasebook.ogg Binary files differnew file mode 100644 index 0000000000..7e6ef77ec4 --- /dev/null +++ b/dom/media/test/beta-phrasebook.ogg diff --git a/dom/media/test/beta-phrasebook.ogg^headers^ b/dom/media/test/beta-phrasebook.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/beta-phrasebook.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s Binary files differnew file mode 100644 index 0000000000..266ec4c100 --- /dev/null +++ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4 b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4 Binary files differnew file mode 100644 index 0000000000..7aeb3eca8a --- /dev/null +++ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4 diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/big-short.wav b/dom/media/test/big-short.wav Binary files differnew file mode 100644 index 0000000000..c850e5fd14 --- /dev/null +++ b/dom/media/test/big-short.wav diff --git a/dom/media/test/big-short.wav^headers^ b/dom/media/test/big-short.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/big-short.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/big.wav b/dom/media/test/big.wav Binary files differnew file mode 100644 index 0000000000..5f66bc1f02 --- /dev/null +++ b/dom/media/test/big.wav diff --git a/dom/media/test/big.wav^headers^ b/dom/media/test/big.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/big.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-audio-key1.xml b/dom/media/test/bipbop-cenc-audio-key1.xml new file mode 100644 index 0000000000..a1672eecef --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio-key1.xml @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To + generate the encrypted files, run bipbop-cenc.sh +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 1 --> + <BS bits="32" value="1" /> + <!-- KeyID --> + <BS ID128="0x7e571d047e571d047e571d047e571d21" /> + </DRMInfo> + + <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d047e571d047e571d047e571d21" + value="0x7e5744447e5744447e5744447e574421" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/bipbop-cenc-audio-key2.xml b/dom/media/test/bipbop-cenc-audio-key2.xml new file mode 100644 index 0000000000..b706609052 --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio-key2.xml @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To + generate the encrypted files, run bipbop-cenc.sh +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 1 --> + <BS bits="32" value="1" /> + <!-- KeyID --> + <BS ID128="0x7e571d047e571d047e571d047e571d22" /> + </DRMInfo> + + <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d047e571d047e571d047e571d22" + value="0x7e5744447e5744447e5744447e574422" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/bipbop-cenc-audio1.m4s b/dom/media/test/bipbop-cenc-audio1.m4s Binary files differnew file mode 100644 index 0000000000..63cfd66f7e --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio1.m4s diff --git a/dom/media/test/bipbop-cenc-audio1.m4s^headers^ b/dom/media/test/bipbop-cenc-audio1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-audio2.m4s b/dom/media/test/bipbop-cenc-audio2.m4s Binary files differnew file mode 100644 index 0000000000..04a6cb6ff9 --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio2.m4s diff --git a/dom/media/test/bipbop-cenc-audio2.m4s^headers^ b/dom/media/test/bipbop-cenc-audio2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-audio3.m4s b/dom/media/test/bipbop-cenc-audio3.m4s Binary files differnew file mode 100644 index 0000000000..ad0cd72f90 --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio3.m4s diff --git a/dom/media/test/bipbop-cenc-audio3.m4s^headers^ b/dom/media/test/bipbop-cenc-audio3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-audio3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-audioinit.mp4 b/dom/media/test/bipbop-cenc-audioinit.mp4 Binary files differnew file mode 100644 index 0000000000..b827aa49aa --- /dev/null +++ b/dom/media/test/bipbop-cenc-audioinit.mp4 diff --git a/dom/media/test/bipbop-cenc-audioinit.mp4^headers^ b/dom/media/test/bipbop-cenc-audioinit.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-audioinit.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-video-10s.mp4 b/dom/media/test/bipbop-cenc-video-10s.mp4 Binary files differnew file mode 100644 index 0000000000..abbe4561fd --- /dev/null +++ b/dom/media/test/bipbop-cenc-video-10s.mp4 diff --git a/dom/media/test/bipbop-cenc-video-10s.mp4^headers^ b/dom/media/test/bipbop-cenc-video-10s.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-video-10s.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-video-key1.xml b/dom/media/test/bipbop-cenc-video-key1.xml new file mode 100644 index 0000000000..f0d9878fa2 --- /dev/null +++ b/dom/media/test/bipbop-cenc-video-key1.xml @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To + generate the encrypted files, run bipbop-cenc.sh +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 1 --> + <BS bits="32" value="1" /> + <!-- KeyID --> + <BS ID128="0x7e571d037e571d037e571d037e571d11" /> + </DRMInfo> + + <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d037e571d037e571d037e571d11" + value="0x7e5733337e5733337e5733337e573311" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/bipbop-cenc-video-key2.xml b/dom/media/test/bipbop-cenc-video-key2.xml new file mode 100644 index 0000000000..1f320e6336 --- /dev/null +++ b/dom/media/test/bipbop-cenc-video-key2.xml @@ -0,0 +1,28 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To + generate the encrypted files, run bipbop-cenc.sh +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 1 --> + <BS bits="32" value="1" /> + <!-- KeyID --> + <BS ID128="0x7e571d037e571d037e571d037e571d12" /> + </DRMInfo> + + <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d037e571d037e571d037e571d12" + value="0x7e5733337e5733337e5733337e573312" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/bipbop-cenc-video1.m4s b/dom/media/test/bipbop-cenc-video1.m4s Binary files differnew file mode 100644 index 0000000000..755013c11c --- /dev/null +++ b/dom/media/test/bipbop-cenc-video1.m4s diff --git a/dom/media/test/bipbop-cenc-video1.m4s^headers^ b/dom/media/test/bipbop-cenc-video1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-video1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-video2.m4s b/dom/media/test/bipbop-cenc-video2.m4s Binary files differnew file mode 100644 index 0000000000..c884bd95fc --- /dev/null +++ b/dom/media/test/bipbop-cenc-video2.m4s diff --git a/dom/media/test/bipbop-cenc-video2.m4s^headers^ b/dom/media/test/bipbop-cenc-video2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-video2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc-videoinit.mp4 b/dom/media/test/bipbop-cenc-videoinit.mp4 Binary files differnew file mode 100644 index 0000000000..aa87d0bbe6 --- /dev/null +++ b/dom/media/test/bipbop-cenc-videoinit.mp4 diff --git a/dom/media/test/bipbop-cenc-videoinit.mp4^headers^ b/dom/media/test/bipbop-cenc-videoinit.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-cenc-videoinit.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-cenc.sh b/dom/media/test/bipbop-cenc.sh new file mode 100644 index 0000000000..a00c38ae80 --- /dev/null +++ b/dom/media/test/bipbop-cenc.sh @@ -0,0 +1,29 @@ +mkdir work.tmp + +for r in 225w_175kbps 300_215kbps 300wp_227kbps 360w_253kbps 480_624kbps 480wp_663kbps 480_959kbps 480wp_1001kbps +do + for k in 1 2 + do + # Encrypt bipbop_<res>.mp4 with the keys specified in this file, + # and output to |bipbop_<res>-cenc-{video,audio}.mp4| + MP4Box -crypt bipbop-cenc-audio-key${k}.xml -rem 1 -out work.tmp/bipbop_${r}-cenc-audio-key${k}.mp4 bipbop_${r}.mp4 + MP4Box -crypt bipbop-cenc-video-key${k}.xml -rem 2 -out work.tmp/bipbop_${r}-cenc-video-key${k}.mp4 bipbop_${r}.mp4 + + # Fragment |bipbop_<res>-cenc-*.mp4| into 500ms segments: + MP4Box -dash 500 -rap -segment-name work.tmp/bipbop_${r}-cenc-audio-key${k}- -subsegs-per-sidx 5 work.tmp/bipbop_${r}-cenc-audio-key${k}.mp4 + MP4Box -dash 500 -rap -segment-name work.tmp/bipbop_${r}-cenc-video-key${k}- -subsegs-per-sidx 5 work.tmp/bipbop_${r}-cenc-video-key${k}.mp4 + + # The above command will generate a set of fragments |bipbop_<res>-cenc-{video,audio}-*.m4s + # and |bipbop_<res>-cenc-{video,audio}-init.mp4| containing just the init segment. + + # Remove unneeded mpd files. + rm bipbop_${r}-cenc-{audio,video}-key${k}_dash.mpd + done +done + +# Only keep the first 4 audio & 2 video segments: +cp work.tmp/*-init[.]mp4 ./ +cp work.tmp/*audio*-[1234][.]m4s ./ +cp work.tmp/*video*-[12][.]m4s ./ + +rm -Rf work.tmp diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4 b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4 Binary files differnew file mode 100644 index 0000000000..5e5e30c255 --- /dev/null +++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4 diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^ new file mode 100644 index 0000000000..12a01c4a22 --- /dev/null +++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store
diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4 b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4 Binary files differnew file mode 100644 index 0000000000..447c657475 --- /dev/null +++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4 diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^ new file mode 100644 index 0000000000..12a01c4a22 --- /dev/null +++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store
diff --git a/dom/media/test/bipbop-frag-cenc.xml b/dom/media/test/bipbop-frag-cenc.xml new file mode 100644 index 0000000000..6f6a4d90a9 --- /dev/null +++ b/dom/media/test/bipbop-frag-cenc.xml @@ -0,0 +1,57 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to |bipbop-cenc*|. To + generate the bipbop-cenc files, run the following commands: + + Encrypt bipbop-no-edts.mp4 with the keys specified in this file, + and output to |bipbop-cenc-{video,audio}.mp4| + MP4Box -crypt bipbop-frag-cenc.xml -rem 2 -out bipbop-cenc-video.mp4 bipbop-no-edts.mp4 + MP4Box -crypt bipbop-frag-cenc.xml -rem 1 -out bipbop-cenc-audio.mp4 bipbop-no-edts.mp4 + + Fragment |bipbop-cenc-*.mp4| into 500ms segments: + MP4Box -dash 500 -rap -segment-name bipbop-cenc-video -subsegs-per-sidx 5 bipbop-cenc-video.mp4 + MP4Box -dash 500 -rap -segment-name bipbop-cenc-audio -subsegs-per-sidx 5 bipbop-cenc-audio.mp4 + + The above command will generate a set of fragments in |bipbop-cenc-{video,audio}*.m4s + and |bipbop-cenc-{video,audio}init.mp4| containing just the init segment. + + To cut down the duration, we throw out all but the first 3 audio & 2 video segments: + rm bipbop-cenc-audio{[^123],[123][^.]}.m4s + rm bipbop-cenc-video{[^12],[12][^.]}.m4s + + MP4Box will also have generated some *.mpd files we don't need: + rm bipbop-cenc-*.mpd + + Delete intermediate encrypted files: + rm bipbop-cenc-{audio,video}.mp4 +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 2 --> + <BS bits="32" value="2" /> + <!-- KeyID --> + <BS ID128="0x7e571d037e571d037e571d037e571d03" /> + <BS ID128="0x7e571d047e571d047e571d047e571d04" /> + </DRMInfo> + + <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d037e571d037e571d037e571d03" + value="0x7e5733337e5733337e5733337e573333" /> + </CrypTrack> + + <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d047e571d047e571d047e571d04" + value="0x7e5744447e5744447e5744447e574444" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/bipbop-lateaudio.mp4 b/dom/media/test/bipbop-lateaudio.mp4 Binary files differnew file mode 100644 index 0000000000..5b4cc57095 --- /dev/null +++ b/dom/media/test/bipbop-lateaudio.mp4 diff --git a/dom/media/test/bipbop-lateaudio.mp4^headers^ b/dom/media/test/bipbop-lateaudio.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop-lateaudio.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop-no-edts.mp4 b/dom/media/test/bipbop-no-edts.mp4 Binary files differnew file mode 100644 index 0000000000..63435887df --- /dev/null +++ b/dom/media/test/bipbop-no-edts.mp4 diff --git a/dom/media/test/bipbop.mp4 b/dom/media/test/bipbop.mp4 Binary files differnew file mode 100644 index 0000000000..017d658f31 --- /dev/null +++ b/dom/media/test/bipbop.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..40c3a7bb98 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..986e5fb186 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..547950e516 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..3214f131d4 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..08713078d9 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..0b13fed5f0 --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_225w_175kbps.mp4 b/dom/media/test/bipbop_225w_175kbps.mp4 Binary files differnew file mode 100644 index 0000000000..abe37b9f9d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps.mp4 diff --git a/dom/media/test/bipbop_225w_175kbps.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_225w_175kbps.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..21f3863274 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..bc741cdf86 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..9c6818d06f --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..f327aaa573 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..543f18c24b --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..f850ceaf0a --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..a28a106daf --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..a05a879970 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300_215kbps.mp4 b/dom/media/test/bipbop_300_215kbps.mp4 Binary files differnew file mode 100644 index 0000000000..084d477430 --- /dev/null +++ b/dom/media/test/bipbop_300_215kbps.mp4 diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..40c3a7bb98 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..986e5fb186 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..9c6818d06f --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..ce2e64eb33 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..8592a5b0a3 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..f850ceaf0a --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..d07ce9753e --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..9d2fa23dd4 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_300wp_227kbps.mp4 b/dom/media/test/bipbop_300wp_227kbps.mp4 Binary files differnew file mode 100644 index 0000000000..1499355313 --- /dev/null +++ b/dom/media/test/bipbop_300wp_227kbps.mp4 diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..40c3a7bb98 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..986e5fb186 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..a571d47cfb --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..42dbfec1ed --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..9e4224cac8 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..21763ecbdd --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm Binary files differnew file mode 100644 index 0000000000..4be8f340c3 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm Binary files differnew file mode 100644 index 0000000000..56cf4c483c --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm Binary files differnew file mode 100644 index 0000000000..9f411d0e34 --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_360w_253kbps.mp4 b/dom/media/test/bipbop_360w_253kbps.mp4 Binary files differnew file mode 100644 index 0000000000..6c796f4e1f --- /dev/null +++ b/dom/media/test/bipbop_360w_253kbps.mp4 diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..e626fa4564 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..d7cbb2b6b0 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..805f4bbf3f --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..5bf9994733 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..77c7daba5a --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..c5127beec9 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..b0ff51f74a --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..cfa099c043 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_624kbps.mp4 b/dom/media/test/bipbop_480_624kbps.mp4 Binary files differnew file mode 100644 index 0000000000..27928b85f4 --- /dev/null +++ b/dom/media/test/bipbop_480_624kbps.mp4 diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..c9106aad99 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..888b20ab63 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..796ad13670 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..d02be53198 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..6e0c60f986 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..06778e6f2b --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..4c1c603e8d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..f4a98eca97 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480_959kbps.mp4 b/dom/media/test/bipbop_480_959kbps.mp4 Binary files differnew file mode 100644 index 0000000000..4a9f2ee823 --- /dev/null +++ b/dom/media/test/bipbop_480_959kbps.mp4 diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..416bc7a7ca --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..73d542cfe0 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..796ad13670 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..80824e9ffc --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..5db21d091b --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..06778e6f2b --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..38a081187a --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..bc8bddf505 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_1001kbps.mp4 b/dom/media/test/bipbop_480wp_1001kbps.mp4 Binary files differnew file mode 100644 index 0000000000..600376cf83 --- /dev/null +++ b/dom/media/test/bipbop_480wp_1001kbps.mp4 diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..e2bd754c7e --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..347835feee --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s Binary files differnew file mode 100644 index 0000000000..64b0da69a0 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s Binary files differnew file mode 100644 index 0000000000..864f4248af --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..416bc7a7ca --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4 diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..a8896e069a --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..0f0a35ce79 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s Binary files differnew file mode 100644 index 0000000000..fece52ff42 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s Binary files differnew file mode 100644 index 0000000000..70e61e3d5f --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..73d542cfe0 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4 diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s Binary files differnew file mode 100644 index 0000000000..805f4bbf3f --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s Binary files differnew file mode 100644 index 0000000000..0a40d1cb73 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4 Binary files differnew file mode 100644 index 0000000000..5db21d091b --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4 diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s Binary files differnew file mode 100644 index 0000000000..c5127beec9 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s Binary files differnew file mode 100644 index 0000000000..3f344022a4 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4 Binary files differnew file mode 100644 index 0000000000..bc8bddf505 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4 diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bipbop_480wp_663kbps.mp4 b/dom/media/test/bipbop_480wp_663kbps.mp4 Binary files differnew file mode 100644 index 0000000000..3cc1da69d2 --- /dev/null +++ b/dom/media/test/bipbop_480wp_663kbps.mp4 diff --git a/dom/media/test/black100x100-aspect3to2.ogv b/dom/media/test/black100x100-aspect3to2.ogv Binary files differnew file mode 100644 index 0000000000..81fe51ffb3 --- /dev/null +++ b/dom/media/test/black100x100-aspect3to2.ogv diff --git a/dom/media/test/black100x100-aspect3to2.ogv^headers^ b/dom/media/test/black100x100-aspect3to2.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/black100x100-aspect3to2.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bogus.duh b/dom/media/test/bogus.duh new file mode 100644 index 0000000000..528ae275d0 --- /dev/null +++ b/dom/media/test/bogus.duh @@ -0,0 +1,45 @@ +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus diff --git a/dom/media/test/bogus.ogv b/dom/media/test/bogus.ogv new file mode 100644 index 0000000000..528ae275d0 --- /dev/null +++ b/dom/media/test/bogus.ogv @@ -0,0 +1,45 @@ +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus diff --git a/dom/media/test/bogus.ogv^headers^ b/dom/media/test/bogus.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bogus.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bogus.wav b/dom/media/test/bogus.wav new file mode 100644 index 0000000000..528ae275d0 --- /dev/null +++ b/dom/media/test/bogus.wav @@ -0,0 +1,45 @@ +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus +bogus bogus bogus diff --git a/dom/media/test/bogus.wav^headers^ b/dom/media/test/bogus.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bogus.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/browser/browser.ini b/dom/media/test/browser/browser.ini new file mode 100644 index 0000000000..d828f4b6f9 --- /dev/null +++ b/dom/media/test/browser/browser.ini @@ -0,0 +1,7 @@ +[DEFAULT] +tags = mediacontrol +support-files = + file_media.html + ../gizmo.mp4 + +[browser_tab_visibility_and_play_time.js] diff --git a/dom/media/test/browser/browser_tab_visibility_and_play_time.js b/dom/media/test/browser/browser_tab_visibility_and_play_time.js new file mode 100644 index 0000000000..4eb956b9ea --- /dev/null +++ b/dom/media/test/browser/browser_tab_visibility_and_play_time.js @@ -0,0 +1,120 @@ +/** + * This test is used to ensure that invisible play time would be accumulated + * when tab is in background. However, this test won't directly check the + * reported telemetry result, because we can't check the snapshot histogram in + * the content process. + * The actual probe checking happens in `test_accumulated_play_time.html`. + */ +"use strict"; + +const PAGE_URL = + "https://example.com/browser/dom/media/test/browser/file_media.html"; + +add_task(async function testChangingTabVisibilityAffectsInvisiblePlayTime() { + const originalTab = gBrowser.selectedTab; + const mediaTab = await openMediaTab(PAGE_URL); + + info(`measuring play time when tab is in foreground`); + await startMedia({ + mediaTab, + shouldAccumulateTime: true, + shouldAccumulateInvisibleTime: false, + }); + await pauseMedia(mediaTab); + + info(`measuring play time when tab is in foreground`); + await BrowserTestUtils.switchTab(window.gBrowser, originalTab); + await startMedia({ + mediaTab, + shouldAccumulateTime: true, + shouldAccumulateInvisibleTime: true, + }); + await pauseMedia(mediaTab); + + BrowserTestUtils.removeTab(mediaTab); +}); + +/** + * Following are helper functions. + */ +async function openMediaTab(url) { + info(`open tab for media playback`); + const tab = await BrowserTestUtils.openNewForegroundTab(window.gBrowser, url); + info(`add content helper functions and variables`); + await SpecialPowers.spawn(tab.linkedBrowser, [], _ => { + content.assertAttributeDefined = (videoChrome, checkType) => { + ok(videoChrome[checkType] != undefined, `${checkType} exists`); + }; + content.assertValueEqualTo = (videoChrome, checkType, expectedValue) => { + content.assertAttributeDefined(videoChrome, checkType); + is( + videoChrome[checkType], + expectedValue, + `${checkType} equals to ${expectedValue}` + ); + }; + content.assertValueConstantlyIncreases = (videoChrome, checkType) => { + content.assertAttributeDefined(videoChrome, checkType); + const valueSnapshot = videoChrome[checkType]; + ok( + videoChrome[checkType] > valueSnapshot, + `${checkType} keeps increasing` + ); + }; + content.assertValueKeptUnchanged = (videoChrome, checkType) => { + content.assertAttributeDefined(videoChrome, checkType); + const valueSnapshot = videoChrome[checkType]; + ok( + videoChrome[checkType] == valueSnapshot, + `${checkType} keeps unchanged` + ); + }; + }); + return tab; +} + +function startMedia({ + mediaTab, + shouldAccumulateTime, + shouldAccumulateInvisibleTime, +}) { + return SpecialPowers.spawn( + mediaTab.linkedBrowser, + [shouldAccumulateTime, shouldAccumulateInvisibleTime], + async (accumulateTime, accumulateInvisibleTime) => { + const video = content.document.getElementById("video"); + ok( + await video.play().then( + () => true, + () => false + ), + "video started playing" + ); + const videoChrome = SpecialPowers.wrap(video); + if (accumulateTime) { + content.assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + } else { + content.assertValueKeptUnchanged(videoChrome, "totalPlayTime"); + } + if (accumulateInvisibleTime) { + content.assertValueConstantlyIncreases( + videoChrome, + "invisiblePlayTime" + ); + } else { + content.assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + } + } + ); +} + +function pauseMedia(tab) { + return SpecialPowers.spawn(tab.linkedBrowser, [], async _ => { + const video = content.document.getElementById("video"); + video.pause(); + ok(true, "video paused"); + const videoChrome = SpecialPowers.wrap(video); + content.assertValueKeptUnchanged(videoChrome, "totalPlayTime"); + content.assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + }); +} diff --git a/dom/media/test/browser/file_media.html b/dom/media/test/browser/file_media.html new file mode 100644 index 0000000000..498c2eaad6 --- /dev/null +++ b/dom/media/test/browser/file_media.html @@ -0,0 +1,9 @@ +<!DOCTYPE html> +<html> +<head> +<title>Non-Autoplay page</title> +</head> +<body> +<video id="video" src="gizmo.mp4" loop></video> +</body> +</html> diff --git a/dom/media/test/bug1066943.webm b/dom/media/test/bug1066943.webm Binary files differnew file mode 100644 index 0000000000..64a24ec898 --- /dev/null +++ b/dom/media/test/bug1066943.webm diff --git a/dom/media/test/bug1066943.webm^headers^ b/dom/media/test/bug1066943.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug1066943.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug1301226-odd.wav b/dom/media/test/bug1301226-odd.wav Binary files differnew file mode 100644 index 0000000000..dd2df4e4dd --- /dev/null +++ b/dom/media/test/bug1301226-odd.wav diff --git a/dom/media/test/bug1301226-odd.wav^headers^ b/dom/media/test/bug1301226-odd.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug1301226-odd.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug1301226.wav b/dom/media/test/bug1301226.wav Binary files differnew file mode 100644 index 0000000000..0128486f07 --- /dev/null +++ b/dom/media/test/bug1301226.wav diff --git a/dom/media/test/bug1301226.wav^headers^ b/dom/media/test/bug1301226.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug1301226.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug1377278.webm b/dom/media/test/bug1377278.webm Binary files differnew file mode 100644 index 0000000000..802019f39f --- /dev/null +++ b/dom/media/test/bug1377278.webm diff --git a/dom/media/test/bug1377278.webm^headers^ b/dom/media/test/bug1377278.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug1377278.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug461281.ogg b/dom/media/test/bug461281.ogg Binary files differnew file mode 100644 index 0000000000..d7f6a0ccf4 --- /dev/null +++ b/dom/media/test/bug461281.ogg diff --git a/dom/media/test/bug461281.ogg^headers^ b/dom/media/test/bug461281.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug461281.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug482461-theora.ogv b/dom/media/test/bug482461-theora.ogv Binary files differnew file mode 100644 index 0000000000..941b8d8efd --- /dev/null +++ b/dom/media/test/bug482461-theora.ogv diff --git a/dom/media/test/bug482461-theora.ogv^headers^ b/dom/media/test/bug482461-theora.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug482461-theora.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug482461.ogv b/dom/media/test/bug482461.ogv Binary files differnew file mode 100644 index 0000000000..6cf6aed330 --- /dev/null +++ b/dom/media/test/bug482461.ogv diff --git a/dom/media/test/bug482461.ogv^headers^ b/dom/media/test/bug482461.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug482461.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug495129.ogv b/dom/media/test/bug495129.ogv Binary files differnew file mode 100644 index 0000000000..44eb9296f5 --- /dev/null +++ b/dom/media/test/bug495129.ogv diff --git a/dom/media/test/bug495129.ogv^headers^ b/dom/media/test/bug495129.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug495129.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug495794.ogg b/dom/media/test/bug495794.ogg Binary files differnew file mode 100644 index 0000000000..1c19a64061 --- /dev/null +++ b/dom/media/test/bug495794.ogg diff --git a/dom/media/test/bug495794.ogg^headers^ b/dom/media/test/bug495794.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug495794.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug498380.ogv b/dom/media/test/bug498380.ogv Binary files differnew file mode 100644 index 0000000000..1179ecb70a --- /dev/null +++ b/dom/media/test/bug498380.ogv diff --git a/dom/media/test/bug498380.ogv^headers^ b/dom/media/test/bug498380.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug498380.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug498855-1.ogv b/dom/media/test/bug498855-1.ogv Binary files differnew file mode 100644 index 0000000000..95a524da4c --- /dev/null +++ b/dom/media/test/bug498855-1.ogv diff --git a/dom/media/test/bug498855-1.ogv^headers^ b/dom/media/test/bug498855-1.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug498855-1.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug498855-2.ogv b/dom/media/test/bug498855-2.ogv Binary files differnew file mode 100644 index 0000000000..795a308ae1 --- /dev/null +++ b/dom/media/test/bug498855-2.ogv diff --git a/dom/media/test/bug498855-2.ogv^headers^ b/dom/media/test/bug498855-2.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug498855-2.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug498855-3.ogv b/dom/media/test/bug498855-3.ogv Binary files differnew file mode 100644 index 0000000000..714858dfed --- /dev/null +++ b/dom/media/test/bug498855-3.ogv diff --git a/dom/media/test/bug498855-3.ogv^headers^ b/dom/media/test/bug498855-3.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug498855-3.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug499519.ogv b/dom/media/test/bug499519.ogv Binary files differnew file mode 100644 index 0000000000..62c0922d36 --- /dev/null +++ b/dom/media/test/bug499519.ogv diff --git a/dom/media/test/bug499519.ogv^headers^ b/dom/media/test/bug499519.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug499519.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug500311.ogv b/dom/media/test/bug500311.ogv Binary files differnew file mode 100644 index 0000000000..2cf27ef1ee --- /dev/null +++ b/dom/media/test/bug500311.ogv diff --git a/dom/media/test/bug500311.ogv^headers^ b/dom/media/test/bug500311.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug500311.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug501279.ogg b/dom/media/test/bug501279.ogg Binary files differnew file mode 100644 index 0000000000..e266f07ee8 --- /dev/null +++ b/dom/media/test/bug501279.ogg diff --git a/dom/media/test/bug501279.ogg^headers^ b/dom/media/test/bug501279.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug501279.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug504613.ogv b/dom/media/test/bug504613.ogv Binary files differnew file mode 100644 index 0000000000..5c7fd015e9 --- /dev/null +++ b/dom/media/test/bug504613.ogv diff --git a/dom/media/test/bug504613.ogv^headers^ b/dom/media/test/bug504613.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug504613.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug504644.ogv b/dom/media/test/bug504644.ogv Binary files differnew file mode 100644 index 0000000000..46fb4a876b --- /dev/null +++ b/dom/media/test/bug504644.ogv diff --git a/dom/media/test/bug504644.ogv^headers^ b/dom/media/test/bug504644.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug504644.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug504843.ogv b/dom/media/test/bug504843.ogv Binary files differnew file mode 100644 index 0000000000..94b4750865 --- /dev/null +++ b/dom/media/test/bug504843.ogv diff --git a/dom/media/test/bug504843.ogv^headers^ b/dom/media/test/bug504843.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug504843.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug506094.ogv b/dom/media/test/bug506094.ogv Binary files differnew file mode 100644 index 0000000000..142b7b9ad1 --- /dev/null +++ b/dom/media/test/bug506094.ogv diff --git a/dom/media/test/bug506094.ogv^headers^ b/dom/media/test/bug506094.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug506094.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug516323.indexed.ogv b/dom/media/test/bug516323.indexed.ogv Binary files differnew file mode 100644 index 0000000000..7bd76eeccc --- /dev/null +++ b/dom/media/test/bug516323.indexed.ogv diff --git a/dom/media/test/bug516323.indexed.ogv^headers^ b/dom/media/test/bug516323.indexed.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug516323.indexed.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug516323.ogv b/dom/media/test/bug516323.ogv Binary files differnew file mode 100644 index 0000000000..8f2f38b983 --- /dev/null +++ b/dom/media/test/bug516323.ogv diff --git a/dom/media/test/bug516323.ogv^headers^ b/dom/media/test/bug516323.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug516323.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug520493.ogg b/dom/media/test/bug520493.ogg Binary files differnew file mode 100644 index 0000000000..6eb23198f4 --- /dev/null +++ b/dom/media/test/bug520493.ogg diff --git a/dom/media/test/bug520493.ogg^headers^ b/dom/media/test/bug520493.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug520493.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug520500.ogg b/dom/media/test/bug520500.ogg Binary files differnew file mode 100644 index 0000000000..b91d3dd97d --- /dev/null +++ b/dom/media/test/bug520500.ogg diff --git a/dom/media/test/bug520500.ogg^headers^ b/dom/media/test/bug520500.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug520500.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug520908.ogv b/dom/media/test/bug520908.ogv Binary files differnew file mode 100644 index 0000000000..093158432a --- /dev/null +++ b/dom/media/test/bug520908.ogv diff --git a/dom/media/test/bug520908.ogv^headers^ b/dom/media/test/bug520908.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug520908.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug523816.ogv b/dom/media/test/bug523816.ogv Binary files differnew file mode 100644 index 0000000000..ca9a31b6da --- /dev/null +++ b/dom/media/test/bug523816.ogv diff --git a/dom/media/test/bug523816.ogv^headers^ b/dom/media/test/bug523816.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug523816.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug533822.ogg b/dom/media/test/bug533822.ogg Binary files differnew file mode 100644 index 0000000000..a8e506910e --- /dev/null +++ b/dom/media/test/bug533822.ogg diff --git a/dom/media/test/bug533822.ogg^headers^ b/dom/media/test/bug533822.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug533822.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug556821.ogv b/dom/media/test/bug556821.ogv Binary files differnew file mode 100644 index 0000000000..8d76fee45e --- /dev/null +++ b/dom/media/test/bug556821.ogv diff --git a/dom/media/test/bug556821.ogv^headers^ b/dom/media/test/bug556821.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug556821.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug557094.ogv b/dom/media/test/bug557094.ogv Binary files differnew file mode 100644 index 0000000000..b4fc0799a6 --- /dev/null +++ b/dom/media/test/bug557094.ogv diff --git a/dom/media/test/bug557094.ogv^headers^ b/dom/media/test/bug557094.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug557094.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug603918.webm b/dom/media/test/bug603918.webm Binary files differnew file mode 100644 index 0000000000..c430e97f40 --- /dev/null +++ b/dom/media/test/bug603918.webm diff --git a/dom/media/test/bug603918.webm^headers^ b/dom/media/test/bug603918.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug603918.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bug604067.webm b/dom/media/test/bug604067.webm Binary files differnew file mode 100644 index 0000000000..86bdfdc91f --- /dev/null +++ b/dom/media/test/bug604067.webm diff --git a/dom/media/test/bug604067.webm^headers^ b/dom/media/test/bug604067.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/bug604067.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/bunny.webm b/dom/media/test/bunny.webm Binary files differnew file mode 100644 index 0000000000..823439d1c5 --- /dev/null +++ b/dom/media/test/bunny.webm diff --git a/dom/media/test/can_play_type_dash.js b/dom/media/test/can_play_type_dash.js new file mode 100644 index 0000000000..b4760545db --- /dev/null +++ b/dom/media/test/can_play_type_dash.js @@ -0,0 +1,27 @@ +function check_dash(v, enabled) { + function check(type, expected) { + is(v.canPlayType(type), enabled ? expected : "", type); + } + + // DASH types + check("application/dash+xml", "probably"); + + // Supported Webm codecs + check("application/dash+xml; codecs=vorbis", "probably"); + check("application/dash+xml; codecs=vorbis", "probably"); + check("application/dash+xml; codecs=vorbis,vp8", "probably"); + check("application/dash+xml; codecs=vorbis,vp8.0", "probably"); + check('application/dash+xml; codecs="vorbis,vp8"', "probably"); + check('application/dash+xml; codecs="vorbis,vp8.0"', "probably"); + check('application/dash+xml; codecs="vp8, vorbis"', "probably"); + check('application/dash+xml; codecs="vp8.0, vorbis"', "probably"); + check("application/dash+xml; codecs=vp8", "probably"); + check("application/dash+xml; codecs=vp8.0", "probably"); + + // Unsupported codecs + check("application/dash+xml; codecs=xyz", ""); + check("application/dash+xml; codecs=xyz,vorbis", ""); + check("application/dash+xml; codecs=vorbis,xyz", ""); + check("application/dash+xml; codecs=xyz,vp8.0", ""); + check("application/dash+xml; codecs=vp8.0,xyz", ""); +} diff --git a/dom/media/test/can_play_type_ogg.js b/dom/media/test/can_play_type_ogg.js new file mode 100644 index 0000000000..c03f8b2d3e --- /dev/null +++ b/dom/media/test/can_play_type_ogg.js @@ -0,0 +1,72 @@ +function check_ogg(v, enabled, finish) { + function check(type, expected) { + is(v.canPlayType(type), enabled ? expected : "", type); + } + + function basic_test() { + return new Promise(function(resolve, reject) { + // Ogg types + check("video/ogg", "maybe"); + check("audio/ogg", "maybe"); + check("application/ogg", "maybe"); + + // Supported Ogg codecs + check("audio/ogg; codecs=vorbis", "probably"); + check("video/ogg; codecs=vorbis", "probably"); + check("video/ogg; codecs=vorbis,theora", "probably"); + check('video/ogg; codecs="vorbis, theora"', "probably"); + check("video/ogg; codecs=theora", "probably"); + + resolve(); + }); + } + + // Verify Opus support + function verify_opus_support() { + return new Promise(function(resolve, reject) { + var OpusEnabled = SpecialPowers.getBoolPref( + "media.opus.enabled", + undefined + ); + if (OpusEnabled != undefined) { + resolve(); + } else { + console.log( + "media.opus.enabled pref not found; skipping Opus validation" + ); + reject(); + } + }); + } + + function opus_enable() { + return SpecialPowers.pushPrefEnv({ + set: [["media.opus.enabled", true]], + }).then(function() { + check("audio/ogg; codecs=opus", "probably"); + }); + } + + function opus_disable() { + return SpecialPowers.pushPrefEnv({ + set: [["media.opus.enabled", false]], + }).then(function() { + check("audio/ogg; codecs=opus", ""); + }); + } + + function unspported_ogg() { + // Unsupported Ogg codecs + check("video/ogg; codecs=xyz", ""); + check("video/ogg; codecs=xyz,vorbis", ""); + check("video/ogg; codecs=vorbis,xyz", ""); + + finish.call(); + } + + basic_test() + .then(verify_opus_support) + .then(opus_enable) + .then(opus_disable) + .then(unspported_ogg, unspported_ogg); +} diff --git a/dom/media/test/can_play_type_wave.js b/dom/media/test/can_play_type_wave.js new file mode 100644 index 0000000000..a5e087aa40 --- /dev/null +++ b/dom/media/test/can_play_type_wave.js @@ -0,0 +1,30 @@ +function check_wave(v, enabled) { + function check(type, expected) { + is(v.canPlayType(type), enabled ? expected : "", type); + } + + // Wave types + check("audio/wave", "maybe"); + check("audio/wav", "maybe"); + check("audio/x-wav", "maybe"); + check("audio/x-pn-wav", "maybe"); + + // Supported Wave codecs + check("audio/wave; codecs=1", "probably"); + check("audio/wave; codecs=3", "probably"); + check("audio/wave; codecs=6", "probably"); + check("audio/wave; codecs=7", "probably"); + // "no codecs" should be supported, I guess + check("audio/wave; codecs=", "maybe"); + check('audio/wave; codecs=""', "maybe"); + + // Unsupported Wave codecs + check("audio/wave; codecs=0", ""); + check("audio/wave; codecs=2", ""); + check("audio/wave; codecs=xyz,1", ""); + check("audio/wave; codecs=1,xyz", ""); + check('audio/wave; codecs="xyz, 1"', ""); + // empty codec names + check("audio/wave; codecs=,", ""); + check('audio/wave; codecs="0, 1,"', ""); +} diff --git a/dom/media/test/can_play_type_webm.js b/dom/media/test/can_play_type_webm.js new file mode 100644 index 0000000000..ee57da66c6 --- /dev/null +++ b/dom/media/test/can_play_type_webm.js @@ -0,0 +1,39 @@ +async function check_webm(v, enabled) { + function check(type, expected) { + is( + v.canPlayType(type), + enabled ? expected : "", + type + "='" + expected + "'" + ); + } + + // WebM types + check("video/webm", "maybe"); + check("audio/webm", "maybe"); + + var video = ["vp8", "vp8.0", "vp9", "vp9.0"]; + var audio = ["vorbis", "opus"]; + + audio.forEach(function(acodec) { + check("audio/webm; codecs=" + acodec, "probably"); + check("video/webm; codecs=" + acodec, "probably"); + }); + video.forEach(function(vcodec) { + check("video/webm; codecs=" + vcodec, "probably"); + audio.forEach(function(acodec) { + check('video/webm; codecs="' + vcodec + ", " + acodec + '"', "probably"); + check('video/webm; codecs="' + acodec + ", " + vcodec + '"', "probably"); + }); + }); + + // Unsupported WebM codecs + check("video/webm; codecs=xyz", ""); + check("video/webm; codecs=xyz,vorbis", ""); + check("video/webm; codecs=vorbis,xyz", ""); + + await SpecialPowers.pushPrefEnv({ set: [["media.av1.enabled", true]] }); + check('video/webm; codecs="av1"', "probably"); + + await SpecialPowers.pushPrefEnv({ set: [["media.av1.enabled", false]] }); + check('video/webm; codecs="av1"', ""); +} diff --git a/dom/media/test/cancellable_request.sjs b/dom/media/test/cancellable_request.sjs new file mode 100644 index 0000000000..9a4a12076e --- /dev/null +++ b/dom/media/test/cancellable_request.sjs @@ -0,0 +1,153 @@ +function parseQuery(request, key) { + var params = request.queryString.split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +function push32BE(array, input) { + array.push(String.fromCharCode((input >> 24) & 0xff)); + array.push(String.fromCharCode((input >> 16) & 0xff)); + array.push(String.fromCharCode((input >> 8) & 0xff)); + array.push(String.fromCharCode((input) & 0xff)); +} + +function push32LE(array, input) { + array.push(String.fromCharCode((input) & 0xff)); + array.push(String.fromCharCode((input >> 8) & 0xff)); + array.push(String.fromCharCode((input >> 16) & 0xff)); + array.push(String.fromCharCode((input >> 24) & 0xff)); +} + +function push16LE(array, input) { + array.push(String.fromCharCode((input) & 0xff)); + array.push(String.fromCharCode((input >> 8) & 0xff)); +} + +function buildWave(samples, sample_rate) { + const RIFF_MAGIC = 0x52494646; + const WAVE_MAGIC = 0x57415645; + const FRMT_MAGIC = 0x666d7420; + const DATA_MAGIC = 0x64617461; + const RIFF_SIZE = 44; + + var header = []; + push32BE(header, RIFF_MAGIC); + push32LE(header, RIFF_SIZE + samples.length * 2); + push32BE(header, WAVE_MAGIC); + push32BE(header, FRMT_MAGIC); + push32LE(header, 16); + push16LE(header, 1); + push16LE(header, 1); + push32LE(header, sample_rate); + push32LE(header, sample_rate); + push16LE(header, 2); + push16LE(header, 16); + push32BE(header, DATA_MAGIC); + push32LE(header, samples.length * 2); + for (var i = 0; i < samples.length; ++i) { + push16LE(header, samples[i], 2); + } + return header; +} + +const CC = Components.Constructor; +const Timer = CC("@mozilla.org/timer;1", "nsITimer", "initWithCallback"); +const BinaryOutputStream = CC("@mozilla.org/binaryoutputstream;1", + "nsIBinaryOutputStream", + "setOutputStream"); + +function poll(f) { + if (f()) { + return; + } + new Timer(function() { poll(f); }, 100, Ci.nsITimer.TYPE_ONE_SHOT); +} + +function handleRequest(request, response) +{ + var cancel = parseQuery(request, "cancelkey"); + if (cancel) { + setState(cancel[1], "cancelled"); + response.setStatusLine(request.httpVersion, 200, "OK"); + response.write("Cancel approved!"); + return; + } + + var samples = []; + for (var i = 0; i < 1000000; ++i) { + samples.push(0); + } + var bytes = buildWave(samples, 44100).join(""); + + var key = parseQuery(request, "key"); + response.setHeader("Content-Type", "audio/x-wav"); + response.setHeader("Content-Length", ""+bytes.length, false); + + var out = new BinaryOutputStream(response.bodyOutputStream); + + var start = 0, end = bytes.length - 1; + if (request.hasHeader("Range")) + { + var rangeMatch = request.getHeader("Range").match(/^bytes=(\d+)?-(\d+)?$/); + + if (rangeMatch[1] !== undefined) + start = parseInt(rangeMatch[1], 10); + + if (rangeMatch[2] !== undefined) + end = parseInt(rangeMatch[2], 10); + + // No start given, so the end is really the count of bytes from the + // end of the file. + if (start === undefined) + { + start = Math.max(0, bytes.length - end); + end = bytes.length - 1; + } + + // start and end are inclusive + if (end === undefined || end >= bytes.length) + end = bytes.length - 1; + + if (end < start) + { + response.setStatusLine(request.httpVersion, 200, "OK"); + start = 0; + end = bytes.length - 1; + } + else + { + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + var contentRange = "bytes " + start + "-" + end + "/" + bytes.length; + response.setHeader("Content-Range", contentRange); + } + } + + if (start > 0) { + // Send all requested data + out.write(bytes.slice(start, end + 1), end + 1 - start); + return; + } + + // Write the first 1.2M of the Wave file. We know the cache size is set to + // 100K so this will fill the cache and and cause a "suspend" event on + // the loading element. + out.write(bytes, 1200000); + + response.processAsync(); + // Now wait for the message to cancel this response + poll(function() { + if (getState(key[1]) != "cancelled") { + return false; + } + response.finish(); + return true; + }); +} diff --git a/dom/media/test/chain.ogg b/dom/media/test/chain.ogg Binary files differnew file mode 100644 index 0000000000..3535b280f4 --- /dev/null +++ b/dom/media/test/chain.ogg diff --git a/dom/media/test/chain.ogg^headers^ b/dom/media/test/chain.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/chain.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/chain.ogv b/dom/media/test/chain.ogv Binary files differnew file mode 100644 index 0000000000..3e684b64a5 --- /dev/null +++ b/dom/media/test/chain.ogv diff --git a/dom/media/test/chain.ogv^headers^ b/dom/media/test/chain.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/chain.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/chain.opus b/dom/media/test/chain.opus Binary files differnew file mode 100644 index 0000000000..9fa67f94c3 --- /dev/null +++ b/dom/media/test/chain.opus diff --git a/dom/media/test/chain.opus^headers^ b/dom/media/test/chain.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/chain.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/chained-audio-video.ogg b/dom/media/test/chained-audio-video.ogg Binary files differnew file mode 100644 index 0000000000..adda68bb47 --- /dev/null +++ b/dom/media/test/chained-audio-video.ogg diff --git a/dom/media/test/chained-audio-video.ogg^headers^ b/dom/media/test/chained-audio-video.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/chained-audio-video.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/chained-video.ogv b/dom/media/test/chained-video.ogv Binary files differnew file mode 100644 index 0000000000..a6288ef6c9 --- /dev/null +++ b/dom/media/test/chained-video.ogv diff --git a/dom/media/test/chained-video.ogv^headers^ b/dom/media/test/chained-video.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/chained-video.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/chrome/chrome.ini b/dom/media/test/chrome/chrome.ini new file mode 100644 index 0000000000..429c3a093c --- /dev/null +++ b/dom/media/test/chrome/chrome.ini @@ -0,0 +1,6 @@ +[DEFAULT] +subsuite = media +support-files = + ../gizmo.mp4 + +[test_accumulated_play_time.html] diff --git a/dom/media/test/chrome/test_accumulated_play_time.html b/dom/media/test/chrome/test_accumulated_play_time.html new file mode 100644 index 0000000000..509ebd5a07 --- /dev/null +++ b/dom/media/test/chrome/test_accumulated_play_time.html @@ -0,0 +1,355 @@ +<!DOCTYPE HTML> +<html> +<head> +<title>Test Video Play Time Related Permenant Telemetry Probes</title> +<script src="chrome://mochikit/content/tests/SimpleTest/SimpleTest.js"></script> +<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +<script type="application/javascript"> + +/** + * This test is used to ensure that we accumulate time for video playback + * correctly, and the results would be used in Telemetry probes. + * Currently this test covers following probes + * - VIDEO_PLAY_TIME_MS + * - VIDEO_HIDDEN_PLAY_TIME_MS + * - VIDEO_HIDDEN_PLAY_TIME_PERCENTAGE + * - VIDEO_INFERRED_DECODE_SUSPEND_PERCENTAGE + */ +const histNames = ["VIDEO_PLAY_TIME_MS", "VIDEO_HIDDEN_PLAY_TIME_MS"]; +const keyedHistNames = ["VIDEO_HIDDEN_PLAY_TIME_PERCENTAGE", "VIDEO_INFERRED_DECODE_SUSPEND_PERCENTAGE"]; + +add_task(async function setTestPref() { + await SpecialPowers.pushPrefEnv({ + set: [["media.testing-only-events", true], + ["media.test.video-suspend", true], + ["media.suspend-bkgnd-video.enabled", true], + ["media.suspend-bkgnd-video.delay-ms", 0]]}); +}); + +add_task(async function testTotalPlayTime() { + const video = document.createElement('video'); + video.src = "gizmo.mp4"; + document.body.appendChild(video); + + info(`all accumulated time should be zero`); + const videoChrome = SpecialPowers.wrap(video); + await new Promise(r => video.onloadeddata = r); + assertValueEqualTo(videoChrome, "totalPlayTime", 0); + assertValueEqualTo(videoChrome, "invisiblePlayTime", 0); + assertValueEqualTo(videoChrome, "videoDecodeSuspendedTime", 0); + + info(`start accumulating play time after media starts`); + video.autoplay = true; + await Promise.all([ + once(video, "playing"), + once(video, "moztotalplaytimestarted"), + ]); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime"); + + info(`should not accumulate time for paused video`); + video.pause(); + await once(video, "moztotalplaytimepaused"); + assertValueKeptUnchanged(videoChrome, "totalPlayTime"); + + info(`should start accumulating time again`); + let rv = await Promise.all([ + onceWithTrueReturn(video, "moztotalplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started again"); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + await cleanUpMediaAndCheckTelemetry(video); +}); + +add_task(async function testHiddenPlayTime() { + const invisibleReasons = ["notInTree", "notInConnectedTree", "invisibleInDisplay"]; + for (let reason of invisibleReasons) { + const video = document.createElement('video'); + video.src = "gizmo.mp4"; + video.loop = true; + info(`invisible video due to '${reason}'`); + + if (reason == "notInConnectedTree") { + let disconnected = document.createElement("div") + disconnected.appendChild(video); + } else if (reason == "invisibleInDisplay") { + document.body.appendChild(video); + video.style.display = "none"; + } else if (reason == "notInTree") { + // video is already created in the `notInTree` situation. + } else { + ok(false, "undefined reason"); + } + + info(`start invisible video should start accumulating timers`); + const videoChrome = SpecialPowers.wrap(video); + let rv = await Promise.all([ + onceWithTrueReturn(video, "mozinvisibleplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started playing"); + assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime"); + + info(`should not accumulate time for paused video`); + video.pause(); + await once(video, "mozinvisibleplaytimepaused"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + + info(`should start accumulating time again`); + rv = await Promise.all([ + onceWithTrueReturn(video, "mozinvisibleplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started again"); + assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime"); + + info(`make video visible should stop accumulating invisible related time`); + if (reason == "notInTree" || reason == "notInConnectedTree") { + document.body.appendChild(video); + } else if (reason == "invisibleInDisplay") { + // If we set only `display` the video would still be hidden, so setting + // width and height to make it visible. + video.style.display = "unset"; + video.style.width = "300px"; + video.style.height = "150px"; + } else { + ok(false, "undefined reason"); + } + await once(video, "mozinvisibleplaytimepaused"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + await cleanUpMediaAndCheckTelemetry(video); + } +}); + +// Note that video suspended time is not always align with the invisible play +// time even if `media.suspend-bkgnd-video.delay-ms` is `0`, because not all +// invisible videos would be suspended under current strategy. +add_task(async function testDecodeSuspendedTime() { + const video = document.createElement('video'); + video.src = "gizmo.mp4"; + document.body.appendChild(video); + + info(`start video should start accumulating timers`); + const videoChrome = SpecialPowers.wrap(video); + let rv = await Promise.all([ + onceWithTrueReturn(video, "moztotalplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started playing"); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime"); + + info(`make it invisible and force to suspend decoding`); + video.setVisible(false); + await once(video, "mozvideodecodesuspendedstarted"); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime"); + assertValueConstantlyIncreases(videoChrome, "videoDecodeSuspendedTime"); + + info(`make it visible and resume decoding`); + video.setVisible(true); + await Promise.all([ + once(video, "mozinvisibleplaytimepaused"), + once(video, "mozvideodecodesuspendedpaused"), + ]); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime"); + await cleanUpMediaAndCheckTelemetry(video); +}); + +add_task(async function reuseSameElementForPlayback() { + const video = document.createElement('video'); + video.src = "gizmo.mp4"; + document.body.appendChild(video); + + info(`start accumulating play time after media starts`); + const videoChrome = SpecialPowers.wrap(video); + let rv = await Promise.all([ + onceWithTrueReturn(video, "moztotalplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started again"); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + + info(`reset its src and all accumulated value should be reset after then`); + // After setting its src to nothing, that would trigger a failed load and set + // the error. If the following step tries to set the new resource and `play()` + // , then they should be done after receving the `error` from that failed load + // first. + await Promise.all([ + once(video, "error"), + cleanUpMediaAndCheckTelemetry(video), + ]); + // video doesn't have a decoder, so the return value would be -1 (error). + assertValueEqualTo(videoChrome, "totalPlayTime", -1); + assertValueEqualTo(videoChrome, "invisiblePlayTime", -1); + + info(`resue same element, make it visible and start playback again`); + video.src = "gizmo.mp4"; + rv = await Promise.all([ + onceWithTrueReturn(video, "moztotalplaytimestarted"), + video.play().then(_ => true, _ => false), + ]); + ok(returnTrueWhenAllValuesAreTrue(rv), "video started"); + assertValueConstantlyIncreases(videoChrome, "totalPlayTime"); + await cleanUpMediaAndCheckTelemetry(video); +}); + +add_task(async function testNoReportedTelemetryResult() { + info(`No result for empty video`); + const video = document.createElement('video'); + assertAllProbeRelatedAttributesKeptUnchanged(video); + await assertNoReportedTelemetryResult(video); + + info(`No result for video which hasn't started playing`); + video.src = "gizmo.mp4"; + document.body.appendChild(video); + ok(await once(video, "loadeddata").then(_ => true), "video loaded data"); + assertAllProbeRelatedAttributesKeptUnchanged(video); + await assertNoReportedTelemetryResult(video); + + info(`No result for video with error`); + video.src = "filedoesnotexist.mp4"; + ok(await video.play().then(_ => false, _ => true), "video failed to play"); + ok(video.error != undefined, "video got error"); + assertAllProbeRelatedAttributesKeptUnchanged(video); + await assertNoReportedTelemetryResult(video); + + info(`No result for playing an audio`); + const audio = document.createElement('audio'); + audio.src = "gizmo.mp4"; + document.body.appendChild(audio); + ok(await audio.play().then(_ => true, _ => false), "audio started playing"); + audio.pause(); + await assertNoReportedTelemetryResult(audio); +}); + +/** + * Following are helper functions + */ +async function cleanUpMediaAndCheckTelemetry(media, { shouldReport = true } = {}) { + media.src = ""; + await checkReportedTelemetry(media, shouldReport); +} + +async function assertNoReportedTelemetryResult(media) { + await checkReportedTelemetry(media, false); +} + +async function checkReportedTelemetry(media, shouldReport) { + const reportResultPromise = once(media, "mozreportedtelemetry"); + info(`check telemetry result, shouldReport=${shouldReport}`); + if (shouldReport) { + await reportResultPromise; + } + for (const name of histNames) { + try { + const hist = SpecialPowers.Services.telemetry.getHistogramById(name); + /** + * Histogram's snapshot looks like that + * { + * "bucket_count": X, + * "histogram_type": Y, + * "sum": Z, + * "range": [min, max], + * "values": { "value1" : "num1", "value2" : "num2", ...} + * } + */ + const entriesNums = Object.entries(hist.snapshot().values).length; + if (shouldReport) { + ok(entriesNums > 0, `Reported result for ${name}`); + } else { + ok(entriesNums == 0, `Reported nothing for ${name}`); + } + hist.clear(); + } catch (e) { + ok(false , `histogram '${name}' doesn't exist`); + } + } + for (const name of keyedHistNames) { + try { + const hist = SpecialPowers.Services.telemetry.getKeyedHistogramById(name); + /** + * Keyed Histogram's snapshot looks like that + * { + * "Key1" : { + * "bucket_count": X, + * "histogram_type": Y, + * "sum": Z, + * "range": [min, max], + * "values": { "value1" : "num1", "value2" : "num2", ...} + * }, + * "Key2" : {...}, + * } + */ + const items = Object.entries(hist.snapshot()); + if (items.length > 0) { + for (const [key, value] of items) { + const entriesNums = Object.entries(value.values).length; + ok(shouldReport && entriesNums > 0, `Reported ${key} for ${name}`); + } + } else { + ok(!shouldReport, `Reported nothing for ${name}`); + } + // Avoid to pollute next test task. + hist.clear(); + } catch (e) { + ok(false , `keyed histogram '${name}' doesn't exist`); + } + } +} + +function once(target, name) { + return new Promise(r => target.addEventListener(name, r, { once: true })); +} + +function onceWithTrueReturn(target, name) { + return once(target, name).then(_ => true); +} + +function returnTrueWhenAllValuesAreTrue(arr) { + for (let val of arr) { + if (!val) { + return false; + } + } + return true; +} + +function assertAttributeDefined(mediaChrome, checkType) { + ok(mediaChrome[checkType] != undefined, `${checkType} exists`); +} + +function assertValueEqualTo(mediaChrome, checkType, expectedValue) { + assertAttributeDefined(mediaChrome, checkType); + is(mediaChrome[checkType], expectedValue, `${checkType} equals to ${expectedValue}`); +} + +function assertValueConstantlyIncreases(mediaChrome, checkType) { + assertAttributeDefined(mediaChrome, checkType); + const valueSnapshot = mediaChrome[checkType]; + ok(mediaChrome[checkType] > valueSnapshot, `${checkType} keeps increasing`); +} + +function assertValueKeptUnchanged(mediaChrome, checkType) { + assertAttributeDefined(mediaChrome, checkType); + const valueSnapshot = mediaChrome[checkType]; + ok(mediaChrome[checkType] == valueSnapshot, `${checkType} keeps unchanged`); +} + +function assertAllProbeRelatedAttributesKeptUnchanged(video) { + const videoChrome = SpecialPowers.wrap(video); + assertValueKeptUnchanged(videoChrome, "totalPlayTime"); + assertValueKeptUnchanged(videoChrome, "invisiblePlayTime"); + assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime"); +} + +</script> +</head> +<body> +</body> +</html> diff --git a/dom/media/test/chromeHelper.js b/dom/media/test/chromeHelper.js new file mode 100644 index 0000000000..ff83660fe2 --- /dev/null +++ b/dom/media/test/chromeHelper.js @@ -0,0 +1,23 @@ +/* -*- Mode: javascript; indent-tabs-mode: nil; js-indent-level: 2 -*- */ +/* vim: set ft=javascript ts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +/* eslint-env mozilla/frame-script */ + +"use strict"; + +// eslint-disable-next-line mozilla/use-services +const dirSvc = Cc["@mozilla.org/file/directory_service;1"].getService( + Ci.nsIProperties +); + +addMessageListener("media-test:getcwd", () => { + let cwd; + try { + cwd = dirSvc.get("CurWorkD", Ci.nsIFile).path; + } finally { + sendAsyncMessage("media-test:cwd", cwd); + } +}); diff --git a/dom/media/test/cloneElementVisually_helpers.js b/dom/media/test/cloneElementVisually_helpers.js new file mode 100644 index 0000000000..ab81aad9ed --- /dev/null +++ b/dom/media/test/cloneElementVisually_helpers.js @@ -0,0 +1,232 @@ +const TEST_VIDEO_1 = + "http://mochi.test:8888/tests/dom/media/test/bipbop_225w_175kbps.mp4"; +const TEST_VIDEO_2 = + "http://mochi.test:8888/tests/dom/media/test/pixel_aspect_ratio.mp4"; +const LONG_VIDEO = "http://mochi.test:8888/tests/dom/media/test/gizmo.mp4"; + +/** + * Ensure that the original <video> is prepped and ready to play + * before starting any other tests. + */ +async function setup() { + await SpecialPowers.pushPrefEnv({ + set: [ + ["media.test.video-suspend", true], + ["media.suspend-bkgnd-video.enabled", true], + ["media.suspend-bkgnd-video.delay-ms", 500], + ["media.dormant-on-pause-timeout-ms", 0], + ["media.cloneElementVisually.testing", true], + ], + }); + + let originalVideo = document.getElementById("original"); + await setVideoSrc(originalVideo, TEST_VIDEO_1); +} + +/** + * Given a canvas, as well as a width and height of something to be + * blitted onto that canvas, makes any necessary adjustments to the + * canvas to work with the current display resolution. + * + * @param canvas (<canvas> element) + * The canvas to be adjusted. + * @param width (int) + * The width of the image to be blitted. + * @param height (int) + * The height of the image to be blitted. + * @return CanvasRenderingContext2D (SpecialPowers wrapper) + */ +function getWrappedScaledCanvasContext(canvas, width, height) { + let devRatio = window.devicePixelRatio || 1; + let ctx = SpecialPowers.wrap(canvas.getContext("2d")); + let backingRatio = ctx.webkitBackingStorePixelRatio || 1; + + let ratio = 1; + ratio = devRatio / backingRatio; + canvas.width = ratio * width; + canvas.height = ratio * height; + canvas.style.width = width + "px"; + canvas.style.height = height + "px"; + ctx.scale(ratio, ratio); + + return ctx; +} + +/** + * Given a <video> element in the DOM, figures out its location, and captures + * a snapshot of what it's currently presenting. + * + * @param video (<video> element) + * @return ImageData (SpecialPowers wrapper) + */ +function captureFrameImageData(video) { + // Flush layout first, just in case the decoder has recently + // updated the dimensions of the video. + let rect = video.getBoundingClientRect(); + + let width = video.videoWidth; + let height = video.videoHeight; + + let canvas = document.createElement("canvas"); + let ctx = getWrappedScaledCanvasContext(canvas, width, height); + ctx.drawWindow(window, rect.left, rect.top, width, height, "rgb(0,0,0)"); + + return ctx.getImageData(0, 0, width, height); +} + +/** + * Given two <video> elements, captures snapshots of what they're currently + * displaying, and asserts that they're identical. + * + * @param video1 (<video> element) + * A video element to compare against. + * @param video2 (<video> element) + * A video to compare with video1. + * @return Promise + * @resolves + * Resolves as true if the two videos match. + */ +async function assertVideosMatch(video1, video2) { + let video1Frame = captureFrameImageData(video1); + let video2Frame = captureFrameImageData(video2); + + let left = document.getElementById("left"); + let leftCtx = getWrappedScaledCanvasContext( + left, + video1Frame.width, + video1Frame.height + ); + leftCtx.putImageData(video1Frame, 0, 0); + + let right = document.getElementById("right"); + let rightCtx = getWrappedScaledCanvasContext( + right, + video2Frame.width, + video2Frame.height + ); + rightCtx.putImageData(video2Frame, 0, 0); + + if (video1Frame.data.length != video2Frame.data.length) { + return false; + } + + let leftDataURL = left.toDataURL(); + let rightDataURL = right.toDataURL(); + + if (leftDataURL != rightDataURL) { + dump("Left frame: " + leftDataURL + "\n\n"); + dump("Right frame: " + rightDataURL + "\n\n"); + + return false; + } + + return true; +} + +/** + * Testing helper function that constructs a node clone of a video, + * injects it into the DOM, and then runs an async testing function. + * This also does the work of removing the clone before resolving. + * + * @param video (<video> element) + * The video to clone the node from. + * @param asyncFn (async function) + * A test function that will be passed the new clone as its + * only argument. + * @return Promise + * @resolves + * When the asyncFn resolves and the clone has been removed + * from the DOM. + */ +async function withNewClone(video, asyncFn) { + let clone = video.cloneNode(); + clone.id = "clone"; + clone.src = ""; + let content = document.getElementById("content"); + content.appendChild(clone); + + try { + await asyncFn(clone); + } finally { + clone.remove(); + } +} + +/** + * Sets the src on a video and waits until its ready. + * + * @param video (<video> element) + * The video to set the src on. + * @param src (string) + * The URL to set as the source on a video. + * @return Promise + * @resolves + * When the video fires the "canplay" event. + */ +async function setVideoSrc(video, src) { + let promiseReady = waitForEventOnce(video, "canplay"); + video.src = src; + await promiseReady; +} + +/** + * Returns a Promise once target emits a particular event + * once. + * + * @param target (DOM node) + * The target to monitor for the event. + * @param event (string) + * The name of the event to wait for. + * @return Promise + * @resolves + * When the event fires, and resolves to the event. + */ +function waitForEventOnce(target, event) { + return new Promise(resolve => { + target.addEventListener(event, resolve, { once: true }); + }); +} + +/** + * Polls the video debug data as a hacky way of knowing when + * when the decoders have shut down. + * + * @param video (<video> element) + * @return Promise + * @resolves + * When the decoder has shut down. + */ +async function waitForShutdownDecoder(video) { + await SimpleTest.promiseWaitForCondition(async () => { + let readerData = SpecialPowers.wrap(video).mozDebugReaderData; + return readerData.includes(": shutdown"); + }, "Video decoder should eventually shut down."); +} + +/** + * Ensures that both hiding and pausing the video causes the + * video to suspend and make dormant its decoders, respectively. + * + * @param video (<video element) + */ +async function ensureVideoSuspendable(video) { + video = SpecialPowers.wrap(video); + + ok(!video.hasSuspendTaint(), "Should be suspendable"); + + // First, we'll simulate putting the video in the background by + // making it invisible. + let suspendPromise = waitForEventOnce(video, "mozentervideosuspend"); + video.setVisible(false); + await suspendPromise; + ok(true, "Suspended after the video was made invisible."); + video.setVisible(true); + + ok(!video.hasSuspendTaint(), "Should still be suspendable."); + + // Next, we'll pause the video. + await video.pause(); + await waitForShutdownDecoder(video); + ok(true, "Shutdown decoder after the video was paused."); + await video.play(); +} diff --git a/dom/media/test/contentType.sjs b/dom/media/test/contentType.sjs new file mode 100644 index 0000000000..a7a0caf669 --- /dev/null +++ b/dom/media/test/contentType.sjs @@ -0,0 +1,77 @@ +// Parse the query string, and give us the value for a certain key, or false if +// it does not exist. +function parseQuery(request, key) { + var params = request.queryString.split('?')[0].split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +function handleRequest(request, response) { + try { + // Get the filename to send back. + var filename = parseQuery(request, "file"); + + const CC = Components.Constructor; + const BinaryOutputStream = CC("@mozilla.org/binaryoutputstream;1", + "nsIBinaryOutputStream", + "setOutputStream"); + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/" + filename; + dump(paths + '\n'); + var split = paths.split("/"); + for(var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + + // handle range requests + var partialstart = 0, + partialend = file.fileSize - 1; + if (request.hasHeader("Range")) { + var range = request.getHeader("Range"); + var parts = range.replace(/bytes=/, "").split("-"); + var partialstart = parts[0]; + var partialend = parts[1]; + if (!partialend.length) { + partialend = file.fileSize - 1; + } + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + var contentRange = "bytes " + partialstart + "-" + partialend + "/" + file.fileSize; + response.setHeader("Content-Range", contentRange); + } + + fis.seek(Components.interfaces.nsISeekableStream.NS_SEEK_SET, partialstart); + bis.setInputStream(fis); + + var sendContentType = parseQuery(request, "nomime"); + if (sendContentType == false) { + var contentType = parseQuery(request, "type"); + if (contentType == false) { + // This should not happen. + dump("No type specified without having \'nomime\' in parameters."); + return; + } + response.setHeader("Content-Type", contentType, false); + } + response.setHeader("Content-Length", ""+bis.available(), false); + + var bytes = bis.readBytes(bis.available()); + response.write(bytes, bytes.length); + } catch (e) { + dump ("ERROR : " + e + "\n"); + } +} diff --git a/dom/media/test/crashtests/0-timescale.html b/dom/media/test/crashtests/0-timescale.html new file mode 100644 index 0000000000..db845096dd --- /dev/null +++ b/dom/media/test/crashtests/0-timescale.html @@ -0,0 +1,14 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +</head> +<body> +<!-- This video file has a timescale of 0 in the mdhd atom --> +<video src="0-timescale.mp4" + autoplay + onerror="document.documentElement.className=undefined" + onloadedmetadata="this.src=''; document.documentElement.className=undefined"> +<!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. --> +</video> +</body> +</html> diff --git a/dom/media/test/crashtests/0-timescale.mp4 b/dom/media/test/crashtests/0-timescale.mp4 Binary files differnew file mode 100644 index 0000000000..32df5dc5a5 --- /dev/null +++ b/dom/media/test/crashtests/0-timescale.mp4 diff --git a/dom/media/test/crashtests/1012609.html b/dom/media/test/crashtests/1012609.html new file mode 100644 index 0000000000..1dad783a07 --- /dev/null +++ b/dom/media/test/crashtests/1012609.html @@ -0,0 +1,9 @@ +<script> +try{var r0=new AudioContext();}catch(e){} +try{var r32=r0.createOscillator();}catch(e){} +try{var r58=r0.createPeriodicWave(new Float32Array(1997),new Float32Array(1997));}catch(e){} +try{r32.start(0);}catch(e){} +try{r32.setPeriodicWave(r58);}catch(e){} +try{r32.frequency.value=-1;}catch(e){} +</script> + diff --git a/dom/media/test/crashtests/1015662.html b/dom/media/test/crashtests/1015662.html new file mode 100644 index 0000000000..20407c807d --- /dev/null +++ b/dom/media/test/crashtests/1015662.html @@ -0,0 +1,4 @@ +<!DOCTYPE html> +<body> +<video><track src="javascript:5"></track></video> +</body> diff --git a/dom/media/test/crashtests/1028458.html b/dom/media/test/crashtests/1028458.html new file mode 100644 index 0000000000..bb01e3343a --- /dev/null +++ b/dom/media/test/crashtests/1028458.html @@ -0,0 +1,23 @@ +<html class="reftest-wait"> +<audio id="testAudio" controls></audio> +<script type="text/javascript"> +navigator.mozGetUserMedia({audio: true, fake: true}, function(stream) { + stream.getAudioTracks()[0].enabled = false; + var testAudio = document.getElementById('testAudio'); + // Wait some time for good measure + var eventReceived = 0; + testAudio.addEventListener("timeupdate", function() { + if (++eventReceived == 3) { + document.querySelector("html").className = ""; + } + }) + testAudio.srcObject = stream; + testAudio.play(); + }, function(err) { + // Don't go orange if we can't get an audio input stream, + // as this is not what we are trying to test and can happen on Windows. + document.querySelector("html").className = ""; + }); +</script> + +</html> diff --git a/dom/media/test/crashtests/1041466.html b/dom/media/test/crashtests/1041466.html new file mode 100644 index 0000000000..0f064d186c --- /dev/null +++ b/dom/media/test/crashtests/1041466.html @@ -0,0 +1,21 @@ +<html> +<script> +try{var Context1= new (window.webkitAudioContext || window.AudioContext)()}catch(e){} +try{var Delay0=Context1.createDelay();}catch(e){} +try{var ScriptProcessor0=Context1.createScriptProcessor(512,26,7);}catch(e){} +try{var ChannelSplitter0=Context1.createChannelSplitter(91);}catch(e){} +try{var Gain1=Context1.createGain();}catch(e){} +try{var WaveShaper0=Context1.createWaveShaper();}catch(e){} +try{var Analyser1=Context1.createAnalyser();}catch(e){} +try{Gain1.connect(Delay0);}catch(e){} +try{Analyser1.connect(BiquadFilter0);}catch(e){} +try{Gain1.connect(Context1.destination);}catch(e){} +try{WaveShaper0.connect(ScriptProcessor0);}catch(e){} +try{ChannelSplitter0.connect(BiquadFilter1);}catch(e){} +try{Delay0.connect(Gain1);}catch(e){} +try{ScriptProcessor0.connect(Context1.destination);}catch(e){} +try{WaveShaper0.connect(Gain1);}catch(e){} +try{WaveShaper0.connect(ChannelSplitter0);}catch(e){} +try{ScriptProcessor0.connect(WaveShaper0);}catch(e){} +</script> +</html> diff --git a/dom/media/test/crashtests/1045650.html b/dom/media/test/crashtests/1045650.html new file mode 100644 index 0000000000..32acd2ce75 --- /dev/null +++ b/dom/media/test/crashtests/1045650.html @@ -0,0 +1,18 @@ +<html><body><script> + +var r0=new AudioContext(); + +var splitter=r0.createChannelSplitter(); +var delay=r0.createDelay(); +var scriptp=r0.createScriptProcessor(); +var cmerger=r0.createChannelMerger(); +var gain=r0.createGain(); + +splitter.connect(delay,2); +delay.connect(scriptp); +scriptp.connect(cmerger); +cmerger.connect(splitter); +gain.connect(gain); +gain.connect(cmerger); + +</script></body></html> diff --git a/dom/media/test/crashtests/1080986.html b/dom/media/test/crashtests/1080986.html new file mode 100644 index 0000000000..1de3075169 --- /dev/null +++ b/dom/media/test/crashtests/1080986.html @@ -0,0 +1,3 @@ +<html> +<audio autoplay src="1080986.wav"></audio> +</html> diff --git a/dom/media/test/crashtests/1080986.wav b/dom/media/test/crashtests/1080986.wav Binary files differnew file mode 100644 index 0000000000..b96c59b7ec --- /dev/null +++ b/dom/media/test/crashtests/1080986.wav diff --git a/dom/media/test/crashtests/1122218.html b/dom/media/test/crashtests/1122218.html new file mode 100644 index 0000000000..984487dd21 --- /dev/null +++ b/dom/media/test/crashtests/1122218.html @@ -0,0 +1,24 @@ +<html> +<head> +<script> +function boom() { + var r0=new AudioContext(); + + var cm=r0.createChannelMerger(20); + + var o1=r0.createOscillator(); + var o2=r0.createOscillator(); + + var pw=r0.createPeriodicWave(new Float32Array(4),new Float32Array(4)); + o2.setPeriodicWave(pw); + + o1.connect(cm); + cm.connect(o2.frequency); + + o1.start(); + o2.start(0.476); +} +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/1127188.html b/dom/media/test/crashtests/1127188.html new file mode 100644 index 0000000000..650f07dd47 --- /dev/null +++ b/dom/media/test/crashtests/1127188.html @@ -0,0 +1,3 @@ +<script> + (new AudioContext).close(); +</script> diff --git a/dom/media/test/crashtests/1157994.html b/dom/media/test/crashtests/1157994.html new file mode 100644 index 0000000000..2e3001302a --- /dev/null +++ b/dom/media/test/crashtests/1157994.html @@ -0,0 +1,21 @@ +<!DOCTYPE html> +<html> +<head> +<script> + +function boom() +{ + var ac = new AudioContext(); + // first "suspended" -> "running" transition + ac.onstatechange = function() { + ac.onstatechange = null; + ac.suspend(); + ac.close(); + } +} + +</script> +</head> +<body onload="boom();"></body> +</html> + diff --git a/dom/media/test/crashtests/1158427.html b/dom/media/test/crashtests/1158427.html new file mode 100644 index 0000000000..b544a942a1 --- /dev/null +++ b/dom/media/test/crashtests/1158427.html @@ -0,0 +1,21 @@ +<!DOCTYPE html> +<html> +<head> +<script> + +function boom() +{ + dump("before capture\n"); + document.createElement("audio").mozCaptureStreamUntilEnded(); + dump("before context\n"); + new window.AudioContext(); + dump("before gc\n"); + SpecialPowers.forceCC(); + dump("after gc\n"); +} + +</script> +</head> +<body onload="boom();"></body> +</html> + diff --git a/dom/media/test/crashtests/1180881.html b/dom/media/test/crashtests/1180881.html new file mode 100644 index 0000000000..d5bf2f5642 --- /dev/null +++ b/dom/media/test/crashtests/1180881.html @@ -0,0 +1,8 @@ +<!DOCTYPE html> +<html> +<head> +</head> +<body> +<video src="1180881.webm" autoplay></video> +</body> +</html> diff --git a/dom/media/test/crashtests/1180881.webm b/dom/media/test/crashtests/1180881.webm Binary files differnew file mode 100644 index 0000000000..2fb2be7a7f --- /dev/null +++ b/dom/media/test/crashtests/1180881.webm diff --git a/dom/media/test/crashtests/1185176.html b/dom/media/test/crashtests/1185176.html new file mode 100644 index 0000000000..d5e9b68a29 --- /dev/null +++ b/dom/media/test/crashtests/1185176.html @@ -0,0 +1,24 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script> + +function boom() +{ + var ac = new window.AudioContext(); + var oscillator = ac.createOscillator(); + oscillator.start(0); + oscillator.stop(0.1); + setTimeout(function() { + oscillator.channelCount = 1; + oscillator.channelCountMode = "explicit"; + oscillator.channelInterpretation = "speakers"; + document.documentElement.removeAttribute("class"); + }, 1000); +} + +</script> +</head> +<body onload="boom();"></body> +</html> + diff --git a/dom/media/test/crashtests/1185192.html b/dom/media/test/crashtests/1185192.html new file mode 100644 index 0000000000..46fa311aa2 --- /dev/null +++ b/dom/media/test/crashtests/1185192.html @@ -0,0 +1,18 @@ +<!DOCTYPE html> +<html> +<head> +<script> + +function boom() +{ + new Audio().mozCaptureStreamUntilEnded(); + var ac = new window.AudioContext(); + ac.resume(); + ac.close(); +} + +</script> +</head> +<body onload="boom();"></body> +</html> + diff --git a/dom/media/test/crashtests/1197935.html b/dom/media/test/crashtests/1197935.html new file mode 100644 index 0000000000..dd8ad0382d --- /dev/null +++ b/dom/media/test/crashtests/1197935.html @@ -0,0 +1,8 @@ +<!DOCTYPE html> +<html> +<head> +</head> +<body> +<video src="1197935.mp4" autoplay></video> +</body> +</html> diff --git a/dom/media/test/crashtests/1197935.mp4 b/dom/media/test/crashtests/1197935.mp4 Binary files differnew file mode 100644 index 0000000000..f00de75627 --- /dev/null +++ b/dom/media/test/crashtests/1197935.mp4 diff --git a/dom/media/test/crashtests/1223670.html b/dom/media/test/crashtests/1223670.html new file mode 100644 index 0000000000..94cad43e23 --- /dev/null +++ b/dom/media/test/crashtests/1223670.html @@ -0,0 +1,23 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script> + +function boom() +{ + + var ac = new window.AudioContext("publicnotification"); + + setTimeout(function() { + document.documentElement.removeAttribute("class"); + var htmlAudio = new Audio(); + var stream = htmlAudio.mozCaptureStreamUntilEnded(); + ac.createMediaStreamSource(stream); + }, 0); +} + +</script> +</head> +<body onload="boom();"> +</body> +</html> diff --git a/dom/media/test/crashtests/1236639.html b/dom/media/test/crashtests/1236639.html new file mode 100644 index 0000000000..5c4634a4d4 --- /dev/null +++ b/dom/media/test/crashtests/1236639.html @@ -0,0 +1,9 @@ +<!DOCTYPE html> +<html> +<head> + <title>Bug 1236639: Crash on audio playback due to invalid XING headers</title> +</head> +<body> +<audio autoplay src="1236639.mp3"></audio> +</body> +</html> diff --git a/dom/media/test/crashtests/1236639.mp3 b/dom/media/test/crashtests/1236639.mp3 Binary files differnew file mode 100644 index 0000000000..8ef96e8cc7 --- /dev/null +++ b/dom/media/test/crashtests/1236639.mp3 diff --git a/dom/media/test/crashtests/1257700.html b/dom/media/test/crashtests/1257700.html new file mode 100644 index 0000000000..5377b17da5 --- /dev/null +++ b/dom/media/test/crashtests/1257700.html @@ -0,0 +1,8 @@ +<!DOCTYPE html> +<html> +<head> +</head> +<body> +<video src="1257700.webm" autoplay></video> +</body> +</html> diff --git a/dom/media/test/crashtests/1257700.webm b/dom/media/test/crashtests/1257700.webm Binary files differnew file mode 100644 index 0000000000..63e53c8c0a --- /dev/null +++ b/dom/media/test/crashtests/1257700.webm diff --git a/dom/media/test/crashtests/1267263.html b/dom/media/test/crashtests/1267263.html new file mode 100644 index 0000000000..a4d0e621cd --- /dev/null +++ b/dom/media/test/crashtests/1267263.html @@ -0,0 +1,19 @@ +<!DOCTYPE html> +<html> +<head> +<meta charset="UTF-8"> +<script> + +function boom() +{ + vid.setMediaKeys(null); + vid.fastSeek(111); +} + +</script> +</head> + +<body onload="boom();"> + <video id="vid" src="../../../../layout/reftests/webm-video/frames.webm"></video> +</body> +</html> diff --git a/dom/media/test/crashtests/1270303.html b/dom/media/test/crashtests/1270303.html new file mode 100644 index 0000000000..23608bb2b8 --- /dev/null +++ b/dom/media/test/crashtests/1270303.html @@ -0,0 +1,8 @@ +<!DOCTYPE html> +<html> +<head> +</head> +<body> +<video src="1270303.webm" autoplay></video> +</body> +</html> diff --git a/dom/media/test/crashtests/1270303.webm b/dom/media/test/crashtests/1270303.webm Binary files differnew file mode 100644 index 0000000000..c9b0003d6b --- /dev/null +++ b/dom/media/test/crashtests/1270303.webm diff --git a/dom/media/test/crashtests/1291702.html b/dom/media/test/crashtests/1291702.html new file mode 100644 index 0000000000..facc52af7b --- /dev/null +++ b/dom/media/test/crashtests/1291702.html @@ -0,0 +1,72 @@ +<script> +Logger={}; Logger.JSError=function(e){}; +try { o0 = new Audio("media/audio/mono-uncompressed-8bit-44100hz.wav") } catch(e) { Logger.JSError(e); } +try { o1 = o0.mozCaptureStreamUntilEnded() } catch(e) { Logger.JSError(e); } +try { o2 = new window.AudioContext(); } catch(e) { Logger.JSError(e); } +try { o3 = o2.createBufferSource(); } catch(e) { Logger.JSError(e); } +try { o5 = o2.createChannelMerger(3); } catch(e) { Logger.JSError(e); } +try { o3.start(0) } catch(e) { Logger.JSError(e); } +try { o2.listener.setPosition(0.0417049336344248, 0.9932504594310304, 32) } catch(e) { Logger.JSError(e); } +try { o5.disconnect(0) } catch(e) { Logger.JSError(e); } +try { o0.mozGetMetadata() } catch(e) { Logger.JSError(e); } +try { o3.channelCount = 1; } catch(e) { Logger.JSError(e); } +try { o3.buffer = function anonymous() { +var buffer = o2.createBuffer(1, 512, o2.sampleRate);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<512;i++) {data[i] = i % 512}}return buffer; +}(); } catch(e) { Logger.JSError(e); } +try { o3.loop = false; } catch(e) { Logger.JSError(e); } +try { o0.mozPreservesPitch = false; } catch(e) { Logger.JSError(e); } +try { o2.destination.channelCount = 1; } catch(e) { Logger.JSError(e); } +try { o3.loopStart = 8; } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.value = 0.46271130895770884; } catch(e) { Logger.JSError(e); } +try { o2.listener.setVelocity(0.34781960219792546, 4, 2048) } catch(e) { Logger.JSError(e); } +try { o5.connect(o5, 0, 0) } catch(e) { Logger.JSError(e); } +try { o3.loopStart = -0.24696638021780326; } catch(e) { Logger.JSError(e); } +try { o0.mozSetup(1, 44100) } catch(e) { Logger.JSError(e); } +try { o3.buffer.copyToChannel(function anonymous() { +var buffer=new Float32Array(256);for(var i=0;i<256;i++){buffer[i]=i / 256}return buffer; +}(), 1, 2048, 64) } catch(e) { Logger.JSError(e); } +try { o3.loop = false; } catch(e) { Logger.JSError(e); } +try { setInterval(function anonymous() { +try { o0.pause() } catch(e) { Logger.JSError(e); } +}, 12.902067779658143) } catch(e) { Logger.JSError(e); } +try { o2.listener.setPosition(256, 256, 16) } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.setValueCurveAtTime(function anonymous() { +var buffer=new Float32Array(4);for(var i=0;i<4;i++){buffer[i]=i / 4}return buffer; +}(), 2, 0.40792575814014437) } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.value = 0.4997270553139334; } catch(e) { Logger.JSError(e); } +try { o0.loop = true; } catch(e) { Logger.JSError(e); } +try { o3.loopStart = 4; } catch(e) { Logger.JSError(e); } +try { setInterval(function anonymous() { +try { o3.buffer = function anonymous() { +var buffer = o2.createBuffer(1, 1, o2.sampleRate);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<1;i++) {data[i] = Math.sin(Math.sin(i))}}return buffer; +}() } catch(e) { Logger.JSError(e); } +}, 54.32078602859342) } catch(e) { Logger.JSError(e); } +try { o3.connect(o2.destination); } catch(e) { Logger.JSError(e); } +try { o3.channelCountMode = 'max'; } catch(e) { Logger.JSError(e); } +try { setInterval(function anonymous() { +try { o3.channelCount = 1; } catch(e) { Logger.JSError(e); } +}, 55.448587039802966) } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.cancelScheduledValues(0.7190983131805198) } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.cancelScheduledValues(16) } catch(e) { Logger.JSError(e); } +try { o3.connect(o5, 0, 2) } catch(e) { Logger.JSError(e); } +try { o3.loopEnd = 0.5864771678080962; } catch(e) { Logger.JSError(e); } +try { o0.playbackRate = 0.2781783298771312; } catch(e) { Logger.JSError(e); } +try { o0.loop = false; } catch(e) { Logger.JSError(e); } +try { setInterval(function anonymous() { +try { o5.disconnect(0) } catch(e) { Logger.JSError(e); } +}, 29.75776777646425) } catch(e) { Logger.JSError(e); } +try { o3.playbackRate.setValueCurveAtTime(function anonymous() { +var buffer=new Float32Array(8);for(var i=0;i<8;i++){buffer[i]=8 % 8}return buffer; +}(), 0.4972710112336257, 64) } catch(e) { Logger.JSError(e); } +try { setInterval(function anonymous() { +try { o0.controls = false; } catch(e) { Logger.JSError(e); } +}, 22.550249570567694) } catch(e) { Logger.JSError(e); } +try { o2.listener.setOrientation(0.6531494410366634, 64, 0.5120918081402992, -64, 0.32912433155093446, 256) } catch(e) { Logger.JSError(e); } +try { o3.loop = true; } catch(e) { Logger.JSError(e); } +try { o3.connect(o5, 0, 0) } catch(e) { Logger.JSError(e); } +try { o3.buffer = function anonymous() { +var buffer = o2.createBuffer(1, 2048, 48000);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<2048;i++) {data[i] = Math.sin(Math.sin(2048 * 0.2519529190035427))}}return buffer; +}(); } catch(e) { Logger.JSError(e); } +try { o3.disconnect(0) } catch(e) { Logger.JSError(e); } +</script> + diff --git a/dom/media/test/crashtests/1368490.html b/dom/media/test/crashtests/1368490.html new file mode 100644 index 0000000000..8a2d9f9674 --- /dev/null +++ b/dom/media/test/crashtests/1368490.html @@ -0,0 +1,30 @@ +<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title> Bug 1368490 : Crash if media recorder source stream reduces number of channels. </title>
+</head>
+<meta charset="utf-8">
+<script type="text/javascript">
+
+function boom() {
+ let audioContext = new window.AudioContext();
+ let oscillator = audioContext.createOscillator();
+ let dst = audioContext.createMediaStreamDestination();
+ oscillator.channelCount = 4;
+ dst.channelCount = 4;
+ oscillator.connect(dst, 0, 0);
+ oscillator.start();
+ mediaRec = new MediaRecorder(dst.stream);
+
+ mediaRec.start(100);
+ setTimeout(() => {
+ dst.channelCount = 1;
+ setTimeout(() => {
+ mediaRec.stop();
+ document.documentElement.removeAttribute("class");
+ }, 100);
+ }, 100);
+}
+</script>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/1378826.html b/dom/media/test/crashtests/1378826.html new file mode 100644 index 0000000000..e1913cd0f5 --- /dev/null +++ b/dom/media/test/crashtests/1378826.html @@ -0,0 +1,46 @@ +<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<title>Bug 1378826 : Removing last video track from recorder stream crashes.</title>
+</head>
+<body>
+<canvas id="canvas"></canvas>
+<script type="text/javascript">
+
+function wait(ms) {
+ return new Promise(resolve => setTimeout(resolve, ms));
+}
+
+function boom() {
+ let canvas = document.getElementById("canvas");
+ let ctx = canvas.getContext('2d');
+ ctx.fillRect(10, 10, 100, 100);
+ let stream = canvas.captureStream();
+ let rec = new MediaRecorder(stream);
+ // At the time of fixing this bug onstop would fire, but this may change in
+ // future. As such defensively listen for onerror too to prevent this test
+ // timing out.
+ let stoppedPromise = new Promise(y => (rec.onstop = y,
+ rec.onerror = e => y));
+ rec.onstart = () => {
+ // Remove the video track from the stream we're recording
+ stream.removeTrack(stream.getTracks()[0]);
+ // Recorder should stop or error in response to the above
+ return stoppedPromise
+ .then(() => {
+ // Little wait to help get bad writes if they're going to happen
+ wait(100)
+ .then(() => {
+ // Didn't crash, finish
+ document.documentElement.removeAttribute("class");
+ });
+ });
+ };
+ rec.start();
+}
+
+window.onload = boom;
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1384248.html b/dom/media/test/crashtests/1384248.html new file mode 100644 index 0000000000..5d9c60edda --- /dev/null +++ b/dom/media/test/crashtests/1384248.html @@ -0,0 +1,10 @@ +<html> + <head> + <script> + try { o1 = document.createElement('audio') } catch(e) { } + try { o2 = document.implementation.createDocument('', '', null).adoptNode(o1); } catch(e) { }; + try { o3 = new AudioContext('alarm') } catch(e) { } + try { o3.createMediaElementSource(o1) } catch(e) { } + </script> + </head> +</html> diff --git a/dom/media/test/crashtests/1388372.html b/dom/media/test/crashtests/1388372.html new file mode 100644 index 0000000000..977ebddf53 --- /dev/null +++ b/dom/media/test/crashtests/1388372.html @@ -0,0 +1,13 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +navigator.mediaDevices.getUserMedia({audio: { + echoCancellation: false, + noiseSuppression: false, + autoGainControl: false + }, fake: true +}).then((stream) => { + document.documentElement.removeAttribute("class"); +}) +</script> +</html> diff --git a/dom/media/test/crashtests/1389304.html b/dom/media/test/crashtests/1389304.html new file mode 100644 index 0000000000..df419c51d7 --- /dev/null +++ b/dom/media/test/crashtests/1389304.html @@ -0,0 +1,32 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: Negative duration.</title> +</head> +<body> + +<video id="v" controls src="1389304.mp4"> +</video> +<p id="msg"></p> + +<script type="text/javascript"> + +function log(x) { + msg.innerHTML = x + "<br>"; +} + +v.play(); +v.onended = function() { + log("endded!"); + let seekable = v.seekable; + for (let i = 0; i < seekable.length; ++i) { + let start = seekable.start(i); + let end = seekable.end(i); + log(`[${i}]: start=${start} end=${end}`); + } +} + +</script> + +</body> +</html> diff --git a/dom/media/test/crashtests/1389304.mp4 b/dom/media/test/crashtests/1389304.mp4 Binary files differnew file mode 100644 index 0000000000..25cd746972 --- /dev/null +++ b/dom/media/test/crashtests/1389304.mp4 diff --git a/dom/media/test/crashtests/1393272.webm b/dom/media/test/crashtests/1393272.webm Binary files differnew file mode 100644 index 0000000000..1f1cade6dc --- /dev/null +++ b/dom/media/test/crashtests/1393272.webm diff --git a/dom/media/test/crashtests/1411322.html b/dom/media/test/crashtests/1411322.html new file mode 100644 index 0000000000..772b68f0cc --- /dev/null +++ b/dom/media/test/crashtests/1411322.html @@ -0,0 +1,18 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Bug 1411322: Shutdown after getting memory reports from MediaRecorder</title> + <script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<audio id="audio"></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +let recorder = new MediaRecorder(audio.mozCaptureStream()); +recorder.start(); +SpecialPowers.getMemoryReports(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/crashtests/1450845.html b/dom/media/test/crashtests/1450845.html new file mode 100644 index 0000000000..451d116e83 --- /dev/null +++ b/dom/media/test/crashtests/1450845.html @@ -0,0 +1,34 @@ +<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title>Bug 1450845: Avoid seek to next frame when already seeking</title>
+ <script>
+ async function boom() {
+ let video = document.getElementById('video');
+
+ // Internally play causes a seek, make sure we don't crash during this
+ video.play();
+ try {
+ await document.getElementById('video').seekToNextFrame();
+ } catch (e) {
+ // We don't mind if the promise was rejected so long as we don't crash
+ }
+ // Didn't crash
+
+ // Stop playback and cause a seek to 0
+ video.pause();
+ video.currentTime = 0;
+ try {
+ await document.getElementById('video').seekToNextFrame();
+ } finally {
+ // Didn't crash
+ document.documentElement.removeAttribute("class");
+ }
+ }
+ window.addEventListener('load', boom)
+ </script>
+</head>
+<body>
+ <video id='video' src='data:video/webm;base64,GkXfowEAAAAAAAAfQoaBAUL3gQFC8oEEQvOBQoWBAhhTgGcBAAAAAAAB6BFNm3RALE27i1OrhBVJqWZTrIHfTbuMU6uEFlSua1OsggEwTbuMU6uEHFO7a1OsggHL7AEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAVSalmAQAAAAAAAEUqTYCNTGF2ZjU3LjI5LjEwMVdBjUxhdmY1Ny4yOS4xMDFzpJBAb17Yv2oNAF1ZEESuco33RImIQFCAAAAAAAAWVK5rAQAAAAAAADyuAQAAAAAAADPXgQFzxYEBnIEAIrWcg3VuZIaFVl9WUDmDgQEj44OEAfygVeABAAAAAAAAB7CCAUC6gfAfQ7Z1AQAAAAAAAEfngQCjqYEAAICCSYNCABPwDvYAOCQcGFQAAFBh9jAAABML7AAATEnjdRwIJ+gAo5eBACEAhgBAkpwATEAABCasAABekcXgAB'>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1489160.html b/dom/media/test/crashtests/1489160.html new file mode 100644 index 0000000000..c4f643700c --- /dev/null +++ b/dom/media/test/crashtests/1489160.html @@ -0,0 +1,10 @@ +<html> +<head> +<script> +audio = new AudioContext() +audio.close() +audio.close() +</script> +</head> +</html> + diff --git a/dom/media/test/crashtests/1494073.html b/dom/media/test/crashtests/1494073.html new file mode 100644 index 0000000000..41e7a36554 --- /dev/null +++ b/dom/media/test/crashtests/1494073.html @@ -0,0 +1,19 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> + <title>Bug 1494073: Setting playback rate too high</title> + <script> + function finish() { + document.documentElement.removeAttribute("class"); + } + var src="data:audio/wav;base64,UklGRqsTAABXQVZFZm10IBAAAAABAAMA8FUAANABAQADAAgAZGF0YYcTAAAUCnYAPm6YALYJSgA4IO8A9UuWALiiZQAYWc4AnMNoAKcc9gBmWLcAtL0NAAGU0QDbSgMA777tAEkfrQD14E4A2nb/AERxxAD5c+0AmplbADUo0AC14DQAXWgRAGsl4wB0oGIAkVlkAOWmTwCm/fEAZqErAEHpqgAMXjkA5AlGAMsOOwAUQAsABvRfAJyVUwDTKM8AtIMYAKh6lQDPkq0A9JtqAKiudAD2I3wAHTsPAOy36ACLtz8ATQELAJCXQgDpxksA4BzzABSR7gD7vMEAgo/6AKovngCZNLoA0A5UALhiFABNVRQAjXIuABmhFACj8/sAt+bQAP71dgCbdGoAXuf5AN5zVAACgggAlHiqAGTPswC0x5oAsgxDAEimygCYlMUAS3r2AKFS0gCZzCYAnXPZAEU8cgAf2gQADqF2ACCY+ABSVIEAW17CANxZWQCk0s0AIF1kALYn7gC9U3MAQqRFAHSUegC1UR8Au99GADrbaQCQQksAOQ6kALUmMgAC0YQAS9rjAMNL9QAQKkcAayg7AOjHXgBNZxcAzIUrAAgq+ACVJPMAV5s2APTVsgAsxX4AMP9cAInlqADcltMAv10FAAzJAgDiHQ4AbsW4ACDTowC897AAx1A8AGw3JwBe6msAf/jJAL1PbwCYYRkAscdyACi8+QCXCEwABWnAAOz4QQC0W60AcToLAH4/+wBion0AzzBXAJCuHgAsv+gAvDmdAA1LaAC04xgA/KU6AAurkwCbpwkAzkTeAA9+SABQqJoAPFHNAK96FgDxMNkAaPelAOG5YgCJsS8AtUp8ALhn8gD7sIYAxoR+AMWFHgA3aRIA00/wABwXJwCmQVQAxU7xAGnkjABvBaUAMz8UAL7qjQBokUEAx8eOAAd/ogDKKqcAsx8EAEpW/AADoDAA/D1+AKVqMwDnkmYA/fOAAGpcfwAlm3UA5+t8AM5tMwCdW3cA2UzjABHxWQBKIx4A7bf3AC+RkgARtBAA27gWADcb/ADjO68AS+g0AMutgwAzRawAtbgLAGjfnwCGINIAowqrAFsDmACIyhgAHfz+AByCiABBwdYASnWAANadowA7KFYADTx1ALG1WAAm4csAti5iACQMZgAeg7MAtrnPAMRmsAAsONAAKR6XAEu5TQCrqMYAd0hcABAU8gCLBFEA2C0MAJ7nZwAETAYA6i53APxR2gDRwBsAf392AH/csgAidosA3MaAAKgCrQBwid4A4xUfAHpqnACYWcQAHfIoAO67cQBlKVUAfR6pAEvQyQDFcWoA82ivAGsVXAA5gcQA8YCIAH9igAC6EzIAvrJEACTfdgBM21wAE6W+ABkqTwAL0+wA85kAALdRfQDbetYAT9RxALUxoQDScSoACRY8AIOzSwAUQS4A6yG1ALDLIwCf5vUAUvQTAK8r+QC5zVoAckBEABQfYwBFzoQAEUyXANlojQBGhjMA0WNjAI0qOQC7TIkAMbboAAp5igBw640ArVL3ADsU7QBoR/cAemWCAChwZQDuILoApIhRAOXapADr/IAApX7IAKvnxABvSHgAV1k1ABqwegD4s6IAT4m1AH/wPgBEnc0AwuIVAO5lzgDsHIAA8O8QAPZZeAC68JcAMzVvAAVKUwBGXv0AaOoBACuF6QD9uBQAwGpOAEiF1wCq/1kAXLJ4AGP4xgC8uzMAHgMmAF3WXwCssiUAya68AP7FUwAkZqYAqnfsAJkuUgC+08cANBobALJvTgAtK7oAaIIoALfiPgAW5qAAO1YyAM9+owCFIMMAbL82ABfR7wBJrDgASc3DAB8q7AAi/80AxleNALQ+3wARtIIA3D9uAFWiEwBtshQAoPHFAITU8gA1S4EAB8w6ANZxHgBhLBEAx70EADf8KgADzHIAQ4ONAEyVRgAXxdIAoS0hAHkIhwBUeFgAOP1YAM6KtgAJaJ0AuaUnADdSkQDNaVIAJkoIAFVlkAARLYwAc2jBAEHoeQBMnFIAQS8hAKUu2ABZafMAQRGbAElVMgAdHRgA9ImMAF4/NADLcZMAiaXbAB0rpQB1RTEAGqsMAGYPWgAB4nkAEk/2AOg7lQC5bI8A7x5cALOosgDQV7YA8mwTACI31wCjuIAAegycABfkWwBSqBQApdm/AIqmCgBEp3QANDopAHSFjgDOIKgABWL8AOppagBvyq0AXyT+AGvkZwDaq1MA8J0CAO0qCADs3gcAj/T/AFHxnwBnObIAAHr0AGNgYQCyAeoAerH5ANuJsQAZnSwAqPWRABmF5gBKI5IAkQpVAOQq0gCdZxQArSpTAFC4aACZFzgA2vozAKKrjwADOGYAxejeANjGkgAogccAnJtJAHa0iQDNsMQAvkg0AAxqhQABL88A0+rRAGam/AAZaiwAakBeAIkbewAlIpIAMQjsAHKsxAAVPRIAH5HeABuShQBm4ocAtOdpAIvy0QDAdbAAnRZ2AFNL6QCFcQYACIehACvVNgChty8Ah5A8AFc4iwBQwRwAv3PeAO+FjABRt0kAMfNNAMz2/QDaFEEAzLRaAJ5hlQBihAoAP/CjAPH4bAC/fxwAPapOAO3BCQDgg2gA7IyRABgp/AAO+QgAXrMFAHPiZwDc83IAmMzHALPPaAAlyrkAAk6uAD33pwBo0iAAC56eAOolMACgdmQAKdJeAA0UPgBtua8AIH66AJMoxwCYv7UA4JcKAN5c/wDgWDwAcWbUANvmLwD73kIAu/vCAFUiCQB5CJIAxOlxADKN8AB4H8EAh8gnAC8h8QCdBUgAX7cyADXejABFCWYAm7v7AOENigCsVMYAY8UeABEpaADYLmMAOWcMANk9hAANhM8AAQHaAMfmVwB2p2AA5MbYAOw44gCKrqQADZOHAM1RlwBhnvAAGkToADueDQDmbJ0AFYYTAGE6UgDjbEgAeJhCAOOGgAC0s6QANh6iAMu3kQAMMBQAl7YOANsIXwCkd9oA8mqMANf38wCxC7EAv7gjAKa0HwCgqV4ApgI/AIe1oQBaGE8AcxcBAObgYQBMMWwA1G0gAF+J9ABkAl8A2buoAIlKWAA/ZykAl6S0ABtBMQAkyOEAzFtyAHIMdgCCfMgAG9m4ABOX7wBfi/wAeg74AELcSAC7RgEAP1V0AD/FBwABALUAdGX+APlznwBqD1MAfTHIALA+HgDRP/QAaraOANCTrgBWaVUANUQjAFVulwBFlpkAsU3sAN86OACsleQAcFHoANanugC10xgAjD4TAA0g3gDcmx4A/eNDAIsxFAC6N5UAJ7fDAFRwZQD/+pAALV5yACfwHwBzltQA9CAuAD0nQQDbjogAKLenAHydkQDwD+IABvePAF9zWAAvQBwA4Ae3AJk+hQCB0ugAHXjgAOBbKQC4/ksAP4CtAAELZgDZ6kEAO2TIAPH6ZAC80CwAd9CIANrJ3QBbk7QAcT0AAE58HwDa9QMAOYjoABoHGQArjcwADSF4AFNcRgCnkjIAsG/cAC7iCwCh3qEAramDAAYS3gBO0jcA+2p2ADpCZwATvIsAa0FuABWrbwD+0jUA+PYEAPXTnQCnvXQAYtYoAKfT2wCPsNUA0bqlAG1+sQBnnqEA7hvxABQYewBmeO4Ackj6AJhuRADlYpAA9O9TAPSSeACW+ZAAa0nLAErmuAABaxcA+WJsAAJrYgDjOEkAwldNAC3lLwDy81wAoYZzADYpmwC509wAtFJdAIFkwgDiIEsAkyBvAMmSIgDb/sgAjTw2AJzz6QA+0Z8AwXaTAD/3lwB0Kg8A63iFAFe95wARFK8AFOY5AMAmfwB+Rj4AVleaAJNp0wBDKCUAh08qAEy8BwAX0CUAs+xgABUglAD3+TAA8jt9AM83uwBUVaEAvErrACRYKgB8iGkAb5fUAP71dwDEJIoAx7wfAAIduQALb3kAjm+cABeJCADKk10AdfclABorYAC0WOgA167pAGsx/QC8htoA+7auAATccACYfLsAszNLADwo8ACOsr0AdJmfABl+xgD7MQ4Ax2x6AMpK4gBV7FkA0g/QAFmDRQA01bkAFKhtAH9PEwCEoukAV2UiAC2B+wCuwRsA560TAK92ZAABWWcAovTVAPS/6AC+1C8AXCkSAM2NegBYrWQAmozZAE2CIgBAgY4A9YmBAKY0DgD3ZLAAOgXaAOOGSwDofs8AHlvXAN204wCjH2YAlRVHAA2cxwBDjN8AwumLAC5MEwAKK9YAyELrANJYkACbtlYAKfF2AGpW8ACRmC0AWHqZAB1CzwAGY9sA0N9+AOS4igAHJNkACm+yAA+sXwBKUUIAaipZAJuECwDrgV0AFMJpAAKMjgAtdggAJYIXAHDtegAgFIoASxGoALACPgA2KPEASXlaAGIX7QAjLsYA62u0ABm8HQB5dUsATxadAK6qQwD8VEkAd1ARAFOFzADC26QAfe2GANRbdwC7nK8AEwy4AB2XrgDAHIsAZSA6AEo5AgAM8GUAth6kAINTsACbAcEAxoCEAIZGPAD4GlwA3aV9AK7CDgCjga4ArWf0AHD1VQBd2TgA+eGcABMlUACIulIAOzg4AHP9zACB2VIAkyv0ALSeowCib4MA3fyrAMP7+gBVtSMAzpdQAMFI1AATJq8ALcZ1AGvp0gC/4oMAH3/0AFPf7wB5xTwAHeDZAALL8gDOEuQA+iN+AMs3AgCP3lIAs+6FABBzswBwlFkAxPuuAEg6pgCvKoIAU9EoAGAPigBqilYACnYGADJYkQCkihkA7FTKAISADQCODaoAYE04ALH1iwAUi7MAzURTAORgwQCts3wAc78IAFkTXwB+A5UAJiRXAJFADwAjr7MAdjo6ANNvLACcXagA09XGAOzjAwC0GJ0AV3hVAI1RkwAuDVgAtcfaABkNCQAljBgABQNSABEgDQB5kmwAIre6AFs6+wAnad4AT4iAAMSvawA7Fn8AlLC9AOscDABRf54ATp8wAMzgnwD29YQASW0XAEj2NwDw2U4A9NokAMTdYQBw+5UAgxCpAEwmpwCyDPwANRUYALHKyQB5mGsA1yJ8AJqjDQCaHDEAMtJrAEvLIQCDVUoAwGn3AETZ3wCLIwEAeJNzAMkj8AAUk2wAe+YVAOVR8wDCzpcAu4jkACv7dAA3vOgApBWtAO+tkAD8tu8Ac7noACsS7QAXfvMA7mjTACJc9QCea+gAx2jFALywtACyxdUA7r2wANGY2wD4JvYAg5VOANb0ggDtytkA8WGyAHMOtwA0vqAAwO5NAJSJDAAUaRUAoDvEAIfetwA9PqoA1jC3AA61XADx70MA7fFUAPQtRgBpiV0AP2XDAKNQvgCyW6UAJ9esADvTdQBm6f4AF1lDADTKvgCXNQQA6SFtAEuSVQBsXjUAmGbSANERuABhSpUAD2OJAGLtFABRMFUA0IDAAPc3TwCtToMASSGaAMDpVwCjm40At6keAG6chAAgkRAAz7GGAG8lHQA7qgwA8BucAMnEEQDEC+4A9g4MALMwnwAEFpMAqzf0AK8BoAA6L6sA0l0OACWT/gCVqp8AfumWABzjDQCtsygAHG0JAHebagCmi2EAztDRALUodwCq1+sAEcGyAGmDQQAVnXMAxMu8APC5wACRds0AzdJ4AEUdzwD/LLAAUdAQADW6IQBIUcUAXGxOAHdGZAARzd8AA7Y5ADE7cABegAcA0eDuAMBHYwA6skcARPfCAEYMxwB2/g8APBr5AFNusABVe1kAw4+OAOxwgwB7LiwAOltxAI0eAwALkk4AG9EZAESUzgD9fdQA4YOFAJVkHACkj4MAsWJxABeFJABHocwATeLwAG+OuQDdhzIAabHNAPogKwCTMKQAkanLAAsZLQDFv6YAXjQFAK1LdwDK4XsAcM7eAGCurgA7dq8AgD8dAHK87QDpwi8AwH3WAM/BcgBW7sAAyUeVAGw91wCHzM8Aa5fYABuR9QBhOk8Ay+OSAGYDiwAHCJAAYjifABxdQABItm0Ame2tAE/4XABIC6sA0aPxAAwi2ACitbQAtsWpAA+jPwB2lCgAqaATAMx3JADF2o4A6fhSABNVmwC6giQAYfj6AD0dkwBkmecAU5sTANWMgwD7D+cAp9QWAI6+QQAkZvcAkvGvAMaNOgAtsuEAF+LgADEdxgDAc9MA7+36APqe/wD41wQAIIDKAHsx1gATEKgAnWdaAFgB0wAyMAEAxTsPADTrDACBAuMALov9APMo1wB1wawAlCj7AP3EbAD1lDQA3xUvAAPdwABvPXQAwPeDAGxwkQAaAUEAbJQ2ACGgEQBn8XAAuu81AKaQhgAzfIQAkp8jADQ4TQBSSaIALjCgAJuAjQBP8AsAHL5SAKmH7ACyx5cAP/VjAG71zgCuL+IAV2JLAN+yFgA7QvgAG0HdAFO6AwCx+AwArlawAB83cABUNtcAe9LbACB2gAAZVTUA4DRYAEFV4wByz2QAWUgkALXS9ACCaEYAxjjiAI0qiAArChIAr7twAPvaOwDhyfkAEoYVAIM5KABarEsAol47AIBNdQDfMHQAN8GNADM6XwBbJFMAMAEbALILpgCxRIsA3mxlAGqpZACd9SkA9uwTABtT4ABUZDIA3cxyAGnKygA3zW4Age3TANrX0QBc2KEAAab2AEQksQDabWAAdmSYAOBdUwDsLkgAOa4fAPi45QAjr5gAZvEs" + var audio=new Audio(src) + audio.onended = finish; + audio.onerror = finish; + try{audio.play()}catch(e){} + try{audio.preload=1}catch(e){} + try{audio.muted=1}catch(e){} + try{audio.playbackRate=567312.2079031984}catch(e){} + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/1526044.html b/dom/media/test/crashtests/1526044.html new file mode 100644 index 0000000000..1914dfc2f5 --- /dev/null +++ b/dom/media/test/crashtests/1526044.html @@ -0,0 +1,19 @@ +<script> + function start () { + try { o1 = new AudioContext({}) } catch (e) { } + try { o2 = new DynamicsCompressorNode(o1, {}) } catch (e) { } + try { o3 = o2.attack } catch (e) { } + try { o4 = new XMLHttpRequest({mozAnon: true, mozSystem: true}) } catch (e) { } + try { o4.open('GET', '', false) } catch (e) { } + try { o4.send() } catch (e) { } + for (let i = 0; i < 20; i++) { + try { o5 = o1.createGain() } catch (e) { } + try { o1.suspend().then(function () { }) } catch (e) { } + try { o5.connect(o3) } catch (e) { } + } + setTimeout('location.reload()', 200) + } + + window.addEventListener('load', start) +</script> + diff --git a/dom/media/test/crashtests/1538727.html b/dom/media/test/crashtests/1538727.html new file mode 100644 index 0000000000..6371d54383 --- /dev/null +++ b/dom/media/test/crashtests/1538727.html @@ -0,0 +1,14 @@ +<script> +const canvas = document.createElement('canvas') +const context = canvas.getContext('2d', {}) +const xhr = new XMLHttpRequest({}) +const stream = canvas.captureStream(new Float64Array([1593177632.1689904])[0]) +recorder = new MediaRecorder(stream) +recorder.start(100) +xhr.open('G', '', false) +xhr.send() +recorder.stop() +tracks = stream.getVideoTracks() +track = tracks[(1051736525 % tracks.length)] +stream.removeTrack(track) +</script> diff --git a/dom/media/test/crashtests/1545133.html b/dom/media/test/crashtests/1545133.html new file mode 100644 index 0000000000..fb7039aae3 --- /dev/null +++ b/dom/media/test/crashtests/1545133.html @@ -0,0 +1,34 @@ +<html class="reftest-wait"> +<head> +<script> +const xhr = new XMLHttpRequest() + +async function boom () { + await new Promise(r => setTimeout(r, 100)) + + SpecialPowers.forceCC() + SpecialPowers.forceCC() + SpecialPowers.forceCC() + + document.documentElement.removeAttribute("class") +} + +function start () { + const context = new AudioContext({}) + const filter = new BiquadFilterNode(context, {}) + const destination = context.createMediaStreamDestination() + const processor = context.createScriptProcessor(8192, 8, 8) + processor.connect(filter.Q) + processor.disconnect() + xhr.open('G', '', false) + xhr.send() + context.createMediaStreamSource(destination.stream) + processor.connect(filter.Q) + context.close() + context.addEventListener('statechange', boom, true) +} + +document.addEventListener('DOMContentLoaded', start) +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/1547784.html b/dom/media/test/crashtests/1547784.html new file mode 100644 index 0000000000..ee270491f1 --- /dev/null +++ b/dom/media/test/crashtests/1547784.html @@ -0,0 +1,33 @@ +<html class="reftest-wait"> +<head> + <script> + const doc = new Document(); + const video = document.createElementNS('http://www.w3.org/1999/xhtml', 'video'); + const source = new MediaSource(); + + navigator.requestMediaKeySystemAccess('org.w3.clearkey', [{ '': [{ '': '' }] }]) + .then(keySystemAccess => { + return keySystemAccess.createMediaKeys(); + }).then(_ => { + video.src = URL.createObjectURL(source); + source.addEventListener('sourceopen', () => { + doc.adoptNode(video); + }); + }); + + navigator.requestMediaKeySystemAccess('org.w3.clearkey', [{ '': [{ '': '' }] }]) + .then(keySystemAccess => { + return keySystemAccess.createMediaKeys(); + }).then(mediaKeys => { + return video.setMediaKeys(mediaKeys); + }).then(() => { + video.src = URL.createObjectURL(source); + document.documentElement.removeAttribute("class"); + }).catch(e => { + // Catch JS errors caused by raciness in the test. So long as we're + // not crashing we're good. + document.documentElement.removeAttribute("class"); + }); + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/1547899.html b/dom/media/test/crashtests/1547899.html new file mode 100644 index 0000000000..4ffd90c565 --- /dev/null +++ b/dom/media/test/crashtests/1547899.html @@ -0,0 +1,20 @@ +<html> +<head> + <script> + function start () { + const video = document.getElementById('id_0') + const stream_1 = new MediaStream() + const stream_2 = video.mozCaptureStreamUntilEnded() + const track = stream_2.getTracks()[0] + + video.srcObject = stream_1 + stream_1.addTrack(track) + } + + window.addEventListener('load', start) + </script> +</head> +<body> +<video id="id_0" src="data:audio/mpeg;base64,ZQr/+1DEAAAAAAAAAAAAAAAAAAAAAABJbmZvAAAADwAAAAsAAAnKABcXFxcXFxcXFy4uLi4uLi4uLkVFRUVFRUVFRV1dXV1dXV1dXXR0dHR0dHR0dIuLi4uLi4uLi6KioqKioqKiorq6urq6urq6utHR0dHR0dHR0ejo6Ojo6Ojo6P///////////wAAADlMQU1FMy45OHIBpQAAAAAuHQAAFEAkBElCAABAAAAJyuGI2MQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA//tQxAAABmQjXHSRAAH6IfB3OVIDAADLlmjIxWKxWTyIABgmGydGjRox4f8Tg+D4OAgc/BwEDn+UBD+qwH/+8Tny7///KO4IYDgcDgcDgcDgcCgQAAAKKBUFc/nnP30a5O0zyYzMNMkzKlPva2GXgMIBbwDQEAYDQNXGLT2EIgEhoFnQNDb4BAPD9wTFYGJBWJw/8AUWgMBQgIEg4KmAKGP/wRG0csCIWDaQxcFhYvv/8WYMUpEIkXhvnRef//kmsuLPH5OvSoAFKxrJIm26DP/7UsQ"> +</body> +</html> diff --git a/dom/media/test/crashtests/1560215.html b/dom/media/test/crashtests/1560215.html new file mode 100644 index 0000000000..1b76e218d7 --- /dev/null +++ b/dom/media/test/crashtests/1560215.html @@ -0,0 +1,20 @@ +<html class="reftest-wait"> +<head> + <script> + async function start () { + const canvas = document.createElement('canvas') + const context = canvas.getContext('2d') + context.fillStyle = "red" + context.fillRect(0, 0, 1, 1) + const recorder = new MediaRecorder( + canvas.captureStream(), { videoBitsPerSecond: 16 }) + recorder.start(100) + await new Promise(r => recorder.onstart = r) + recorder.pause() + document.documentElement.removeAttribute("class") + } + + window.addEventListener('load', start) + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/1569645.html b/dom/media/test/crashtests/1569645.html new file mode 100644 index 0000000000..b1f1247f26 --- /dev/null +++ b/dom/media/test/crashtests/1569645.html @@ -0,0 +1,23 @@ +<html> +<head> + <script> + function start () { + const canvas = document.getElementById("c") + canvas.getContext("2d") + const video = canvas.captureStream() + const ac = new AudioContext() + const dest = ac.createMediaStreamDestination() + const recorder = new MediaRecorder( + new MediaStream([...video.getTracks(), ...dest.stream.getTracks()]), { + 'mimeType': 'audio/ogg' + }) + recorder.start() + } + + window.addEventListener('load', start) + </script> +</head> +<body> +<canvas id="c"></canvas> +</body> +</html> diff --git a/dom/media/test/crashtests/1575271.html b/dom/media/test/crashtests/1575271.html new file mode 100644 index 0000000000..ceda986553 --- /dev/null +++ b/dom/media/test/crashtests/1575271.html @@ -0,0 +1,25 @@ +<html class="reftest-wait"> +<head> +<script> + async function start () { + const canvas = document.createElement("canvas") + const context = canvas.getContext("2d") + context.fillStyle = "blue" + context.fillRect(0, 0, canvas.width, canvas.height) + const stream = canvas.captureStream() + const track = stream.getTracks()[0] + const recorder = new MediaRecorder(stream) + recorder.start() + await new Promise(r => recorder.onstart = r) + recorder.pause() + stream.removeTrack(track) + recorder.resume() + await new Promise(r => recorder.onstop = r) + document.documentElement.removeAttribute("class") + } + + window.addEventListener('load', start) +</script> +</head> +</html> + diff --git a/dom/media/test/crashtests/1577184.html b/dom/media/test/crashtests/1577184.html new file mode 100644 index 0000000000..a38c4a1265 --- /dev/null +++ b/dom/media/test/crashtests/1577184.html @@ -0,0 +1,15 @@ +<html> +<head> +<script> +function start () { + const frame = document.createElementNS('http://www.w3.org/1999/xhtml', 'frame') + document.documentElement.appendChild(frame) + frame.contentWindow.eval('window.top.context=new AudioContext()') + document.documentElement.innerHTML = '' + context.createMediaElementSource(new Audio('')) +} + +document.addEventListener('DOMContentLoaded', start) +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/1587248.html b/dom/media/test/crashtests/1587248.html new file mode 100644 index 0000000000..10c555e2b1 --- /dev/null +++ b/dom/media/test/crashtests/1587248.html @@ -0,0 +1,23 @@ +<html class="reftest-wait"> +<head> +<script> +function start () { + const audio = document.getElementById('id_4') + const doc = new Document() + const stream = new MediaStream() + const track = document.createElementNS('http://www.w3.org/1999/xhtml', 'track') + audio.srcObject = stream + track.textContent = '�' + setTimeout(() => { + track.replaceChild(audio, track.childNodes[0]) + audio.play().then(function (arg4) { }) + document.documentElement.removeAttribute("class") + }, 157) + doc.adoptNode(audio) +} + +document.addEventListener('DOMContentLoaded', start) +</script> +</head> +<audio class="" id="id_4" itemscope></audio> +</html> diff --git a/dom/media/test/crashtests/1594466.html b/dom/media/test/crashtests/1594466.html new file mode 100644 index 0000000000..276c5fe3d1 --- /dev/null +++ b/dom/media/test/crashtests/1594466.html @@ -0,0 +1,22 @@ +<html> +<head> +<script> +function start() { + const ac = new AudioContext(); + const {stream: audioStream} = ac.createMediaStreamDestination(); + const [audioTrack] = audioStream.getTracks(); + const canvas = document.createElement("canvas"); + const ctx = canvas.getContext("2d"); + const [videoTrack] = canvas.captureStream().getTracks(); + + const rec = new MediaRecorder(new MediaStream([audioTrack, videoTrack]), { + mimeType: 'video/webm; codecs="vp8, opus"' + }); + rec.start(); +} + +document.addEventListener('DOMContentLoaded', start) +</script> +</head> +</html> + diff --git a/dom/media/test/crashtests/1601385.html b/dom/media/test/crashtests/1601385.html new file mode 100644 index 0000000000..7192809076 --- /dev/null +++ b/dom/media/test/crashtests/1601385.html @@ -0,0 +1,12 @@ +<html> +<head> +<script> +const video = document.createElement("video"); +video.srcObject = new MediaStream(); +video.mozCaptureStreamUntilEnded(); +video.src = "sound.ogg"; +video.srcObject = undefined; +</script> +</head> +</html> + diff --git a/dom/media/test/crashtests/1601422.html b/dom/media/test/crashtests/1601422.html new file mode 100644 index 0000000000..9ff3c1a07a --- /dev/null +++ b/dom/media/test/crashtests/1601422.html @@ -0,0 +1,20 @@ +<html class="reftest-wait"> +<head> +<script> +(async _ => { + try { + const video = document.createElement('video') + video.preload = 'metadata' + video.src = 'sound.ogg' + await new Promise(r => video.onloadedmetadata = r) + const stream_1 = video.mozCaptureStreamUntilEnded() + video.src = '' + const stream_2 = video.mozCaptureStreamUntilEnded() + } finally { + document.documentElement.removeAttribute("class") + } +})() +</script> +</head> +</html> + diff --git a/dom/media/test/crashtests/1604941.html b/dom/media/test/crashtests/1604941.html new file mode 100644 index 0000000000..5a9265ea1b --- /dev/null +++ b/dom/media/test/crashtests/1604941.html @@ -0,0 +1,22 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script> + +async function boom() +{ + await SpecialPowers.pushPrefEnv({"set": [ + ["media.cubeb.force_null_context", true], + ]}); + new Audio().mozCaptureStreamUntilEnded(); + var ac = new window.AudioContext(); + ac.resume(); + ac.close(); + document.documentElement.removeAttribute("class"); +} + +</script> +</head> +<body onload="boom();"></body> +</html> + diff --git a/dom/media/test/crashtests/1608286.html b/dom/media/test/crashtests/1608286.html new file mode 100644 index 0000000000..9f52605be6 --- /dev/null +++ b/dom/media/test/crashtests/1608286.html @@ -0,0 +1,50 @@ +<html class="reftest-wait">
+<head>
+ <script>
+ function test() {
+ function checkResolve(value) {
+ // Let the test timeout and fail
+ throw new Error("This promise should not resolve");
+ }
+
+ function checkReject(reason) {
+ if (reason.message !== "Browsing context is no longer available") {
+ // Let the test timeout and fail
+ throw new Error("Unexpected rejected promise reason");
+ }
+ // Otherwise, successfully rejected a request not attached to a
+ // window without crashing
+ }
+
+ var i = document.querySelector("iframe");
+ var nav = i.contentWindow.navigator;
+ i.remove();
+
+ // First, check with valid args
+ nav.requestMediaKeySystemAccess(
+ "com.widevine.alpha",
+ [{
+ initDataTypes: ["webm"],
+ videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }]
+ }]
+ ).then(
+ checkResolve,
+ (reason) => {
+ checkReject(reason);
+
+ // Then, check with invalid args
+ nav.requestMediaKeySystemAccess("", []).then(
+ checkResolve,
+ (reason) => {
+ checkReject(reason);
+ document.documentElement.removeAttribute("class");
+ }
+ );
+ });
+ }
+ </script>
+</head>
+<body onload="test()">
+ <iframe></iframe>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1673525.html b/dom/media/test/crashtests/1673525.html new file mode 100644 index 0000000000..51de202999 --- /dev/null +++ b/dom/media/test/crashtests/1673525.html @@ -0,0 +1,15 @@ +<html> +<head> + <script> + document.addEventListener("DOMContentLoaded", () => { + const audio = document.getElementById("audio"); + audio.autoplay = true; + audio.mozCaptureStream(); + }) + </script> +</head> +<body> + <!-- The data URL is crafted from a fuzzed mp3 so the base64 can be decoded back to an mp3 as needed --> + <audio id="audio" src="data:audio/mpeg;base64,//tQxAAAAAAAAAAAAAAAAAAAAAAASW5mbwAAAA8AAAALAAAJygAXFxcXFxcXFxcuLi4uLi4uLi5FRUVFRUVFRUVdXV1dXV1dXV10dHR0dHR0dHSLi4uLi4uLi4uioqKioqKioqK6urq6urq6urrR0dHR0dHR0dHo6Ojo6Ojo6Oj///////////8AAAA5TEFNRTMuOThyAaUAAAAALh0AABRAJARJQgAAQAAACcrhiNjEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP/7UMQAAAZkI1x0kQAB+iHwdzlSAwAAy5ZoyMVisVk8iAAYJhsnRo0aMeH/E4Pg+DgIHPwcBA5/lAQ/qsB//vE58u///yjuCGA4HA4HA4HA4HAoEAAACigVBXP55z99GuTtM8mMzDTJMypT72thl4DCAW8A0BAGA0DVxi09hCIBIaBZ0DQ2+AQDw/cExWBiQVicP/AFFoDAUICBIOCpgChj/8ERtHLAiFg2kMXBYWL7//FmDFKRCJF4b50Xn//5JrLizx+Tr0qABSsaySJtugz/+1LEBIALgQmLvLUAMXkorjzSjuWd/+FbPOduga79Dhef0OFYNhp+FEPDPRyRvOEERl/Uwbv9F/UuQN5yt9WPM5xwqlB4+hzmfmoYmcceWPD8s3WA8FhYEPkhR0SgMHw="></audio> +</body> +</html> diff --git a/dom/media/test/crashtests/459439-1.html b/dom/media/test/crashtests/459439-1.html new file mode 100644 index 0000000000..7bb0131d51 --- /dev/null +++ b/dom/media/test/crashtests/459439-1.html @@ -0,0 +1,36 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +var i = 0; +function boom() +{ + var div = document.getElementById("div"); + var audio = document.getElementById("audio"); + + audio.onload = null; + + div.textContent = "FAIL"; + audio.src += ""; + div.textContent = "PASS?"; + + ++i; + + setTimeout(done, 1); +} + +function done() +{ + // Note we reset 'src' to release decoder resources and cubeb streams to + // prevent OOM or OpenCubeb() failures. + var audio = document.getElementById("audio"); + audio.src = ""; + document.documentElement.removeAttribute("class"); +} +</script> +</head> +<body> +<audio id="audio" autoplay src="sound.ogg" oncanplaythrough="setTimeout(boom, 1);"></audio> +<div id="div"></div> +</body> +</html> diff --git a/dom/media/test/crashtests/466607-1.html b/dom/media/test/crashtests/466607-1.html new file mode 100644 index 0000000000..e154223efe --- /dev/null +++ b/dom/media/test/crashtests/466607-1.html @@ -0,0 +1,14 @@ +<html> +<head> +<script type="text/javascript"> + +function boom() +{ + document.body.appendChild(document.createElementNS("bar", "audio")); + document.body.appendChild(document.createElementNS("bar", "video")); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/466945-1.html b/dom/media/test/crashtests/466945-1.html new file mode 100644 index 0000000000..ac1ba29e36 --- /dev/null +++ b/dom/media/test/crashtests/466945-1.html @@ -0,0 +1,25 @@ +<html class="reftest-wait"> +<head> +<script type="text/javascript"> + +var s; + +function boom() +{ + s = document.createElement("span"); + s.innerHTML = "<video src='data:text/html,' autoplay='autoplay'><\/video>"; + document.body.appendChild(document.createElement("iframe")); + setTimeout(boom2, 0); +} + +function boom2() +{ + s.innerHTML = ""; + document.documentElement.removeAttribute("class"); +} + +</script> +</head> + +<body onload="setTimeout(boom, 0);"></body> +</html> diff --git a/dom/media/test/crashtests/468763-1.html b/dom/media/test/crashtests/468763-1.html new file mode 100644 index 0000000000..d21e5bc388 --- /dev/null +++ b/dom/media/test/crashtests/468763-1.html @@ -0,0 +1 @@ +<html><head></head><body><video src="nosuchprotocol:"></video></body></html>
\ No newline at end of file diff --git a/dom/media/test/crashtests/474744-1.html b/dom/media/test/crashtests/474744-1.html new file mode 100644 index 0000000000..8a7c70cf05 --- /dev/null +++ b/dom/media/test/crashtests/474744-1.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html> +<head> +<script type="text/javascript"> + +function boom() +{ + document.body.innerHTML += "<video src='aim:yaz'><\/video>"; +} + +</script> +</head> +<body onload="boom();"> +</body> +</html> diff --git a/dom/media/test/crashtests/481136-1.html b/dom/media/test/crashtests/481136-1.html new file mode 100644 index 0000000000..bd264058d9 --- /dev/null +++ b/dom/media/test/crashtests/481136-1.html @@ -0,0 +1,3 @@ +<html> +<div><object data='sound.ogg'></div> +</html> diff --git a/dom/media/test/crashtests/492286-1.xhtml b/dom/media/test/crashtests/492286-1.xhtml new file mode 100644 index 0000000000..627ac38723 --- /dev/null +++ b/dom/media/test/crashtests/492286-1.xhtml @@ -0,0 +1 @@ +<source xmlns="http://www.w3.org/1999/xhtml"/>
\ No newline at end of file diff --git a/dom/media/test/crashtests/493915-1.html b/dom/media/test/crashtests/493915-1.html new file mode 100644 index 0000000000..2a6ae9bd6c --- /dev/null +++ b/dom/media/test/crashtests/493915-1.html @@ -0,0 +1,18 @@ +<html> +<head> +<script type="text/javascript"> + +function boom() +{ + s = document.createElement("span"); + a = document.createElement("audio"); + a['src'] = "javascript:4"; + a['loopend'] = 3; + s.appendChild(a); +} + +</script> +</head> + +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/495794-1.html b/dom/media/test/crashtests/495794-1.html new file mode 100644 index 0000000000..2db69206a1 --- /dev/null +++ b/dom/media/test/crashtests/495794-1.html @@ -0,0 +1,8 @@ +<!DOCTYPE html> +<html class="reftest-wait"> + <body> + <audio src="495794-1.ogg" autoplay onended="this.src=''; document.documentElement.className=undefined"></audio> + <!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. --> + </body> +</html> + diff --git a/dom/media/test/crashtests/495794-1.ogg b/dom/media/test/crashtests/495794-1.ogg Binary files differnew file mode 100644 index 0000000000..1c19a64061 --- /dev/null +++ b/dom/media/test/crashtests/495794-1.ogg diff --git a/dom/media/test/crashtests/497734-1.xhtml b/dom/media/test/crashtests/497734-1.xhtml new file mode 100644 index 0000000000..6df055da39 --- /dev/null +++ b/dom/media/test/crashtests/497734-1.xhtml @@ -0,0 +1,21 @@ +<html xmlns="http://www.w3.org/1999/xhtml"> +<head> +<script type="text/javascript"> + +function boom() +{ + + div = document.createElementNS("http://www.w3.org/1999/xhtml", "div"); + div.appendChild(document.getElementById("v")); + document.body.appendChild(div); +} + +</script> +</head> + +<body onload="boom();"> + +<video id="v"><source></source></video> + +</body> +</html> diff --git a/dom/media/test/crashtests/497734-2.html b/dom/media/test/crashtests/497734-2.html new file mode 100644 index 0000000000..990ac4af46 --- /dev/null +++ b/dom/media/test/crashtests/497734-2.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html> +<script> + +function boom() +{ + audio1 = document.createElement("audio"); + (audio1).appendChild(document.createElement("source")); + (audio1).appendChild(document.createElement("source")); + setTimeout(function() { + audio2 = document.createElement("audio"); + audio2.appendChild(audio1); + }, 100); +} + +</script> +<body onload="boom();"></body> diff --git a/dom/media/test/crashtests/576612-1.html b/dom/media/test/crashtests/576612-1.html new file mode 100644 index 0000000000..04f993e780 --- /dev/null +++ b/dom/media/test/crashtests/576612-1.html @@ -0,0 +1,15 @@ +<html xmlns="http://www.w3.org/1999/xhtml"> +<head> +<script> +function boom() +{ + + var v = document.getElementById("v"); + v.src = "data:text/plain,_"; + document.documentElement.appendChild(v); + +} +</script> +</head> +<body onload="boom();"><video id="v" src="data:video/ogg;codecs="theora,vorbis",1"></video></body> +</html> diff --git a/dom/media/test/crashtests/691096-1.html b/dom/media/test/crashtests/691096-1.html new file mode 100644 index 0000000000..3c3ebfc12a --- /dev/null +++ b/dom/media/test/crashtests/691096-1.html @@ -0,0 +1,31 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script> +var ITERATIONS = 200; + +function stoptest (evt) +{ + if (evt) { + // Note we reset 'src' to release decoder resources and cubeb streams to + // prevent OOM or OpenCubeb() failures. + evt.target.src = ""; + } + document.documentElement.removeAttribute("class"); +} + +function boom() +{ + for (var i = 0; i < ITERATIONS; ++i) { + a = document.createElementNS("http://www.w3.org/1999/xhtml", "audio"); + a.addEventListener("loadedmetadata", stoptest); + a.setAttributeNS(null, "autoplay", ""); + a.setAttributeNS(null, "src", "sound.ogg"); + } + setTimeout(stoptest, 250); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/752784-1.html b/dom/media/test/crashtests/752784-1.html new file mode 100644 index 0000000000..4644eeb89a --- /dev/null +++ b/dom/media/test/crashtests/752784-1.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html> +<head> +<script> +function boom() +{ + document.getElementById("a").mozCaptureStream(); +} +</script> +</head> + +<body onload="boom();"> +<audio id="a" src="sound.ogg"> +</body> +</html> diff --git a/dom/media/test/crashtests/789075-1.html b/dom/media/test/crashtests/789075-1.html new file mode 100644 index 0000000000..6cd673fb75 --- /dev/null +++ b/dom/media/test/crashtests/789075-1.html @@ -0,0 +1,20 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +</head> +<body> +<video src="789075.webm" preload="metadata" id="v"> +</video> +<!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. --> +<script type="application/javascript"> + var video = document.getElementById("v"); + video.onloadeddata = function () { + video.play(); + }; + video.onended = function () { + video.src=""; + document.documentElement.className = undefined; + }; +</script> +</body> +</html> diff --git a/dom/media/test/crashtests/789075.webm b/dom/media/test/crashtests/789075.webm Binary files differnew file mode 100644 index 0000000000..602e53fca2 --- /dev/null +++ b/dom/media/test/crashtests/789075.webm diff --git a/dom/media/test/crashtests/795892-1.html b/dom/media/test/crashtests/795892-1.html new file mode 100644 index 0000000000..d73cea7f2e --- /dev/null +++ b/dom/media/test/crashtests/795892-1.html @@ -0,0 +1,23 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<script> +function boom() +{ + var a = document.getElementById("a"); + a.play(); + a.onplaying = function () { + a.onplaying = null; + // Note we reset 'src' to release decoder resources and cubeb streams to + // prevent OOM or OpenCubeb() failures. + a.src = ""; + document.documentElement.className = ""; + } +} +</script> +</head> + +<body> +<video id="a" src="cors.webm" crossorigin preload="metadata" onloadedmetadata="boom();"> +</body> +</html> diff --git a/dom/media/test/crashtests/844563.html b/dom/media/test/crashtests/844563.html new file mode 100644 index 0000000000..7f9365511f --- /dev/null +++ b/dom/media/test/crashtests/844563.html @@ -0,0 +1,5 @@ +<script> +var a = document.createElementNS("http://www.w3.org/1999/xhtml", "audio"); +a.mozPreservesPitch = a; +</script> + diff --git a/dom/media/test/crashtests/846612.html b/dom/media/test/crashtests/846612.html new file mode 100644 index 0000000000..070c375381 --- /dev/null +++ b/dom/media/test/crashtests/846612.html @@ -0,0 +1,8 @@ +<script> +document.addEventListener("DOMContentLoaded", function() { + var a = document.querySelector("audio"); + a.playbackRate = 2; + a.play(); +}); +</script> +<audio src="495794-1.ogg"></audio> diff --git a/dom/media/test/crashtests/852838.html b/dom/media/test/crashtests/852838.html new file mode 100644 index 0000000000..0bea29351b --- /dev/null +++ b/dom/media/test/crashtests/852838.html @@ -0,0 +1,11 @@ +<!DOCTYPE html> +<html> + <head> + <script> + var o0 = new window.AudioContext(); + var o1 = o0.createBuffer(536870912, 1, 8192); + </script> + </head> + <body> + </body> +</html> diff --git a/dom/media/test/crashtests/865004.html b/dom/media/test/crashtests/865004.html new file mode 100644 index 0000000000..4da39071c4 --- /dev/null +++ b/dom/media/test/crashtests/865004.html @@ -0,0 +1,19 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<meta charset="UTF-8"> +<script> + +function boom() +{ + var ac = new AudioContext(); + for (var j = 0; j < 200; ++j) { + ac.createScriptProcessor(undefined); + } + document.documentElement.removeAttribute("class"); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/865537-1.html b/dom/media/test/crashtests/865537-1.html new file mode 100644 index 0000000000..4065eb3608 --- /dev/null +++ b/dom/media/test/crashtests/865537-1.html @@ -0,0 +1,13 @@ +<!DOCTYPE HTML> +<html> +<body onload="doTest()"> +<base href="../unknown"> +<div id="test3"></div> +<video id="test4"><source src="white.webm"></video> +<script> +function doTest() { + test3.appendChild(test4); +} +</script> +</body> +</html> diff --git a/dom/media/test/crashtests/865550.html b/dom/media/test/crashtests/865550.html new file mode 100644 index 0000000000..b8626e8d67 --- /dev/null +++ b/dom/media/test/crashtests/865550.html @@ -0,0 +1,22 @@ +<html class="reftest-wait"> + <head> + <script> + var i = 0; + var interval; + function crash() { + var o0 = new AudioContext(); + o1 = o0.createBufferSource(); + ++i; + if (i == 2000) { + document.documentElement.removeAttribute("class"); + clearInterval(interval); + } + } + function start() { + interval = setInterval("crash()", 0) + } + </script> + </head> + <body onload="start()"> + </body> +</html> diff --git a/dom/media/test/crashtests/868504.html b/dom/media/test/crashtests/868504.html new file mode 100644 index 0000000000..94090c1c09 --- /dev/null +++ b/dom/media/test/crashtests/868504.html @@ -0,0 +1,14 @@ +<!DOCTYPE html> +<html> +<head> +<script> + +function boom() +{ + new AudioContext().createBufferSource().playbackRate.linearRampToValueAtTime(0, -1); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/874869.html b/dom/media/test/crashtests/874869.html new file mode 100644 index 0000000000..1fe3dbd2fc --- /dev/null +++ b/dom/media/test/crashtests/874869.html @@ -0,0 +1,15 @@ +<script> +var Context0= new AudioContext() +var Panner0=Context0.createPanner(); +var BufferSource3=Context0.createBufferSource(); +Panner0.channelCount=0; +BufferSource3.connect(Panner0); +BufferSource3.buffer=function(){ + var length=1; + var Buffer=Context0.createBuffer(1,length,'44200'); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = 255;}; + return Buffer; +}(); + +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/874915.html b/dom/media/test/crashtests/874915.html new file mode 100644 index 0000000000..59218b3da3 --- /dev/null +++ b/dom/media/test/crashtests/874915.html @@ -0,0 +1,24 @@ +<script> +var Context0= new AudioContext() +var BufferSource6=Context0.createBufferSource(); + +setInterval(function(){ +BufferSource6.buffer=function(){ + var length=11283; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(624))}; + return Buffer; +}(); +},0) + +BufferSource6.start(0.15831333969254047,0.23571860056836158,0.529235512483865); + +BufferSource6.buffer=function(){ + var length=48517; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(365))}; + return Buffer; +}(); +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/874934.html b/dom/media/test/crashtests/874934.html new file mode 100644 index 0000000000..350ce28101 --- /dev/null +++ b/dom/media/test/crashtests/874934.html @@ -0,0 +1,23 @@ +<script> +var Context0= new AudioContext() +var BufferSource0=Context0.createBufferSource(); +BufferSource0.start(0.01932738965842873,0.33345631847623736,0.3893404237460345); +BufferSource0.buffer=function(){ + var length=35887; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0.39765272522345185))}; + return Buffer; +}(); + +BufferSource0.buffer=function(){ + var length=15952; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(85))}; + return Buffer; +}(); + +BufferSource0.playbackRate.value=20.401213286832185; + +</script> diff --git a/dom/media/test/crashtests/874952.html b/dom/media/test/crashtests/874952.html new file mode 100644 index 0000000000..a9a398fdb2 --- /dev/null +++ b/dom/media/test/crashtests/874952.html @@ -0,0 +1,11 @@ +<script> +var Context0= new AudioContext() +var ChannelSplitter0=Context0.createChannelSplitter(); +var BiquadFilter0=Context0.createBiquadFilter(); +var WaveShaper0=Context0.createWaveShaper(); + +ChannelSplitter0.connect(BiquadFilter0,3,0); +ChannelSplitter0.connect(WaveShaper0); +BiquadFilter0.disconnect(); +WaveShaper0.connect(ChannelSplitter0); +</script> diff --git a/dom/media/test/crashtests/875144.html b/dom/media/test/crashtests/875144.html new file mode 100644 index 0000000000..bf5d0d0861 --- /dev/null +++ b/dom/media/test/crashtests/875144.html @@ -0,0 +1,81 @@ +<script> +Logger = {} +Logger.error = function(e) {} +Logger.comment = function(e) {} + +try { o0 = document.createElement('audio'); } catch(e) { Logger.error(Logger.comment(e)); } +try { (document.body || document.documentElement).appendChild(o0); } catch(e) { Logger.error(Logger.comment(e)); } +try { o1 = new AudioContext(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o2 = o1.createGain(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3 = o1.createBufferSource(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o4 = o1.createBuffer(1, 3, 52970); +o5 = o4.getChannelData(0); +for(var i=0; i<3; ++i) { +o5[i] = Math.sin(i * 63); +} +return o4; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o6 = o1.createBuffer(1, 15, 41218); +o7 = o6.getChannelData(0); +for(var i=0; i<15; ++i) { +o7[i] = Math.sin(i * 0); +} +return o6; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o8 = o1.createBuffer(1, 0, 49074); +o9 = o8.getChannelData(0); +for(var i=0; i<0; ++i) { +o9[i] = Math.sin(i * 0); +} +return o8; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o10 = o1.createBuffer(1, 31, 86527); +o11 = o10.getChannelData(0); +for(var i=0; i<31; ++i) { +o11[i] = Math.sin(i * 127); +} +return o10; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.stop(-1) } catch(e) { Logger.error(Logger.comment(e)); } +/* [Exception... "An attempt was made to use an object that is not, or is no longer, usable" code: "11" nsresult: "0x8053000b (InvalidStateError)" location: "file:///Users/cdiehl/dev/projects/peach/Peach/Utilities/JS/undefined.js Line: 602"] */ +try { o3.channelCountMode = 'explicit'; } catch(e) { Logger.error(Logger.comment(e)); } +try { o12 = o1.createBiquadFilter(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o13 = o1.createBuffer(1, 63, 28347); +o14 = o13.getChannelData(0); +for(var i=0; i<63; ++i) { +o14[i] = Math.sin(i * 15); +} +return o13; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.channelCount = 1; } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.connect(GainNode, 65536, 0) } catch(e) { Logger.error(Logger.comment(e)); } +/* TypeError: Value does not implement interface AudioNode. */ +try { o3.buffer = function() { o15 = o1.createBuffer(1, 1, 72540); +o16 = o15.getChannelData(0); +for(var i=0; i<1; ++i) { +o16[i] = Math.sin(i * 7); +} +return o15; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.getFrequencyResponse(new Float32Array(7), new Float32Array(127), new Float32Array(7)) } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.getFrequencyResponse(new Float32Array(15), new Float32Array(127), new Float32Array(7)) } catch(e) { Logger.error(Logger.comment(e)); } +try { o17 = document.createElement('audio'); } catch(e) { Logger.error(Logger.comment(e)); } +try { (document.body || document.documentElement).appendChild(o0); } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o18 = o1.createBuffer(1, 7, 91261); +o19 = o18.getChannelData(0); +for(var i=0; i<7; ++i) { +o19[i] = Math.sin(i * 7); +} +return o18; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.getFrequencyResponse(new Float32Array(31), new Float32Array(31), new Float32Array(127)) } catch(e) { Logger.error(Logger.comment(e)); } +try { o20 = o1.createChannelSplitter(1, 2, 4, 16, 32); } catch(e) { Logger.error(Logger.comment(e)); } +try { o12.channelCountMode = 'explicit'; } catch(e) { Logger.error(Logger.comment(e)); } +try { o3.buffer = function() { o21 = o1.createBuffer(1, 0, 14451); +o22 = o21.getChannelData(0); +for(var i=0; i<0; ++i) { +o22[i] = Math.sin(i * 63); +} +return o21; +}(); } catch(e) { Logger.error(Logger.comment(e)); } +</script> diff --git a/dom/media/test/crashtests/875596.html b/dom/media/test/crashtests/875596.html new file mode 100644 index 0000000000..7bac09dab7 --- /dev/null +++ b/dom/media/test/crashtests/875596.html @@ -0,0 +1,12 @@ +<script> +try { o1 = new AudioContext(); } catch(e) { } +try { o6 = o1.createGain(); } catch(e) { } +try { o8 = o1.createBufferSource(); } catch(e) { } +try { o6.gain.setValueCurveAtTime(new Float32Array(7), 0, 0) } catch(e) { } +try { o8.connect(o6, 0, 0) } catch(e) { } +try { o8.buffer = function() { + o19 = o1.createBuffer(1, 1, 76309); + for(var i=0; i<1; ++i) { } + return o19; +}(); } catch(e) { } +</script> diff --git a/dom/media/test/crashtests/875911.html b/dom/media/test/crashtests/875911.html new file mode 100644 index 0000000000..fbc52642b4 --- /dev/null +++ b/dom/media/test/crashtests/875911.html @@ -0,0 +1,3 @@ +<script> + new OfflineAudioContext(1, 10, 48000); +</script> diff --git a/dom/media/test/crashtests/876024-1.html b/dom/media/test/crashtests/876024-1.html new file mode 100644 index 0000000000..5502d8e42d --- /dev/null +++ b/dom/media/test/crashtests/876024-1.html @@ -0,0 +1,5 @@ +<script> +o1 = new window.AudioContext(2, 8, 44100); +o4 = o1.createBiquadFilter(); +o4.detune.setValueAtTime(0.0843, 1.7976931348623157e+308); +</script> diff --git a/dom/media/test/crashtests/876024-2.html b/dom/media/test/crashtests/876024-2.html new file mode 100644 index 0000000000..4b9beb7453 --- /dev/null +++ b/dom/media/test/crashtests/876024-2.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html> +<head> +<meta charset="UTF-8"> +<script> + +function boom() +{ + var bufferSource = new AudioContext().createScriptProcessor().context.createBufferSource(); + bufferSource.start(0, 0, 0); + bufferSource.stop(562949953421313); +} + +</script></head> + +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/876118.html b/dom/media/test/crashtests/876118.html new file mode 100644 index 0000000000..bc0630350a --- /dev/null +++ b/dom/media/test/crashtests/876118.html @@ -0,0 +1,16 @@ +<script> +var Context0= new AudioContext() +var BufferSource7=Context0.createBufferSource(); + +BufferSource7.connect(Context0.destination); +BufferSource7.buffer=function(){ + var length=7; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0);for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-1))}; + return Buffer; +}(); + + +Context0.destination.channelCountMode="explicit"; +Context0.destination.channelCount=519910189000000; +</script> diff --git a/dom/media/test/crashtests/876207.html b/dom/media/test/crashtests/876207.html new file mode 100644 index 0000000000..7bfcb096b2 --- /dev/null +++ b/dom/media/test/crashtests/876207.html @@ -0,0 +1,30 @@ +<script> +try { o1 = new window.OfflineAudioContext(1, 10, 44100); } catch(e) { } +try { o2 = o1.createChannelSplitter(1); } catch(e) { } +try { o3 = o1.createAnalyser(); } catch(e) { } +try { o4 = o1.createWaveShaper(); } catch(e) { } +try { o5 = o1.createChannelSplitter(6); } catch(e) { } +try { o6 = o1.createAnalyser(); } catch(e) { } +try { o7 = o1.createBufferSource(); } catch(e) { } +try { o7.connect(o1.destination); } catch(e) { } +try { o7.buffer = function() { +o8 = o1.createBuffer(1, 3, o1.sampleRate); +o9 = o8.getChannelData(0); +for(var i = 0; i < 3; ++i) { +o9[i] = Math.sin(i * 127); +} +return o8; +}() +; } catch(e) { } +try { o7.playbackRate.cancelScheduledValues(0) } catch(e) { } +try { o7.channelInterpretation = 'speakers'; } catch(e) { } +try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { } +try { o5.connect(o1.destination, 1, 1) } catch(e) { } +try { o1.listener.setOrientation(0, 1073741824, 1073741824, 4194304, 1, 0) } catch(e) { } +try { o4.curve = new Float32Array(127); } catch(e) { } +try { o6.getByteFrequencyData(new Uint8Array(12)) } catch(e) { } +try { o6.disconnect() } catch(e) { } +try { o1.destination.channelCount = 33554432; } catch(e) { } +try { o7.playbackRate.setTargetAtTime(0, 8388608, 1) } catch(e) { } +try { o6.getByteTimeDomainData(new Uint8Array(12)) } catch(e) { } +</script> diff --git a/dom/media/test/crashtests/876215.html b/dom/media/test/crashtests/876215.html new file mode 100644 index 0000000000..07135e3628 --- /dev/null +++ b/dom/media/test/crashtests/876215.html @@ -0,0 +1,14 @@ +<script> +try { o1 = new window.OfflineAudioContext(1, 10, 44100); } catch(e) { } +try { o6 = o1.createScriptProcessor(0, 0, 2); } catch(e) { } +try { o8 = o1.createBufferSource(); } catch(e) { } +try { o8.connect(o1.destination); } catch(e) { } +try { o8.connect(o6) } catch(e) { } +try { o3.disconnect() } catch(e) { } +try { o8.buffer = function() { +o21 = o1.createBuffer(1, 3, o1.sampleRate); +o22 = o21.getChannelData(0); +return o21; +}() +; } catch(e) { } +</script> diff --git a/dom/media/test/crashtests/876249.html b/dom/media/test/crashtests/876249.html new file mode 100644 index 0000000000..6f72b40bc1 --- /dev/null +++ b/dom/media/test/crashtests/876249.html @@ -0,0 +1,27 @@ +<script> +var Context0= new AudioContext() +var BufferSource0=Context0.createBufferSource(); +var BiquadFilter0=Context0.createBiquadFilter(); +BufferSource0.start(0,0.6167310480959713,0.7142638498917222); +BiquadFilter0.connect(Context0.destination); +BufferSource0.buffer=function(){ + var length=86333; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-1))}; + return Buffer; +}(); + +BufferSource0.connect(BiquadFilter0); + +BufferSource0.buffer=function(){ + var length=21989; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0))}; + return Buffer; +}(); + +BufferSource0.stop(0.04184641852043569); + +</script> diff --git a/dom/media/test/crashtests/876252.html b/dom/media/test/crashtests/876252.html new file mode 100644 index 0000000000..cb9cb63ed3 --- /dev/null +++ b/dom/media/test/crashtests/876252.html @@ -0,0 +1,23 @@ +<script> +var Context0= new AudioContext() +var BufferSource4=Context0.createBufferSource(); +BufferSource4.start(0.05386466556228697,0.397192713804543,0.48810303467325866); + +BufferSource4.buffer=function(){ + var length=109076; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(370))}; + return Buffer; +}(); + +BufferSource4.buffer=function(){ + var length=19339; + var Buffer=Context0.createBuffer(1,length,53362); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-0.16235407581552863))}; + return Buffer; +}(); + +BufferSource4.stop(0.46482366253621876); +</script> diff --git a/dom/media/test/crashtests/876834.html b/dom/media/test/crashtests/876834.html new file mode 100644 index 0000000000..f4ca6ee558 --- /dev/null +++ b/dom/media/test/crashtests/876834.html @@ -0,0 +1,4 @@ +<!DOCTYPE html> +<script> +new OfflineAudioContext(0, 0, 3229622); +</script> diff --git a/dom/media/test/crashtests/877527.html b/dom/media/test/crashtests/877527.html new file mode 100644 index 0000000000..c639d501b7 --- /dev/null +++ b/dom/media/test/crashtests/877527.html @@ -0,0 +1,37 @@ +<script> +try { o1 = new window.AudioContext(2, 5, 44100); } catch(e) { } +try { o2 = o1.createChannelMerger(1); } catch(e) { } +try { o3 = o1.createDelay(10); } catch(e) { } +try { o4 = o1.createBuffer(2, 2048, 8000); } catch(e) { } +try { o5 = o1.createPanner(); } catch(e) { } +try { o6 = o1.createBufferSource(); } catch(e) { } +try { o7 = (function() { +var buf = o1.createBuffer(1, 50000, o1.sampleRate); +for(var j=0; j<1; ++j) { +for(var i=0; i<50000; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (9.8));} +} +return buf; +})(); } catch(e) { } +try { o6.buffer = o7; } catch(e) { } +try { o6.connect(o5); } catch(e) { } +try { o5.connect(o1.destination); } catch(e) { } +try { o1.listener.speedOfSound = 0.0000019073486328125; } catch(e) { } +try { o6.loop = true; } catch(e) { } +try { o8 = (function() { +var buf = o1.createBuffer(2, 1000, o1.sampleRate); +for(var j=0; j<2; ++j) { +for(var i=0; i<1000; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (1));} +} +return buf; +})(); } catch(e) { } +try { o6.buffer = o7; } catch(e) { } +try { o6.connect(o5); } catch(e) { } +try { o5.connect(o1.destination); } catch(e) { } +try { o6.loopEnd = 1.4901161193847656e-8; } catch(e) { } +try { o6.connect(o1.destination); } catch(e) { } +try { o6.buffer = o8; } catch(e) { } +try { o5.setPosition(0.36, o1.destination.context.destination.channelCountMode, o1.destination.context.destination.channelInterpretation) } catch(e) { } +try { o2.channelCountMode = 'explicit'; } catch(e) { } +try { o1.listener.speedOfSound = 4; } catch(e) { } +try { o1.startRendering(); } catch(e) { } +</script> diff --git a/dom/media/test/crashtests/877820.html b/dom/media/test/crashtests/877820.html new file mode 100644 index 0000000000..6d65c1e9d9 --- /dev/null +++ b/dom/media/test/crashtests/877820.html @@ -0,0 +1,4 @@ +<script> +o1 = new window.OfflineAudioContext(2, 5, 0.39); +window.location.reload(); +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/878014.html b/dom/media/test/crashtests/878014.html new file mode 100644 index 0000000000..4a0ca6dce2 --- /dev/null +++ b/dom/media/test/crashtests/878014.html @@ -0,0 +1,31 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +try { o2 = new window.AudioContext(1, 15, 44100); } catch(e) { } +try { o5 = o2.createDelay(0.1); } catch(e) { } +try { o6 = o2.createPanner(); } catch(e) { } +try { o8 = o2.createBufferSource(); } catch(e) { } +try { o9 = (function() { +var buf = o2.createBuffer(1, 12134, o2.sampleRate); +for(var j=0; j<1; ++j) { +for(var i=0; i<12134; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (-127));} +} +return buf; +})(); } catch(e) { } +try { o8.buffer = o9; } catch(e) { } +try { o8.connect(o6); } catch(e) { } +try { o6.connect(o5, 0, 0) } catch(e) { } +try { o8.buffer = (function() { +var buf = o2.createBuffer(1, 5409, o2.sampleRate); +for(var j=0; j<1; ++j) { +for(var i=0; i<5409; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (-1));} +} +return buf; +})(); } catch(e) { } +setTimeout(function() { try { o5.delayTime.setValueAtTime(0.78, 121560862.56366833); } catch(e) { } setTimeout(done, 0); },128) +try { o5.delayTime.value = 0.4283; } catch(e) { } + +function done() { + document.documentElement.removeAttribute("class"); +} +</script> diff --git a/dom/media/test/crashtests/878328.html b/dom/media/test/crashtests/878328.html new file mode 100644 index 0000000000..ec7b39871b --- /dev/null +++ b/dom/media/test/crashtests/878328.html @@ -0,0 +1,5 @@ +<script> +o1 = new window.AudioContext(2, 7, 44100); +o5 = o1.createDelay(1); +o5.delayTime.setValueCurveAtTime(new Float32Array(3), 1.7976931348623157e+308, 0.64); +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/878407.html b/dom/media/test/crashtests/878407.html new file mode 100644 index 0000000000..ae2918dc85 --- /dev/null +++ b/dom/media/test/crashtests/878407.html @@ -0,0 +1,11 @@ +<script> +o1 = new window.AudioContext(); +o4 = o1.createDelay(0.4468); +o6 = o1.createPanner(); +o7 = (function() {var buf = o1.createBuffer(1, 1000, o1.sampleRate);for(var j=0; j<1; ++j) {for(var i=0; i<1000; ++i) {buf.getChannelData(j)[i] = Math.sin(i * (-15));}}return buf;})(); +o8 = o1.createBufferSource(); +o8.buffer = o7; +o8.connect(o6); +o6.connect(o4) +o4.delayTime.setValueAtTime(0.6019893103749466289898, 0.26) +</script> diff --git a/dom/media/test/crashtests/878478.html b/dom/media/test/crashtests/878478.html new file mode 100644 index 0000000000..89a47ddb55 --- /dev/null +++ b/dom/media/test/crashtests/878478.html @@ -0,0 +1,30 @@ +<script> +var Context0= new AudioContext() +var BufferSource0=Context0.createBufferSource(); + +BufferSource0.loop=true; + +BufferSource0.buffer=function(){ + var channels=3; + var length=97252; + var Buffer=Context0.createBuffer(channels,length,Context0.sampleRate); + for(var y=0;y<channels;y++){ + var bufferData= Buffer.getChannelData(y); + for (var i = 0; i < length; ++i) { bufferData[i] = 1;} + }; + return Buffer; +}(); + +BufferSource0.playbackRate.cancelScheduledValues(0.4); + +BufferSource0.loopEnd=5e-324; + +BufferSource0.buffer=function(){ + var length=26590; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = 1}; + return Buffer; +}(); + +</script> diff --git a/dom/media/test/crashtests/880129.html b/dom/media/test/crashtests/880129.html new file mode 100644 index 0000000000..775e9d80ba --- /dev/null +++ b/dom/media/test/crashtests/880129.html @@ -0,0 +1,9 @@ +<script> +try { o1 = new window.AudioContext(3, 2, 44100); } catch(e) { } +try { o6 = o1.createBufferSource(); } catch(e) { } +try { o15 = o1.createAnalyser(); } catch(e) { } +try { o15.fftSize = 32; } catch(e) { } +try { o6.connect(o15,0,0) } catch(e) { } +try { o27 = o1.createPanner(); } catch(e) { } +try { o6.buffer = function(){var buffer = o1.createBuffer(2, 1148, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1148; i++) {buffer.getChannelData(c)[i] = 0;}}return buffer;}() } catch(e) { } +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/880202.html b/dom/media/test/crashtests/880202.html new file mode 100644 index 0000000000..dc0fade9db --- /dev/null +++ b/dom/media/test/crashtests/880202.html @@ -0,0 +1,33 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var Context0= new window.OfflineAudioContext(15,12119,44100) +var BufferSource1=Context0.createBufferSource(); +var Gain0=Context0.createGain(); +var Panner0=Context0.createPanner(); +Context0.listener.setPosition(29,135,158); + +BufferSource1.connect(Gain0); + +BufferSource1.buffer=function(){ + var channels=3; + var length=53325; + var Buffer=Context0.createBuffer(channels,length,Context0.sampleRate); + for(var y=0;y<channels;y++){ + var bufferData= Buffer.getChannelData(y); + for (var i = 0; i < length; ++i) { + bufferData[i] = i*(270); + } + }; + return Buffer; +}(); + +Gain0.connect(Panner0); +Panner0.panningModel="equalpower"; + + +setTimeout(function(){ +document.documentElement.removeAttribute("class"); +},500) + +</script> diff --git a/dom/media/test/crashtests/880342-1.html b/dom/media/test/crashtests/880342-1.html new file mode 100644 index 0000000000..7d1aa0c3e3 --- /dev/null +++ b/dom/media/test/crashtests/880342-1.html @@ -0,0 +1,208 @@ +<script> +try { o1 = new window.AudioContext(2, 10, 1019159); } catch(e) { } +try { o2 = o1.createBufferSource(); } catch(e) { } +try { o3 = o1.createConvolver(); } catch(e) { } +try { o4 = o1.createChannelMerger(2); } catch(e) { } +try { o5 = o1.createAnalyser(); } catch(e) { } +try { o6 = o1.createPanner(); } catch(e) { } +try { o7 = o1.createDynamicsCompressor(); } catch(e) { } +try { o8 = o1.createWaveShaper(); } catch(e) { } +try { o9 = o1.createChannelSplitter(6); } catch(e) { } +try { o10 = o1.createScriptProcessor(8192, 2, 3); } catch(e) { } +try { o1.listener.setPosition(-144115188075855870, 0, -5e-324) } catch(e) { } +try { o3.buffer = function(){var buffer = o1.createBuffer(2, 741, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<741; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); } catch(e) { } +try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { } +try { o3.connect(o5,0,0) } catch(e) { } +try { o9.channelCountMode = 'max'; } catch(e) { } +try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { } +try { o7.channelInterpretation = 'speakers'; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1887, 63346);for(var c=0; c<2; c++) {for(var i=0; i<1887; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.playbackRate.exponentialRampToValueAtTime(3.819464020334534, 32) } catch(e) { } +try { o5.connect(o9,2,0) } catch(e) { } +try { o9.channelCountMode = 'explicit'; } catch(e) { } +try { o1.listener.speedOfSound = 72057594037927940; } catch(e) { } +try { o1.destination.channelCountMode = 'explicit'; } catch(e) { } +try { o7.threshold.value = 0.0646435404346891590021684237399313133209944; } catch(e) { } +try { o7.connect(o10,0,0) } catch(e) { } +try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { } +try { o3.connect(o4,0,0) } catch(e) { } +try { o7.release.value = 23.030355486273447; } catch(e) { } +try { o9.connect(o9,4,0) } catch(e) { } +try { o6.channelCountMode = 'max'; } catch(e) { } +try { o7.threshold.value = 0.6395867363641939418172910336579661816358566284179687500; } catch(e) { } +try { o2.connect(o9,3,0) } catch(e) { } +try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { } +try { o7.connect(o10,0,0) } catch(e) { } +try { o5.connect(o5,0,0) } catch(e) { } +try { o2.playbackRate.value = 5e-324; } catch(e) { } +try { o4.channelInterpretation = 'discrete'; } catch(e) { } +try { o2.playbackRate.value = 1.2211628339508704; } catch(e) { } +try { o2.channelCountMode = 'clamped-max'; } catch(e) { } +try { o1.listener.dopplerFactor = 32; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(22, 1811, 21177);for(var c=0; c<22; c++) {for(var i=0; i<1811; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o5.channelCountMode = 'max'; } catch(e) { } +try { o1.listener.speedOfSound = 1024; } catch(e) { } +try { o2.playbackRate.value = 9.999999999999998e-91; } catch(e) { } +try { o1.listener.speedOfSound = 16; } catch(e) { } +try { o8.curve = new Float32Array(256); } catch(e) { } +try { o1.listener.setVelocity(0, 128, 4) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(23, 483, o1.sampleRate);for(var c=0; c<23; c++) {for(var i=0; i<483; i++) {buffer.getChannelData(c)[i] = 0.026;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 945, 43803);for(var c=0; c<3; c++) {for(var i=0; i<945; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(19, 997, 23781);for(var c=0; c<19; c++) {for(var i=0; i<997; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.listener.speedOfSound = -8; } catch(e) { } +try { o3.channelCountMode = 'explicit'; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1740, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1740; i++) {buffer.getChannelData(c)[i] = 8;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1057, 30602);for(var c=0; c<1; c++) {for(var i=0; i<1057; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 78, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<78; i++) {buffer.getChannelData(c)[i] = 10;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.listener.setPosition(9.999999999999995e-275, 0.0001, 2) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 863, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<863; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o8.channelCount = 3; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1821, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1821; i++) {buffer.getChannelData(c)[i] = 0.3655;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o6.connect(o6,0,0) } catch(e) { } +try { o1.listener.setOrientation(128, -0.479527496, 128, 1, 1, -16) } catch(e) { } +try { o4.connect(o8,0,0) } catch(e) { } +try { o7.reduction.exponentialRampToValueAtTime(1.7976931348623157e+308, 2048) } catch(e) { } +try { o1.listener.dopplerFactor = 0.5754; } catch(e) { } +try { o7.release.value = 0; } catch(e) { } +try { o5.getFloatFrequencyData(new Float32Array(2048)) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 120, 42419);for(var c=0; c<2; c++) {for(var i=0; i<120; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o3.buffer = function(){var buffer = o1.createBuffer(1, 1620, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<1620; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}(); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 823, 77239);for(var c=0; c<1; c++) {for(var i=0; i<823; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { } +try { o1.listener.setPosition(2048, 0.000522899209, -4503599627370495) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 84028, 75483);for(var c=0; c<3; c++) {for(var i=0; i<84028; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.playbackRate.setTargetAtTime(5e-324, 9.999999999999998e-104, 0) } catch(e) { } +try { o7.attack.value = 9.999999999999994e-236; } catch(e) { } +try { o1.listener.dopplerFactor = 1024; } catch(e) { } +try { o3.connect(o7,0,0) } catch(e) { } +try { o2.playbackRate.linearRampToValueAtTime(0, 0) } catch(e) { } +try { o4.connect(o10,0,0) } catch(e) { } +try { o1.listener.setVelocity(0.682, 32, 1e-76) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 201, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<201; i++) {buffer.getChannelData(c)[i] = 4;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.connect(o6,0,0) } catch(e) { } +try { o5.getByteFrequencyData(new Uint8Array(32)) } catch(e) { } +try { o3.connect(o6,0,0) } catch(e) { } +try { o7.channelCount = 1; } catch(e) { } +try { o2.playbackRate.setValueCurveAtTime(new Float32Array(512), 8, 1) } catch(e) { } +try { o3.connect(o4,0,1) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1996, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1996; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.listener.dopplerFactor = 2; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 983, 60517);for(var c=0; c<2; c++) {for(var i=0; i<983; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o6.coneInnerAngle = 360; } catch(e) { } +try { o8.curve = new Float32Array(512); } catch(e) { } +try { o1.listener.setPosition(32, 16, 9.999999999999995e-280) } catch(e) { } +try { o6.connect(o6,0,0) } catch(e) { } +try { o1.listener.setPosition(2097151, 512, 5e-324) } catch(e) { } +try { o7.channelCountMode = 'clamped-max'; } catch(e) { } +try { o1.destination.channelCountMode = 'max'; } catch(e) { } +try { o1.listener.setPosition(0, 1000000, 64) } catch(e) { } +try { o4.connect(o5,0,0) } catch(e) { } +try { o9.connect(o4,0,1) } catch(e) { } +try { o7.attack.setValueCurveAtTime(new Float32Array(8), 256, 2048) } catch(e) { } +try { o9.channelCountMode = 'explicit'; } catch(e) { } +try { o1.listener.dopplerFactor = 5e-324; } catch(e) { } +try { o4.connect(o10,0,0) } catch(e) { } +try { o7.ratio.cancelScheduledValues(1) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1590, 9733);for(var c=0; c<1; c++) {for(var i=0; i<1590; i++) {buffer.getChannelData(c)[i] = 2;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.destination.channelCount = 2; } catch(e) { } +try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { } +try { o5.channelCountMode = 'max'; } catch(e) { } +try { o9.channelCountMode = 'clamped-max'; } catch(e) { } +try { o7.attack.setTargetAtTime(0.11499421161482907549622467513472656719386577606201171875000, 1, 1.7976931348623157e+308) } catch(e) { } +try { o2.playbackRate.value = 1; } catch(e) { } +try { o3.normalize = false; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1866, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1866; i++) {buffer.getChannelData(c)[i] = 0.174908422905697580329587026426452212035655975341797;}}return buffer;}(); } catch(e) { } +try { o2.playbackRate.value = 1.0; } catch(e) { } +try { o7.threshold.value = 100; } catch(e) { } +try { o1.listener.setVelocity(961.4441892370145, 9.999999999999999e-157, 1) } catch(e) { } +try { o5.getByteFrequencyData(new Uint8Array(2)) } catch(e) { } +try { o6.connect(o10,0,0) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1176, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1176; i++) {buffer.getChannelData(c)[i] = 9.753993292300242;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.playbackRate.linearRampToValueAtTime(0.8687, 32) } catch(e) { } +try { o5.channelCountMode = 'explicit'; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(13, 703, 12723);for(var c=0; c<13; c++) {for(var i=0; i<703; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.loopEnd = 1.7976931348623157e+308; } catch(e) { } +try { o3.connect(o7,0,0) } catch(e) { } +try { o1.destination.channelCountMode = 'clamped-max'; } catch(e) { } +try { o1.destination.channelInterpretation = 'discrete'; } catch(e) { } +try { o2.connect(o10,0,0) } catch(e) { } +try { o3.normalize = false; } catch(e) { } +try { /* switch-case did not return anything. */ } catch(e) { } +try { o10.connect(o8,0,0) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 132, 88507);for(var c=0; c<3; c++) {for(var i=0; i<132; i++) {buffer.getChannelData(c)[i] = 1.7976931348623157e+308;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.loopStart = 1.7976931348623157e+308; } catch(e) { } +try { o1.listener.dopplerFactor = 1024; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 40, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<40; i++) {buffer.getChannelData(c)[i] = 1.3785770335346594;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o4.connect(o9,5,0) } catch(e) { } +try { o8.curve = new Float32Array(16); } catch(e) { } +try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { } +try { o6.channelCountMode = 'explicit'; } catch(e) { } +try { o2.playbackRate.value = 0; } catch(e) { } +try { o4.channelCount = 3; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 220, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<220; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.destination.channelInterpretation = 'discrete'; } catch(e) { } +try { o1.listener.setPosition(70368744177664, 2048, 1.7976931348623157e+308) } catch(e) { } +try { o2.playbackRate.setValueCurveAtTime(new Float32Array(128), 1.7976931348623157e+308, 0.75) } catch(e) { } +try { o5.fftSize = 118; } catch(e) { } +try { o5.minDecibels = 1; } catch(e) { } +try { o7.release.value = 5e-324; } catch(e) { } +try { o2.channelCount = 3; } catch(e) { } +try { o2.channelCountMode = 'clamped-max'; } catch(e) { } +try { o3.normalize = false; } catch(e) { } +try { o1.listener.speedOfSound = 2; } catch(e) { } +try { o2.loopStart = 32; } catch(e) { } +try { o3.channelCountMode = 'max'; } catch(e) { } +try { o1.listener.setPosition(0.37976588317653304, -274877906943, 64) } catch(e) { } +try { o8.curve = new Float32Array(4); } catch(e) { } +try { o1.listener.setVelocity(16, -1024, 0) } catch(e) { } +try { o5.minDecibels = 10000; } catch(e) { } +try { o2.playbackRate.value = 5e-324; } catch(e) { } +try { o3.connect(o9,1,0) } catch(e) { } +try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { } +try { o5.connect(o8,0,0) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(3, 554, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<554; i++) {buffer.getChannelData(c)[i] = 0.000001;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o7.release.value = 0; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1289, 29583);for(var c=0; c<1; c++) {for(var i=0; i<1289; i++) {buffer.getChannelData(c)[i] = 5e-324;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.listener.setOrientation(1024, 64, 16, 1024, 1, 68719476735) } catch(e) { } +try { o1.listener.setOrientation(512, 10000000, 274877906944, 9.999999999999998e-113, -8, 3.6191134923595274) } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1453, 15258);for(var c=0; c<1; c++) {for(var i=0; i<1453; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o8.channelInterpretation = 'discrete'; } catch(e) { } +try { o5.channelInterpretation = 'discrete'; } catch(e) { } +try { o7.connect(o9,4,0) } catch(e) { } +try { o8.connect(o10,0,0) } catch(e) { } +try { o1.listener.setPosition(18014398509481984, 16, 0.505129302503804056279079759406158700585365295410156250000) } catch(e) { } +try { o10.connect(o9,5,0) } catch(e) { } +try { o2.loop = false; } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(11, 1673, o1.sampleRate);for(var c=0; c<11; c++) {for(var i=0; i<1673; i++) {buffer.getChannelData(c)[i] = 0;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 577, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<577; i++) {buffer.getChannelData(c)[i] = 0.5611;}}return buffer;}() } catch(e) { } +try { o2.connect(o1.destination); } catch(e) { } +try { o1.listener.setOrientation(64, 2049, 5e-324, 0.1777, 2, 7) } catch(e) { } +try { o5.maxDecibels = 128; } catch(e) { } +try { o2.start(0) } catch(e) { } +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/880342-2.html b/dom/media/test/crashtests/880342-2.html new file mode 100644 index 0000000000..a8ef88ae70 --- /dev/null +++ b/dom/media/test/crashtests/880342-2.html @@ -0,0 +1,8 @@ +<script> +try { o1 = new window.AudioContext(2, 10, 1019159); } catch(e) { } +try { o2 = o1.createBufferSource(); } catch(e) { } +try { o3 = o1.createConvolver(); } catch(e) { } +try { o3.buffer = function(){var buffer = o1.createBuffer(2, 741, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<741; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); } catch(e) { } +try { o2.buffer = function(){var buffer = o1.createBuffer(2, 120, 42419);for(var c=0; c<2; c++) {for(var i=0; i<120; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { } +try { o3.buffer = function(){var buffer = o1.createBuffer(1, 1620, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<1620; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}(); } catch(e) { } +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/880384.html b/dom/media/test/crashtests/880384.html new file mode 100644 index 0000000000..798fbf133d --- /dev/null +++ b/dom/media/test/crashtests/880384.html @@ -0,0 +1,8 @@ +<script> +o1 = new window.AudioContext(3, 256, 44100); +o2 = o1.createBufferSource(); +o3 = o1.createConvolver(); +o3.normalize = false; +o3.buffer = function(){var buffer = o1.createBuffer(2, 1051, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1051; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); +o3.buffer = function(){var buffer = o1.createBuffer(2, 1112, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1112; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); +</script> diff --git a/dom/media/test/crashtests/880404.html b/dom/media/test/crashtests/880404.html new file mode 100644 index 0000000000..bf34932388 --- /dev/null +++ b/dom/media/test/crashtests/880404.html @@ -0,0 +1,6 @@ +<script> +o1 = new window.OfflineAudioContext(2, 32, 44100); +o12 = o1.createConvolver(); +o12.buffer = function(){var buffer = o1.createBuffer(1, 78, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<78; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); +o1.startRendering(); +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/880724.html b/dom/media/test/crashtests/880724.html new file mode 100644 index 0000000000..b57b5f9964 --- /dev/null +++ b/dom/media/test/crashtests/880724.html @@ -0,0 +1,13 @@ +<script> +o1 = new window.AudioContext(); +o5 = o1.createConvolver(); +o5.buffer = function() { + var buffer = o1.createBuffer(2, 3, 33120); + for(var c=0; c<2; c++) { + for(var i=0; i<3; i++) { + buffer.getChannelData(c)[i] = -1; + } + } + return buffer; +}(); +</script>
\ No newline at end of file diff --git a/dom/media/test/crashtests/881775.html b/dom/media/test/crashtests/881775.html new file mode 100644 index 0000000000..d55c45d17e --- /dev/null +++ b/dom/media/test/crashtests/881775.html @@ -0,0 +1,25 @@ +<script> +o1 = new window.AudioContext(2, 16, 44100); +o2 = o1.createBufferSource(); +o12 = o1.createBiquadFilter(); +o1.destination.channelCountMode = 'max'; +o2.buffer = function(){ + var buffer = o1.createBuffer(2, 326, 77632); + for(var c=0; c<2; c++) { + for(var i=0; i<326; i++) { + buffer.getChannelData(c)[i] = -1; + } + } + return buffer; +}(); +o2.connect(o1.destination); +o2.buffer = function(){ + var buffer = o1.createBuffer(3, 405, o1.sampleRate); + for(var c=0; c<3; c++) { + for(var i=0; i<405; i++) { + buffer.getChannelData(c)[i] = 1; + } + } + return buffer; +}(); +</script> diff --git a/dom/media/test/crashtests/882956.html b/dom/media/test/crashtests/882956.html new file mode 100644 index 0000000000..ae7b441f99 --- /dev/null +++ b/dom/media/test/crashtests/882956.html @@ -0,0 +1,15 @@ +<script> +o1 = new window.AudioContext(1, 2048, 44100); +o2 = o1.createBufferSource(); +o1.destination.channelCountMode = 'max'; +o2.connect(o1.destination); +o2.buffer = function(){ + var buffer = o1.createBuffer(30, 442, 94933); + for(var c=0; c<30; c++) { + for(var i=0; i<442; i++) { + buffer.getChannelData(c)[i] = 1; + } + } + return buffer; +}(); +</script> diff --git a/dom/media/test/crashtests/884459.html b/dom/media/test/crashtests/884459.html new file mode 100644 index 0000000000..e321d569f2 --- /dev/null +++ b/dom/media/test/crashtests/884459.html @@ -0,0 +1,12 @@ +<script> +var Context0= new window.OfflineAudioContext(14,191531,44100) +var BufferSource1=Context0.createBufferSource(); + +setInterval(function(){ +BufferSource1.playbackRate.setTargetAtTime(0xC8F461D3EE6B2,(Context0.currentTime+0.0677539280615747),0.826130285160616); +BufferSource1.playbackRate.setValueAtTime(35467.63924283907536607336193,0); +},1) + +Context0.startRendering(); + +</script> diff --git a/dom/media/test/crashtests/889042.html b/dom/media/test/crashtests/889042.html new file mode 100644 index 0000000000..9f74979c58 --- /dev/null +++ b/dom/media/test/crashtests/889042.html @@ -0,0 +1,4 @@ +<!DOCTYPE html> +<script> + (new AudioContext()).createConvolver().buffer = null; +</script> diff --git a/dom/media/test/crashtests/907986-1.html b/dom/media/test/crashtests/907986-1.html new file mode 100644 index 0000000000..320b8eadd0 --- /dev/null +++ b/dom/media/test/crashtests/907986-1.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 100, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +// zero front vector +context.listener.setOrientation(0, 0, 0, 6.311749985202524e+307, 0, 0); +var panner = context.createPanner(); +panner.setPosition(6.311749985202524e+307, 4, 6.311749985202524e+307); +panner.connect(context.destination); +var source = context.createOscillator(); +source.connect(panner); +source.start(0); +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/907986-2.html b/dom/media/test/crashtests/907986-2.html new file mode 100644 index 0000000000..e0626ba2c2 --- /dev/null +++ b/dom/media/test/crashtests/907986-2.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 100, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +// zero up vector +context.listener.setOrientation(0, 6.311749985202524e+307, 0, 0, 0, 0); +var panner = context.createPanner(); +panner.setPosition(1, 2, 3); +panner.connect(context.destination); +var source = context.createOscillator(); +source.connect(panner); +source.start(0); +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/907986-3.html b/dom/media/test/crashtests/907986-3.html new file mode 100644 index 0000000000..75b756c363 --- /dev/null +++ b/dom/media/test/crashtests/907986-3.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 100, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +// linearly dependent +context.listener.setOrientation(0, 0, -6.311749985202524e+307, 0, 0, 6.311749985202524e+307); +var panner = context.createPanner(); +panner.setPosition(1, 2, 3); +panner.connect(context.destination); +var source = context.createOscillator(); +source.connect(panner); +source.start(0); +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/907986-4.html b/dom/media/test/crashtests/907986-4.html new file mode 100644 index 0000000000..a73500efca --- /dev/null +++ b/dom/media/test/crashtests/907986-4.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 100, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +var panner = context.createPanner(); +panner.setPosition(0, 3, 0); // directly overhead +panner.connect(context.destination); +var source = context.createOscillator(); +source.connect(panner); +source.start(0); +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/910171-1.html b/dom/media/test/crashtests/910171-1.html new file mode 100644 index 0000000000..9f3ec831be --- /dev/null +++ b/dom/media/test/crashtests/910171-1.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 4096, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +var delay = context.createDelay(); +delay.connect(context.destination); +delay.delayTime.value = 1.0; +var buffer = context.createBuffer(1, 2048, context.sampleRate); +var source = context.createBufferSource(); +source.buffer = buffer; +source.connect(delay); +source.start(); +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/920987.html b/dom/media/test/crashtests/920987.html new file mode 100644 index 0000000000..6e8b2992a6 --- /dev/null +++ b/dom/media/test/crashtests/920987.html @@ -0,0 +1,6 @@ +<script> +var ctx = new AudioContext(); +var buffer = ctx.createBuffer(1, 1000, ctx.sampleRate); +var array = new Float32Array(900); +buffer.copyToChannel(array, 0, 0xfffffff8); +</script> diff --git a/dom/media/test/crashtests/925619-1.html b/dom/media/test/crashtests/925619-1.html new file mode 100644 index 0000000000..146c531f9b --- /dev/null +++ b/dom/media/test/crashtests/925619-1.html @@ -0,0 +1,14 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +// 1024 > 89478.5 * 48000 - (1 << 32) +var context = new window.OfflineAudioContext(1, 1024, 48000); +context.oncomplete = function(e) { + document.documentElement.removeAttribute("class"); +}; +var buffer = context.createBuffer(1, 2048, context.sampleRate); +var source = context.createBufferSource(); +source.buffer = buffer; +source.start(89478.5); // 89478.5 is a little greater than 2^32 / 48000. +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/925619-2.html b/dom/media/test/crashtests/925619-2.html new file mode 100644 index 0000000000..e734b8bcad --- /dev/null +++ b/dom/media/test/crashtests/925619-2.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 2048, 48000); +// 1024 > 89478.5 * 48000 - (1 << 32) +var buffer = context.createBuffer(1, 1024, context.sampleRate); +var source = context.createBufferSource(); +source.buffer = buffer; +source.onended = function(e) { + document.documentElement.removeAttribute("class"); +}; +source.start(0); +source.stop(89478.5); // 89478.5 is a little greater than 2^32 / 48000. +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/926619.html b/dom/media/test/crashtests/926619.html new file mode 100644 index 0000000000..2ead02af4e --- /dev/null +++ b/dom/media/test/crashtests/926619.html @@ -0,0 +1,24 @@ +<!DOCTYPE html> +<html> +<head> +<meta charset="UTF-8"> +<script> + +function boom() +{ + var ac = new window.AudioContext(); + + var _ChannelMergerNode = ac.createChannelMerger(4); + + var _MediaStreamAudioDestinationNode = ac.createMediaStreamDestination(); + var _MediaStream = _MediaStreamAudioDestinationNode.stream; + var _MediaStreamAudioSourceNode = ac.createMediaStreamSource(_MediaStream); + + _ChannelMergerNode.connect(_MediaStreamAudioDestinationNode, 0, 0); + _MediaStreamAudioSourceNode.connect(_ChannelMergerNode, 0, 0); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/933151.html b/dom/media/test/crashtests/933151.html new file mode 100644 index 0000000000..3d45f7af38 --- /dev/null +++ b/dom/media/test/crashtests/933151.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html> +<head> +<script> + +function boom() +{ + var ac = new window.AudioContext(); + var buffer = ac.createBuffer(1, 24313, 47250); + buffer.copyFromChannel(buffer.getChannelData(0), ''); +} + +</script> +</head> +<body onload="boom();"></body> +</html> diff --git a/dom/media/test/crashtests/933156.html b/dom/media/test/crashtests/933156.html new file mode 100644 index 0000000000..b89445a43d --- /dev/null +++ b/dom/media/test/crashtests/933156.html @@ -0,0 +1,23 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +function boom() +{ + var ac = new window.AudioContext(); + + var panner = ac.createPanner(); + panner.setPosition(8, 0.7643051305237005, 0.10292575673733972); + + var oscillator = ac.createOscillator(); + oscillator.connect(panner); + oscillator.start(0); + + setTimeout(function() { + panner.panningModel = 'equalpower'; + oscillator.stop(0); + document.documentElement.removeAttribute("class"); + }, 0.5); +} +boom(); +</script> +</html> diff --git a/dom/media/test/crashtests/944851.html b/dom/media/test/crashtests/944851.html new file mode 100644 index 0000000000..4f663accc4 --- /dev/null +++ b/dom/media/test/crashtests/944851.html @@ -0,0 +1,17 @@ +<!DOCTYPE html> +<html> +<head> +<meta charset="UTF-8"> +<script> + +var ac = new AudioContext(1, 1354, 44100); +var shaper = ac.createWaveShaper(); +var biquad = ac.createBiquadFilter(); +shaper.connect(biquad.frequency); +biquad.getFrequencyResponse(new Float32Array(55785), + new Float32Array(62876), + new Float32Array(45111)); + +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/952756.html b/dom/media/test/crashtests/952756.html new file mode 100644 index 0000000000..ffced2e400 --- /dev/null +++ b/dom/media/test/crashtests/952756.html @@ -0,0 +1,19 @@ +<script> +var Context0= new AudioContext() +var BufferSource0=Context0.createBufferSource(); +BufferSource0.buffer=function(){ + var length=35887; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0.39765272522345185))}; + return Buffer; +}(); +BufferSource0.start(0.01932738965842873,0.33345631847623736,0.3893404237460345); +BufferSource0.buffer=function(){ + var length=15952; + var Buffer=Context0.createBuffer(1,length,Context0.sampleRate); + var bufferData= Buffer.getChannelData(0); + for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(85))}; + return Buffer; +}(); +</script> diff --git a/dom/media/test/crashtests/986901.html b/dom/media/test/crashtests/986901.html new file mode 100644 index 0000000000..343df2c0ed --- /dev/null +++ b/dom/media/test/crashtests/986901.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html> +<head> +<meta charset="UTF-8"> +<script> + var ac = new window.AudioContext(); + var delay1 = ac.createDelay(0.02); + var delay2 = ac.createDelay(0.002); + var source = ac.createOscillator(); + source.start(0); + source.connect(delay1, 0, 0); + delay2.connect(delay1, 0, 0); + delay1.connect(delay2, 0, 0); +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/990794.html b/dom/media/test/crashtests/990794.html new file mode 100644 index 0000000000..8b40088b1a --- /dev/null +++ b/dom/media/test/crashtests/990794.html @@ -0,0 +1,22 @@ +<!DOCTYPE html> +<script> +var ctx = new AudioContext(); +var source = ctx.createOscillator(); +source.start(0); + +function appendMerger(src) { + const inputCount = 18; + + var merger = ctx.createChannelMerger(32); + + for (var i = 0; i < inputCount; ++i) { + src.connect(merger, 0, i); + } + + return merger; +} + +for (var i = 0; i < 6; ++i) { + source = appendMerger(source); +} +</script> diff --git a/dom/media/test/crashtests/995289.html b/dom/media/test/crashtests/995289.html new file mode 100644 index 0000000000..c988f41fa8 --- /dev/null +++ b/dom/media/test/crashtests/995289.html @@ -0,0 +1,9 @@ +<!DOCTYPE html> +<script> +var r0=new AudioContext(); +var r5=r0.createOscillator(); +var r6=r0.createPeriodicWave(new Float32Array(1),new Float32Array(1)); +r5.frequency.value = 4294967295; +r5.start(0); +r5.setPeriodicWave(r6); +</script> diff --git a/dom/media/test/crashtests/analyser-channels-1.html b/dom/media/test/crashtests/analyser-channels-1.html new file mode 100644 index 0000000000..2f3133cf13 --- /dev/null +++ b/dom/media/test/crashtests/analyser-channels-1.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> + var context = new window.OfflineAudioContext(1, 256, 48000); + var analyser = context.createAnalyser(); + analyser.channelCount = 2; + analyser.channelCountMode = "explicit"; + analyser.fftSize = 32; + var source = context.createOscillator(); + source.connect(analyser); + source.start(0); + context.startRendering(). + then(function() { + document.documentElement.removeAttribute("class"); + }); +</script> diff --git a/dom/media/test/crashtests/audiocontext-after-unload-1.html b/dom/media/test/crashtests/audiocontext-after-unload-1.html new file mode 100644 index 0000000000..9b4f1181d2 --- /dev/null +++ b/dom/media/test/crashtests/audiocontext-after-unload-1.html @@ -0,0 +1,27 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<title>Test for bug 1646601</title> +<script> +document.addEventListener('DOMContentLoaded', async () => { + const frame = document.createElement('iframe'); + document.body.appendChild(frame); + frame.srcdoc = '<html></html>'; + await new Promise(resolve => frame.onload = resolve); + const subwin = frame.contentWindow; + const subcontext = subwin.AudioContext; + // Construct an AudioContext while the subdocument is fully active to start + // a MediaTrackGraph. + new subcontext(); + // Unload the subdocument and wait for completion. + // This shuts down the MediaTrackGraph. + subwin.location.reload(); + await new Promise(resolve => frame.onload = resolve); + // Test that a new AudioContext on the inactive subdocument does not attempt + // to use the shut-down MediaTrackGraph. + try { new subcontext() } catch {} + document.documentElement.removeAttribute('class'); +}); +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/audiocontext-double-suspend.html b/dom/media/test/crashtests/audiocontext-double-suspend.html new file mode 100644 index 0000000000..98399549bd --- /dev/null +++ b/dom/media/test/crashtests/audiocontext-double-suspend.html @@ -0,0 +1,5 @@ +<script> +var ac = new AudioContext(); +ac.resume(); +ac.resume(); +</script> diff --git a/dom/media/test/crashtests/audioworkletnode-after-unload-1.html b/dom/media/test/crashtests/audioworkletnode-after-unload-1.html new file mode 100644 index 0000000000..7da8d1161a --- /dev/null +++ b/dom/media/test/crashtests/audioworkletnode-after-unload-1.html @@ -0,0 +1,27 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> +<title>Test for bug 1634200 and bug 1655544</title> +<script> +document.addEventListener('DOMContentLoaded', async () => { + const frame = document.createElement('iframe'); + document.body.appendChild(frame); + frame.srcdoc = '<html></html>'; + await new Promise(resolve => frame.onload = resolve); + + const subwin = frame.contentWindow; + const ctx = new subwin.AudioContext(); + const url = URL.createObjectURL( + new Blob([`registerProcessor("noop", + class extends AudioWorkletProcessor {})`]), + {type: "application/javascript"}); + await ctx.audioWorklet.addModule(url); + + frame.remove(); + new subwin.AudioWorkletNode(ctx, 'noop') + + document.documentElement.removeAttribute('class'); +}); +</script> +</head> +</html> diff --git a/dom/media/test/crashtests/buffer-source-duration-1.html b/dom/media/test/crashtests/buffer-source-duration-1.html new file mode 100644 index 0000000000..df8d7a37d5 --- /dev/null +++ b/dom/media/test/crashtests/buffer-source-duration-1.html @@ -0,0 +1,14 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +const rate = 44100; +var context = new window.OfflineAudioContext(1, 512, rate); +var buffer = context.createBuffer(1, 128, rate); +var source = context.createBufferSource(); +source.buffer = buffer; +source.start(0, 0, 86400); +context.startRendering(). + then(function() { + document.documentElement.removeAttribute("class"); + }); +</script> diff --git a/dom/media/test/crashtests/buffer-source-ended-1.html b/dom/media/test/crashtests/buffer-source-ended-1.html new file mode 100644 index 0000000000..de8546316c --- /dev/null +++ b/dom/media/test/crashtests/buffer-source-ended-1.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new AudioContext(); + +var source = context.createBufferSource(); +source.buffer = context.createBuffer(1, 2.0 * context.sampleRate, context.sampleRate); +source.onended = function(e) { + document.documentElement.removeAttribute("class"); +} +source.start(0.0, 1.0); +setTimeout( + function() { + source.buffer = context.createBuffer(1, 1, context.sampleRate); + }, 0); +</script> diff --git a/dom/media/test/crashtests/buffer-source-resampling-start-1.html b/dom/media/test/crashtests/buffer-source-resampling-start-1.html new file mode 100644 index 0000000000..55db8591ed --- /dev/null +++ b/dom/media/test/crashtests/buffer-source-resampling-start-1.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +const rate = 44101; // not divisible by 2 +var context = new window.OfflineAudioContext(1, 512, rate); +var buffer = context.createBuffer(1, 128, rate); +buffer.getChannelData(0)[0] = 1.0; +var source = context.createBufferSource(); +source.buffer = buffer; +source.playbackRate.value = rate / (Math.pow(2, 30) * 1.0000001); +source.start(512 / rate); +context.startRendering(). + then(function() { + document.documentElement.removeAttribute("class"); + }); +</script> diff --git a/dom/media/test/crashtests/buffer-source-slow-resampling-1.html b/dom/media/test/crashtests/buffer-source-slow-resampling-1.html new file mode 100644 index 0000000000..5d8a50442b --- /dev/null +++ b/dom/media/test/crashtests/buffer-source-slow-resampling-1.html @@ -0,0 +1,34 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +const blockSize = 128; +// The sample rate is a prime number so that the resampler is not expected to +// simplify in/out fractions. +const rate = 44101; +var context = new window.OfflineAudioContext(1, 3 * blockSize, rate); +// Non-zero buffer, so it can't be optimized away. +var buffer = context.createBuffer(1, 128, rate); +buffer.getChannelData(0)[0] = 1.0; +var source = context.createBufferSource(); +source.buffer = buffer; +source.loop = true; +// Initialize the resampler with a slow input rate. +// With the current (Mar 2017) implementation, very slow rates give the +// resampler a very large denominator. +source.playbackRate.setValueAtTime(rate / 0x7fffffff, 0.0); +// Change to a moderate input rate. +// With the current implementation, skip_frac_num increases by den_rate for +// each output sample and so one block before the change in playback rate is +// enough for high skip_frac_num at the time of the change. +const changeBlock = 1; +const changeBlockSeconds = changeBlock * blockSize / rate; +// With the current speex_resampler_set_rate_frac() implementation, the +// moderate resampler denominator is still large enough to trigger overflow of +// 32-bit unsigned integer arithmetic. +source.playbackRate.setValueAtTime(rate / (rate + 1), changeBlockSeconds); +source.start(0); +context.startRendering(). + then(function() { + document.documentElement.removeAttribute("class"); + }); +</script> diff --git a/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4 b/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4 Binary files differnew file mode 100644 index 0000000000..92bf3722f2 --- /dev/null +++ b/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4 diff --git a/dom/media/test/crashtests/convolver-memory-report-1.html b/dom/media/test/crashtests/convolver-memory-report-1.html new file mode 100644 index 0000000000..a49a281d1c --- /dev/null +++ b/dom/media/test/crashtests/convolver-memory-report-1.html @@ -0,0 +1,25 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> + <title>Bug 1481745: Exercise ConvolverNode memory reporting</title> + <script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script> + <script> +let context = new AudioContext(); +let response = new AudioBuffer({length: 128, + sampleRate: context.sampleRate}); +response.getChannelData(0)[response.length - 1] = 1; +let convolver = new ConvolverNode(context, + {disableNormalization: true, + buffer: response}); +convolver.connect(context.destination); +let osc = new OscillatorNode(context); +osc.connect(convolver); +osc.start(); +osc.stop(128/context.sampleRate); +osc.onended = (e) => { + SpecialPowers.getMemoryReports(); + document.documentElement.removeAttribute("class"); +}; + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/copyFromChannel-2.html b/dom/media/test/crashtests/copyFromChannel-2.html new file mode 100644 index 0000000000..8d3d5a2124 --- /dev/null +++ b/dom/media/test/crashtests/copyFromChannel-2.html @@ -0,0 +1,16 @@ +<!DOCTYPE html> +<html> +<head> + <title>Crashtest for bug 1548816</title> + <script> +let cx = new OfflineAudioContext({numberOfChannels: 1, + length: 1, sampleRate: 44100}); +let buffer = new AudioBuffer({numberOfChannels: 13, + length: 22050, sampleRate: 44100}); +buffer.getChannelData(12)[0] = 1.0; +let o2248 = new AudioBufferSourceNode(cx, {buffer: buffer}); +let array = new Float32Array(52428); +buffer.copyFromChannel(array, 12); + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/cors.webm b/dom/media/test/crashtests/cors.webm Binary files differnew file mode 100644 index 0000000000..72b0297233 --- /dev/null +++ b/dom/media/test/crashtests/cors.webm diff --git a/dom/media/test/crashtests/cors.webm^headers^ b/dom/media/test/crashtests/cors.webm^headers^ new file mode 100644 index 0000000000..cb762eff80 --- /dev/null +++ b/dom/media/test/crashtests/cors.webm^headers^ @@ -0,0 +1 @@ +Access-Control-Allow-Origin: * diff --git a/dom/media/test/crashtests/crashtests.list b/dom/media/test/crashtests/crashtests.list new file mode 100644 index 0000000000..f3e5683733 --- /dev/null +++ b/dom/media/test/crashtests/crashtests.list @@ -0,0 +1,142 @@ +load 0-timescale.html # bug 1229166 +skip-if(Android) pref(media.autoplay.default,0) load 459439-1.html # bug 888557 +load 466607-1.html +load 466945-1.html +load 468763-1.html +load 474744-1.html +HTTP load 481136-1.html # needs to be HTTP to recognize the ogg as an audio file? +load 492286-1.xhtml +load 493915-1.html +pref(media.autoplay.default,0) load 495794-1.html +load 497734-1.xhtml +load 497734-2.html +load 576612-1.html +load 752784-1.html +skip-if(Android) load 789075-1.html # bug 1374405 for android +skip-if(Android&&AndroidVersion=='22') HTTP load 795892-1.html # bug 1358718 +load 844563.html +load 846612.html +skip-if(verify&&isDebugBuild&>kWidget) load 852838.html +load 865004.html +load 865537-1.html +load 865550.html +skip-if(verify&&isDebugBuild&>kWidget) load 868504.html +skip-if(verify&&isDebugBuild&>kWidget) load 874869.html +skip-if(verify&&isDebugBuild&>kWidget) load 874915.html +skip-if(verify&&isDebugBuild&>kWidget) load 874934.html +skip-if(verify&&isDebugBuild&>kWidget) load 874952.html +skip-if(verify&&isDebugBuild&>kWidget) load 875144.html +skip-if(verify&&isDebugBuild&>kWidget) load 875596.html +load 875911.html +load 876024-1.html +skip-if(verify&&isDebugBuild&>kWidget) load 876024-2.html +skip-if(verify&&isDebugBuild&>kWidget) load 876118.html +load 876207.html +load 876215.html +skip-if(verify&&isDebugBuild&>kWidget) load 876249.html +skip-if(verify&&isDebugBuild&>kWidget) load 876252.html +load 876834.html +load 877527.html +load 877820.html +load 878014.html +load 878328.html +skip-if(verify&&isDebugBuild&>kWidget) load 878407.html +skip-if(verify&&isDebugBuild&>kWidget) load 878478.html +load 880129.html +load 880202.html +load 880342-1.html +load 880342-2.html +load 880384.html +load 880404.html +skip-if(verify&&isDebugBuild&>kWidget) load 880724.html +load 881775.html +load 882956.html +load 884459.html +skip-if(verify&&isDebugBuild&>kWidget) load 889042.html +load 907986-1.html +load 907986-2.html +load 907986-3.html +load 907986-4.html +load 910171-1.html +skip-if(verify&&isDebugBuild&>kWidget) load 920987.html +load 925619-1.html +load 925619-2.html +skip-if(verify&&isDebugBuild&>kWidget) load 926619.html +skip-if(verify&&isDebugBuild&>kWidget) load 933151.html +skip-if(verify&&isDebugBuild&>kWidget) load 933156.html +load 944851.html +skip-if(verify&&isDebugBuild&>kWidget) load 952756.html +skip-if(verify&&isDebugBuild&>kWidget) load 986901.html +skip-if(verify&&isDebugBuild&>kWidget) load 990794.html +skip-if(verify&&isDebugBuild&>kWidget) load 995289.html +skip-if(verify&&isDebugBuild&>kWidget) load 1012609.html +load 1015662.html +skip-if(Android) test-pref(media.navigator.permission.disabled,true) load 1028458.html # bug 1048863 +skip-if(verify&&isDebugBuild&>kWidget) load 1041466.html +skip-if(verify&&isDebugBuild&>kWidget) load 1045650.html +load 1080986.html +skip-if(Android&&AndroidVersion=='21') load 1180881.html # bug 1409365 +load 1197935.html +skip-if(verify&&isDebugBuild&>kWidget) load 1122218.html +load 1127188.html +skip-if(verify&&isDebugBuild&>kWidget) load 1157994.html +load 1158427.html +skip-if(verify&&isDebugBuild&>kWidget) load 1185176.html +load 1185192.html +skip-if(Android) load 1257700.html # bug 1575666 +load 1267263.html +load 1270303.html +skip-if(verify&&isDebugBuild&>kWidget) load 1368490.html +skip-if(verify&&isDebugBuild&>kWidget) load 1291702.html +load 1378826.html +load 1384248.html +load 1389304.html +load 1393272.webm +load 1411322.html +load 1450845.html +load 1489160.html +load disconnect-wrong-destination.html +load analyser-channels-1.html +load audiocontext-after-unload-1.html +skip-if(verify&&isDebugBuild&>kWidget) load audiocontext-double-suspend.html +skip-if(Android) load audioworkletnode-after-unload-1.html # Needs secure context +load buffer-source-duration-1.html +skip-if(verify&&isDebugBuild&>kWidget) load buffer-source-ended-1.html +load buffer-source-resampling-start-1.html +load buffer-source-slow-resampling-1.html +load convolver-memory-report-1.html +load copyFromChannel-2.html +load empty-buffer-source.html +skip-if(verify&&isDebugBuild&>kWidget) HTTP load media-element-source-seek-1.html +skip-if(verify&&isDebugBuild&>kWidget) load offline-buffer-source-ended-1.html +load oscillator-ended-1.html +load oscillator-ended-2.html +skip-if(Android&&AndroidVersion=='22') load video-replay-after-audio-end.html # bug 1315125, bug 1358876 +# This needs to run at the end to avoid leaking busted state into other tests. +skip-if(Android) load 691096-1.html # Bug 1365451 +load 1236639.html +test-pref(media.navigator.permission.disabled,true) test-pref(media.devices.insecure.enabled,true) test-pref(media.getusermedia.insecure.enabled,true) load 1388372.html +load 1494073.html +skip-if(Android) load 1526044.html # Bug 1528391 +skip-if(Android&&AndroidVersion<21) load encrypted-track-with-bad-sample-description-index.mp4 # Bug 1533211, unkip after bug 1550912 +load encrypted-track-without-tenc.mp4 # Bug 1533215 +asserts-if(Android,0-1) load encrypted-track-with-sample-missing-cenc-aux.mp4 # Bug 1533625, bug 1588967 +load 1538727.html +load empty-samples.webm # Bug 1540580 +test-pref(media.autoplay.block-webaudio,false) load 1545133.html +load track-with-zero-dimensions.mp4 # Bug 1542539 +load 1560215.html +skip-if(Android) load 1547784.html # Skip on Android as clearkey is not supported +load 1547899.html +load 1569645.html +load 1575271.html +load 1577184.html +pref(media.autoplay.default,0) load 1587248.html +load 1594466.html +load 1601385.html +load 1601422.html +load 1604941.html +pref(media.autoplay.default,0) load 1673525.html +skip-if(!winWidget) load 1608286.html +load channel-count-in-metadata-different-than-in-content.mp4 # Bug 1584959 +load mp4_box_emptyrange.mp4 # Bug 1667480 diff --git a/dom/media/test/crashtests/disconnect-wrong-destination.html b/dom/media/test/crashtests/disconnect-wrong-destination.html new file mode 100644 index 0000000000..515ca3c877 --- /dev/null +++ b/dom/media/test/crashtests/disconnect-wrong-destination.html @@ -0,0 +1,13 @@ +<script> + var oc = new OfflineAudioContext(1, 1, 44100); + var splitter = oc.createChannelSplitter(2); + var merger0 = oc.createChannelMerger(2); + var merger1 = oc.createChannelMerger(2); + splitter.connect(merger0, 0); + splitter.connect(merger0, 1); + splitter.connect(merger1, 0); + splitter.connect(merger1, 1); + + splitter.disconnect(merger0, 0); + splitter.disconnect(merger1, 1); +</script> diff --git a/dom/media/test/crashtests/doppler-1.html b/dom/media/test/crashtests/doppler-1.html new file mode 100644 index 0000000000..2af3c8f460 --- /dev/null +++ b/dom/media/test/crashtests/doppler-1.html @@ -0,0 +1,23 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.AudioContext(); +var source = context.createBufferSource(); +source.buffer = context.createBuffer(1, 1, context.sampleRate); +source.onended = + function(e) { + setTimeout( + function() { + var panner = context.createPanner(); + source.connect(panner); + panner.setVelocity(1.0, 0.0, 0.0); + setTimeout( + function() { + document.documentElement.removeAttribute("class"); + }, + 0); + }, + 0); + }; +source.start(0); +</script> diff --git a/dom/media/test/crashtests/empty-buffer-source.html b/dom/media/test/crashtests/empty-buffer-source.html new file mode 100644 index 0000000000..2ce48a9ec1 --- /dev/null +++ b/dom/media/test/crashtests/empty-buffer-source.html @@ -0,0 +1,14 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<head> + <title>Bug 1636540: AudioBufferSourceNode with empty buffer</title> + <script> +const offline = new OfflineAudioContext({length: 128, sampleRate: 16384}); +const buffer = new AudioBuffer({length: 1, sampleRate: 21725}); +const node = new AudioBufferSourceNode(offline, {buffer: buffer}); +node.start(5/offline.sampleRate); +offline.startRendering().then( + () => document.documentElement.removeAttribute("class")); + </script> +</head> +</html> diff --git a/dom/media/test/crashtests/empty-samples.webm b/dom/media/test/crashtests/empty-samples.webm new file mode 100644 index 0000000000..e69de29bb2 --- /dev/null +++ b/dom/media/test/crashtests/empty-samples.webm diff --git a/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4 b/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4 Binary files differnew file mode 100644 index 0000000000..32303f0357 --- /dev/null +++ b/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4 diff --git a/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4 b/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4 Binary files differnew file mode 100644 index 0000000000..875c5dca76 --- /dev/null +++ b/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4 diff --git a/dom/media/test/crashtests/encrypted-track-without-tenc.mp4 b/dom/media/test/crashtests/encrypted-track-without-tenc.mp4 Binary files differnew file mode 100644 index 0000000000..188faebf1b --- /dev/null +++ b/dom/media/test/crashtests/encrypted-track-without-tenc.mp4 diff --git a/dom/media/test/crashtests/media-element-source-seek-1.html b/dom/media/test/crashtests/media-element-source-seek-1.html new file mode 100644 index 0000000000..5c3aed5ae7 --- /dev/null +++ b/dom/media/test/crashtests/media-element-source-seek-1.html @@ -0,0 +1,27 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var audioElement = document.createElement("audio"); +audioElement.autoplay = true; +audioElement.src = "sound.ogg"; +audioElement.onplaying = + function() { + audioElement.onplaying = null; + setTimeout( + function() { + audioElement.onseeked = + function() { + // Note we reset 'src' to release decoder resources and cubeb + // streams to prevent OOM or OpenCubeb() failures. + audioElement.src = ""; + document.documentElement.removeAttribute("class"); + }; + audioElement.currentTime = 0; + }, 100); + }; + +var context = new window.AudioContext(); +var source = context.createMediaElementSource(audioElement); +source.connect(context.destination); +</script> +</html> diff --git a/dom/media/test/crashtests/mp4_box_emptyrange.mp4 b/dom/media/test/crashtests/mp4_box_emptyrange.mp4 Binary files differnew file mode 100644 index 0000000000..83057533a0 --- /dev/null +++ b/dom/media/test/crashtests/mp4_box_emptyrange.mp4 diff --git a/dom/media/test/crashtests/offline-buffer-source-ended-1.html b/dom/media/test/crashtests/offline-buffer-source-ended-1.html new file mode 100644 index 0000000000..0631021126 --- /dev/null +++ b/dom/media/test/crashtests/offline-buffer-source-ended-1.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +var context = new window.OfflineAudioContext(1, 12001, 12000); + +var source = context.createBufferSource(); +source.buffer = context.createBuffer(1, 12000, context.sampleRate); +source.onended = function(e) { + document.documentElement.removeAttribute("class"); +} +source.connect(context.destination); +source.start(0); + +context.startRendering(); +</script> diff --git a/dom/media/test/crashtests/oscillator-ended-1.html b/dom/media/test/crashtests/oscillator-ended-1.html new file mode 100644 index 0000000000..831111261c --- /dev/null +++ b/dom/media/test/crashtests/oscillator-ended-1.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +function createContext() { + var context = new window.AudioContext(); + var source = context.createOscillator(); + source.onended = function(e) { + document.documentElement.removeAttribute("class"); + }; + source.connect(context.destination); + source.start(0.49); + source.stop(0.5); +} +createContext(); +</script> diff --git a/dom/media/test/crashtests/oscillator-ended-2.html b/dom/media/test/crashtests/oscillator-ended-2.html new file mode 100644 index 0000000000..ee9b8cf300 --- /dev/null +++ b/dom/media/test/crashtests/oscillator-ended-2.html @@ -0,0 +1,15 @@ +<!DOCTYPE html> +<html class="reftest-wait"> +<script> +function createContext() { + var context = new window.AudioContext(); + var source = context.createOscillator(); + source.onended = function(e) { + document.documentElement.removeAttribute("class"); + }; + source.connect(context.destination); + source.start(60); + source.stop(0.5); +} +createContext(); +</script> diff --git a/dom/media/test/crashtests/sound.ogg b/dom/media/test/crashtests/sound.ogg Binary files differnew file mode 100644 index 0000000000..edda4e9128 --- /dev/null +++ b/dom/media/test/crashtests/sound.ogg diff --git a/dom/media/test/crashtests/track-with-zero-dimensions.mp4 b/dom/media/test/crashtests/track-with-zero-dimensions.mp4 Binary files differnew file mode 100644 index 0000000000..3f4a1317f3 --- /dev/null +++ b/dom/media/test/crashtests/track-with-zero-dimensions.mp4 diff --git a/dom/media/test/crashtests/video-crash.webm b/dom/media/test/crashtests/video-crash.webm Binary files differnew file mode 100644 index 0000000000..9532113d87 --- /dev/null +++ b/dom/media/test/crashtests/video-crash.webm diff --git a/dom/media/test/crashtests/video-replay-after-audio-end.html b/dom/media/test/crashtests/video-replay-after-audio-end.html new file mode 100644 index 0000000000..9ffd6078de --- /dev/null +++ b/dom/media/test/crashtests/video-replay-after-audio-end.html @@ -0,0 +1,43 @@ +<html class="reftest-wait"> +<head> + <title> Bug 1242774 : video crashed if pause and play again after audio track ends </title> +</head> +<body> +<script type="text/javascript"> +function assert(value, msg) { + if (!value) { + dump("### Error : " + msg + "\n"); + } +} + +var AUDIO_END_TIME = 4.5; +var video = document.createElement('video'); +video.src = "video-crash.webm"; +video.play(); + +video.ontimeupdate = function () { + assert(AUDIO_END_TIME < video.duration, + "AUDIO_END_TIME should be smaller than the duration!"); + + if (video.currentTime > AUDIO_END_TIME) { + dump("### Pause video during silent part.\n"); + video.ontimeupdate = null; + video.pause(); + } + + video.onpause = function () { + video.onpause = null; + setTimeout(function() { + dump("### Re-play after pausing during silent part.\n"); + video.play(); + video.onended = function () { + video.onended = null; + dump("### Video is ended.\n"); + document.documentElement.removeAttribute("class"); + } + }, 1000); + } +} +</script> +</body> +</html>
\ No newline at end of file diff --git a/dom/media/test/dash/dash-manifest-garbled-webm.mpd b/dom/media/test/dash/dash-manifest-garbled-webm.mpd new file mode 100644 index 0000000000..aa78ded3ec --- /dev/null +++ b/dom/media/test/dash/dash-manifest-garbled-webm.mpd @@ -0,0 +1,35 @@ +<?xml version="1.0" encoding="UTF-8"?> +<MPD + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns="urn:mpeg:DASH:schema:MPD:2011" + xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011" + type="static" + mediaPresentationDuration="PT3.958S" + minBufferTime="PT1S" + profiles="urn:webm:dash:profile:webm-on-demand:2012"> + <BaseURL>./</BaseURL> + <Period id="0" start="PT0S" duration="PT3.958S" > + <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true"> + <Representation id="0" bandwidth="54207" width="320" height="180"> + <BaseURL>garbled.webm</BaseURL> + <SegmentBase indexRange="35090-35123"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + <Representation id="1" bandwidth="78006" width="428" height="240"> + <BaseURL>dash-webm-video-428x240.webm</BaseURL> + <SegmentBase indexRange="50173-50206"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + </AdaptationSet> + <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1"> + <Representation id="2" bandwidth="57264"> + <BaseURL>dash-webm-audio-128k.webm</BaseURL> + <SegmentBase indexRange="41927-41946"> + <Initialization range="0-4521" /> + </SegmentBase> + </Representation> + </AdaptationSet> + </Period> +</MPD> diff --git a/dom/media/test/dash/dash-manifest-garbled.mpd b/dom/media/test/dash/dash-manifest-garbled.mpd new file mode 100644 index 0000000000..ac8eadbddc --- /dev/null +++ b/dom/media/test/dash/dash-manifest-garbled.mpd @@ -0,0 +1 @@ +PD94bWwgdmVyc2lvbj0iMS4wIiA/PjxNUEQgbWVkaWFQcmVzZW50YXRpb25EdXJhdGlvbj0iUFQxOS41MVMiIG1pbkJ1ZmZlclRpbWU9IlBUMVMiIHByb2ZpbGVzPSJ1cm46d2VibTpkYXNoOnByb2ZpbGU6d2VibS1vbi1kZW1hbmQ6MjAxMiIgdHlwZT0ic3RhdGljIiB4bWxucz0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiIHhtbG5zOnhzaT0iaHR0cDovL3d3dy53My5vcmcvMjAwMS9YTUxTY2hlbWEtaW5zdGFuY2UiIHhzaTpzY2hlbWFMb2NhdGlvbj0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiPjxCYXNlVVJMPmh0dHA6Ly93d3cuZ29vZ2xlLmNvbTwvQmFzZVVSTD48UGVyaW9kIGR1cmF0aW9uPSJQVDE5LjUxUyIgaWQ9IjAiIHN0YXJ0PSJQVDBTIj48QWRhcHRhdGlvblNldCBhdWRpb1NhbXBsaW5nUmF0ZT0iNDgwMDAiIGNvZGVjcz0idm9yYmlzIiBpZD0iMSIgbGFuZz0iZW5nIiBtaW1lVHlwZT0iYXVkaW8vd2VibSIgc3Vic2VnbWVudFN0YXJ0c1dpdGhTQVA9IjEiPjxSZXByZXNlbnRhdGlvbiBiYW5kd2lkdGg9IjIwMTA5IiBpZD0iMiI+PEJhc2VVUkwvPjxTZWdtZW50QmFzZSBpbmRleFJhbmdlPSIzMTk3ODAtMzIwNjEyIj48SW5pdGlhbGl6YXRpb24gcmFuZ2U9IjAtMjA4NzAiLz48L1NlZ21lbnRCYXNlPjwvUmVwcmVzZW50YXRpb24+PC9BZGFwdGF0aW9uU2V0PjwvUGVyaW9kPjwvTVBEPg diff --git a/dom/media/test/dash/dash-manifest-sjs.mpd b/dom/media/test/dash/dash-manifest-sjs.mpd new file mode 100644 index 0000000000..c7ecba3c69 --- /dev/null +++ b/dom/media/test/dash/dash-manifest-sjs.mpd @@ -0,0 +1,35 @@ +<?xml version="1.0" encoding="UTF-8"?> +<MPD + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns="urn:mpeg:DASH:schema:MPD:2011" + xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011" + type="static" + mediaPresentationDuration="PT3.958S" + minBufferTime="PT1S" + profiles="urn:webm:dash:profile:webm-on-demand:2012"> + <BaseURL>./dash_detect_stream_switch.sjs?name=</BaseURL> + <Period id="0" start="PT0S" duration="PT3.958S" > + <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true"> + <Representation id="0" bandwidth="54207" width="320" height="180"> + <BaseURL>dash-webm-video-320x180.webm</BaseURL> + <SegmentBase indexRange="35090-35123"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + <Representation id="1" bandwidth="78006" width="428" height="240"> + <BaseURL>dash-webm-video-428x240.webm</BaseURL> + <SegmentBase indexRange="50173-50206"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + </AdaptationSet> + <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1"> + <Representation id="2" bandwidth="57264"> + <BaseURL>dash-webm-audio-128k.webm</BaseURL> + <SegmentBase indexRange="41927-41946"> + <Initialization range="0-4521" /> + </SegmentBase> + </Representation> + </AdaptationSet> + </Period> +</MPD> diff --git a/dom/media/test/dash/dash-manifest.mpd b/dom/media/test/dash/dash-manifest.mpd new file mode 100644 index 0000000000..98c7a90480 --- /dev/null +++ b/dom/media/test/dash/dash-manifest.mpd @@ -0,0 +1,35 @@ +<?xml version="1.0" encoding="UTF-8"?> +<MPD + xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" + xmlns="urn:mpeg:DASH:schema:MPD:2011" + xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011" + type="static" + mediaPresentationDuration="PT3.958S" + minBufferTime="PT1S" + profiles="urn:webm:dash:profile:webm-on-demand:2012"> + <BaseURL>./</BaseURL> + <Period id="0" start="PT0S" duration="PT3.958S" > + <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true"> + <Representation id="0" bandwidth="54207" width="320" height="180"> + <BaseURL>dash-webm-video-320x180.webm</BaseURL> + <SegmentBase indexRange="35090-35123"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + <Representation id="1" bandwidth="78006" width="428" height="240"> + <BaseURL>dash-webm-video-428x240.webm</BaseURL> + <SegmentBase indexRange="50173-50206"> + <Initialization range="0-228" /> + </SegmentBase> + </Representation> + </AdaptationSet> + <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1"> + <Representation id="2" bandwidth="57264"> + <BaseURL>dash-webm-audio-128k.webm</BaseURL> + <SegmentBase indexRange="41927-41946"> + <Initialization range="0-4521" /> + </SegmentBase> + </Representation> + </AdaptationSet> + </Period> +</MPD> diff --git a/dom/media/test/dash/dash-webm-audio-128k.webm b/dom/media/test/dash/dash-webm-audio-128k.webm Binary files differnew file mode 100644 index 0000000000..f56c042053 --- /dev/null +++ b/dom/media/test/dash/dash-webm-audio-128k.webm diff --git a/dom/media/test/dash/dash-webm-video-320x180.webm b/dom/media/test/dash/dash-webm-video-320x180.webm Binary files differnew file mode 100644 index 0000000000..282e6a2cc3 --- /dev/null +++ b/dom/media/test/dash/dash-webm-video-320x180.webm diff --git a/dom/media/test/dash/dash-webm-video-428x240.webm b/dom/media/test/dash/dash-webm-video-428x240.webm Binary files differnew file mode 100644 index 0000000000..23f2c89616 --- /dev/null +++ b/dom/media/test/dash/dash-webm-video-428x240.webm diff --git a/dom/media/test/dash/garbled.webm b/dom/media/test/dash/garbled.webm new file mode 100644 index 0000000000..ac8eadbddc --- /dev/null +++ b/dom/media/test/dash/garbled.webm @@ -0,0 +1 @@ +PD94bWwgdmVyc2lvbj0iMS4wIiA/PjxNUEQgbWVkaWFQcmVzZW50YXRpb25EdXJhdGlvbj0iUFQxOS41MVMiIG1pbkJ1ZmZlclRpbWU9IlBUMVMiIHByb2ZpbGVzPSJ1cm46d2VibTpkYXNoOnByb2ZpbGU6d2VibS1vbi1kZW1hbmQ6MjAxMiIgdHlwZT0ic3RhdGljIiB4bWxucz0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiIHhtbG5zOnhzaT0iaHR0cDovL3d3dy53My5vcmcvMjAwMS9YTUxTY2hlbWEtaW5zdGFuY2UiIHhzaTpzY2hlbWFMb2NhdGlvbj0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiPjxCYXNlVVJMPmh0dHA6Ly93d3cuZ29vZ2xlLmNvbTwvQmFzZVVSTD48UGVyaW9kIGR1cmF0aW9uPSJQVDE5LjUxUyIgaWQ9IjAiIHN0YXJ0PSJQVDBTIj48QWRhcHRhdGlvblNldCBhdWRpb1NhbXBsaW5nUmF0ZT0iNDgwMDAiIGNvZGVjcz0idm9yYmlzIiBpZD0iMSIgbGFuZz0iZW5nIiBtaW1lVHlwZT0iYXVkaW8vd2VibSIgc3Vic2VnbWVudFN0YXJ0c1dpdGhTQVA9IjEiPjxSZXByZXNlbnRhdGlvbiBiYW5kd2lkdGg9IjIwMTA5IiBpZD0iMiI+PEJhc2VVUkwvPjxTZWdtZW50QmFzZSBpbmRleFJhbmdlPSIzMTk3ODAtMzIwNjEyIj48SW5pdGlhbGl6YXRpb24gcmFuZ2U9IjAtMjA4NzAiLz48L1NlZ21lbnRCYXNlPjwvUmVwcmVzZW50YXRpb24+PC9BZGFwdGF0aW9uU2V0PjwvUGVyaW9kPjwvTVBEPg diff --git a/dom/media/test/dash_detect_stream_switch.sjs b/dom/media/test/dash_detect_stream_switch.sjs new file mode 100644 index 0000000000..a8abf6f2e5 --- /dev/null +++ b/dom/media/test/dash_detect_stream_switch.sjs @@ -0,0 +1,114 @@ +/* -*- Mode: JavaScript; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +/* dash_detect_stream_switch.sjs + * + * Parses requests for DASH manifests and ensures stream switching takes place + * by verifying the subsegments downloaded and the streams they belong to. + * If unexpected subsegments (byte ranges) are requested, the script will + * will respond with a 404. + */ + +var DEBUG = false; + +function parseQuery(request, key) { + var params = request.queryString.split('&'); + if (DEBUG) { + dump("DASH-SJS: request params = \"" + params + "\"\n"); + } + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") === 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key === "") + return p; + } + return false; +} + +function handleRequest(request, response) +{ + try { + var name = parseQuery(request, "name"); + var range = request.hasHeader("Range") ? request.getHeader("Range") + : undefined; + + // Should not get request for 1st subsegment from 2nd stream, nor 2nd + // subsegment from 1st stream. + if (name == "dash-webm-video-320x180.webm" && range == "bytes=25514-32767" || + name == "dash-webm-video-428x240.webm" && range == "bytes=228-35852") + { + throw "Should not request " + name + " with byte-range " + range; + } else { + var rangeSplit = range.split("="); + if (rangeSplit.length != 2) { + throw "DASH-SJS: ERROR: invalid number of tokens (" + rangeSplit.length + + ") delimited by \'=\' in \'Range\' header."; + } + var offsets = rangeSplit[1].split("-"); + if (offsets.length != 2) { + throw "DASH-SJS: ERROR: invalid number of tokens (" + offsets.length + + ") delimited by \'-\' in \'Range\' header."; + } + var startOffset = parseInt(offsets[0]); + var endOffset = parseInt(offsets[1]); + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + + var paths = "tests/dom/media/test/" + name; + var split = paths.split("/"); + for (var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + + fis.init(file, -1, -1, false); + // Exception: start offset should be within file bounds. + if (startOffset > file.fileSize) { + throw "Starting offset [" + startOffset + "] is after end of file [" + + file.fileSize + "]."; + } + // End offset may be too large in the MPD. Real world HTTP servers just + // return what data they can; do the same here - reduce the end offset. + if (endOffset >= file.fileSize) { + if (DEBUG) { + dump("DASH-SJS: reducing endOffset [" + endOffset + "] to fileSize [" + + (file.fileSize-1) + "]\n"); + } + endOffset = file.fileSize-1; + } + fis.seek(Components.interfaces.nsISeekableStream.NS_SEEK_SET, startOffset); + bis.setInputStream(fis); + + var byteLengthToRead = endOffset + 1 - startOffset; + var totalBytesExpected = byteLengthToRead + startOffset; + if (DEBUG) { + dump("DASH-SJS: byteLengthToRead = " + byteLengthToRead + + " byteLengthToRead+startOffset = " + totalBytesExpected + + " fileSize = " + file.fileSize + "\n"); + } + + var bytes = bis.readBytes(byteLengthToRead); + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + response.setHeader("Content-Length", ""+bytes.length, false); + response.setHeader("Content-Type", "application/dash+xml", false); + var contentRange = "bytes " + startOffset + "-" + endOffset + "/" + + file.fileSize; + response.setHeader("Content-Range", contentRange, false); + response.write(bytes, bytes.length); + bis.close(); + } + } catch (e) { + dump ("DASH-SJS-ERROR: " + e + "\n"); + response.setStatusLine(request.httpVersion, 404, "Not found"); + } +} diff --git a/dom/media/test/detodos-recorder-test.opus b/dom/media/test/detodos-recorder-test.opus Binary files differnew file mode 100644 index 0000000000..88b2eab0f8 --- /dev/null +++ b/dom/media/test/detodos-recorder-test.opus diff --git a/dom/media/test/detodos-recorder-test.opus^headers^ b/dom/media/test/detodos-recorder-test.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/detodos-recorder-test.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/detodos-short.opus b/dom/media/test/detodos-short.opus Binary files differnew file mode 100644 index 0000000000..8bda283fc5 --- /dev/null +++ b/dom/media/test/detodos-short.opus diff --git a/dom/media/test/detodos-short.opus^headers^ b/dom/media/test/detodos-short.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/detodos-short.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/detodos-short.webm b/dom/media/test/detodos-short.webm Binary files differnew file mode 100644 index 0000000000..45af2675a6 --- /dev/null +++ b/dom/media/test/detodos-short.webm diff --git a/dom/media/test/detodos-short.webm^headers^ b/dom/media/test/detodos-short.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/detodos-short.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/detodos.opus b/dom/media/test/detodos.opus Binary files differnew file mode 100644 index 0000000000..6c7ba88a66 --- /dev/null +++ b/dom/media/test/detodos.opus diff --git a/dom/media/test/detodos.opus^headers^ b/dom/media/test/detodos.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/detodos.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/detodos.webm b/dom/media/test/detodos.webm Binary files differnew file mode 100644 index 0000000000..39cfa7f537 --- /dev/null +++ b/dom/media/test/detodos.webm diff --git a/dom/media/test/detodos.webm^headers^ b/dom/media/test/detodos.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/detodos.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/dirac.ogg b/dom/media/test/dirac.ogg Binary files differnew file mode 100644 index 0000000000..2986cf1e80 --- /dev/null +++ b/dom/media/test/dirac.ogg diff --git a/dom/media/test/dirac.ogg^headers^ b/dom/media/test/dirac.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/dirac.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/dynamic_resource.sjs b/dom/media/test/dynamic_resource.sjs new file mode 100644 index 0000000000..7731612184 --- /dev/null +++ b/dom/media/test/dynamic_resource.sjs @@ -0,0 +1,48 @@ +function parseQuery(request, key) { + var params = request.queryString.split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +// Return resource1 file content for the first request with a given key. +// All subsequent requests return resource2. Both must be video/ogg. +function handleRequest(request, response) +{ + var key = parseQuery(request, "key"); + var resource1 = parseQuery(request, "res1"); + var resource2 = parseQuery(request, "res2"); + + var resource = getState(key) == "2" ? resource2 : resource1; + setState(key, "2"); + + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/" + resource; + var split = paths.split("/"); + for(var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + dump("file=" + file + "\n"); + bis.setInputStream(fis); + var bytes = bis.readBytes(bis.available()); + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + response.setHeader("Content-Range", "bytes 0-" + (bytes.length - 1) + "/" + bytes.length); + response.setHeader("Content-Length", ""+bytes.length, false); + response.setHeader("Content-Type", "video/ogg", false); + response.write(bytes, bytes.length); + bis.close(); +} diff --git a/dom/media/test/eme.js b/dom/media/test/eme.js new file mode 100644 index 0000000000..fffe71607c --- /dev/null +++ b/dom/media/test/eme.js @@ -0,0 +1,496 @@ +/* import-globals-from manifest.js */ + +const CLEARKEY_KEYSYSTEM = "org.w3.clearkey"; + +const gCencMediaKeySystemConfig = [ + { + initDataTypes: ["cenc"], + videoCapabilities: [{ contentType: "video/mp4" }], + audioCapabilities: [{ contentType: "audio/mp4" }], + }, +]; + +function bail(message) { + return function(err) { + if (err) { + message += "; " + String(err); + } + ok(false, message); + if (err) { + info(String(err)); + } + SimpleTest.finish(); + }; +} + +function ArrayBufferToString(arr) { + var str = ""; + var view = new Uint8Array(arr); + for (var i = 0; i < view.length; i++) { + str += String.fromCharCode(view[i]); + } + return str; +} + +function StringToArrayBuffer(str) { + var arr = new ArrayBuffer(str.length); + var view = new Uint8Array(arr); + for (var i = 0; i < str.length; i++) { + view[i] = str.charCodeAt(i); + } + return arr; +} + +function StringToHex(str) { + var res = ""; + for (var i = 0; i < str.length; ++i) { + res += ("0" + str.charCodeAt(i).toString(16)).slice(-2); + } + return res; +} + +function Base64ToHex(str) { + var bin = window.atob(str.replace(/-/g, "+").replace(/_/g, "/")); + var res = ""; + for (var i = 0; i < bin.length; i++) { + res += ("0" + bin.charCodeAt(i).toString(16)).substr(-2); + } + return res; +} + +function HexToBase64(hex) { + var bin = ""; + for (var i = 0; i < hex.length; i += 2) { + bin += String.fromCharCode(parseInt(hex.substr(i, 2), 16)); + } + return window + .btoa(bin) + .replace(/=/g, "") + .replace(/\+/g, "-") + .replace(/\//g, "_"); +} + +function TimeRangesToString(trs) { + var l = trs.length; + if (l === 0) { + return "-"; + } + var s = ""; + var i = 0; + for (;;) { + s += trs.start(i) + "-" + trs.end(i); + if (++i === l) { + return s; + } + s += ","; + } +} + +function SourceBufferToString(sb) { + return ( + "SourceBuffer{" + + "AppendMode=" + + (sb.AppendMode || "-") + + ", updating=" + + (sb.updating ? "true" : "false") + + ", buffered=" + + TimeRangesToString(sb.buffered) + + ", audioTracks=" + + (sb.audioTracks ? sb.audioTracks.length : "-") + + ", videoTracks=" + + (sb.videoTracks ? sb.videoTracks.length : "-") + + "}" + ); +} + +function SourceBufferListToString(sbl) { + return "SourceBufferList[" + sbl.map(SourceBufferToString).join(", ") + "]"; +} + +function GenerateClearKeyLicense(licenseRequest, keyStore) { + var msgStr = ArrayBufferToString(licenseRequest); + var msg = JSON.parse(msgStr); + + var keys = []; + for (var i = 0; i < msg.kids.length; i++) { + var id64 = msg.kids[i]; + var idHex = Base64ToHex(msg.kids[i]).toLowerCase(); + var key = keyStore[idHex]; + + if (key) { + keys.push({ + kty: "oct", + kid: id64, + k: HexToBase64(key), + }); + } + } + + return new TextEncoder().encode( + JSON.stringify({ + keys, + type: msg.type || "temporary", + }) + ); +} + +function UpdateSessionFunc(test, token, sessionType, resolve, reject) { + return function(ev) { + var license = GenerateClearKeyLicense(ev.message, test.keys); + Log( + token, + "sending update message to CDM: " + new TextDecoder().decode(license) + ); + ev.target + .update(license) + .then(function() { + Log(token, "MediaKeySession update ok!"); + resolve(ev.target); + }) + .catch(function(reason) { + reject(`${token} MediaKeySession update failed: ${reason}`); + }); + }; +} + +function MaybeCrossOriginURI(test, uri) { + if (test.crossOrigin) { + return "https://example.com:443/tests/dom/media/test/allowed.sjs?" + uri; + } + return uri; +} + +function AppendTrack(test, ms, track, token) { + return new Promise(function(resolve, reject) { + var sb; + var curFragment = 0; + var fragments = track.fragments; + var fragmentFile; + + function addNextFragment() { + if (curFragment >= fragments.length) { + Log(token, track.name + ": end of track"); + resolve(); + return; + } + + fragmentFile = MaybeCrossOriginURI(test, fragments[curFragment++]); + + var req = new XMLHttpRequest(); + req.open("GET", fragmentFile); + req.responseType = "arraybuffer"; + + req.addEventListener("load", function() { + Log( + token, + track.name + ": fetch of " + fragmentFile + " complete, appending" + ); + sb.appendBuffer(new Uint8Array(req.response)); + }); + + req.addEventListener("error", function() { + reject(`${token} - ${track.name}: error fetching ${fragmentFile}`); + }); + req.addEventListener("abort", function() { + reject(`${token} - ${track.name}: aborted fetching ${fragmentFile}`); + }); + + Log( + token, + track.name + + ": addNextFragment() fetching next fragment " + + fragmentFile + ); + req.send(null); + } + + Log(token, track.name + ": addSourceBuffer(" + track.type + ")"); + sb = ms.addSourceBuffer(track.type); + sb.addEventListener("updateend", function() { + Log( + token, + track.name + + ": updateend for " + + fragmentFile + + ", " + + SourceBufferToString(sb) + ); + addNextFragment(); + }); + + addNextFragment(); + }); +} + +//Returns a promise that is resolved when the media element is ready to have +//its play() function called; when it's loaded MSE fragments. +function LoadTest(test, elem, token, endOfStream = true) { + if (!test.tracks) { + ok(false, token + " test does not have a tracks list"); + return Promise.reject(); + } + + var ms = new MediaSource(); + elem.src = URL.createObjectURL(ms); + elem.crossOrigin = test.crossOrigin || false; + + return new Promise(function(resolve, reject) { + ms.addEventListener( + "sourceopen", + function() { + Log(token, "sourceopen"); + Promise.all( + test.tracks.map(function(track) { + return AppendTrack(test, ms, track, token); + }) + ) + .then(function() { + Log(token, "Tracks loaded, calling MediaSource.endOfStream()"); + if (endOfStream) { + ms.endOfStream(); + } + resolve(); + }) + .catch(reject); + }, + { once: true } + ); + }); +} + +function EMEPromise() { + var self = this; + self.promise = new Promise(function(resolve, reject) { + self.resolve = resolve; + self.reject = reject; + }); +} + +/* + * Create a new MediaKeys object. + * Return a promise which will be resolved with a new MediaKeys object, + * or will be rejected with a string that describes the failure. + */ +function CreateMediaKeys(v, test, token) { + let p = new EMEPromise(); + + function streamType(type) { + var x = test.tracks.find(o => o.name == type); + return x ? x.type : undefined; + } + + function onencrypted(ev) { + var options = { initDataTypes: [ev.initDataType] }; + if (streamType("video")) { + options.videoCapabilities = [{ contentType: streamType("video") }]; + } + if (streamType("audio")) { + options.audioCapabilities = [{ contentType: streamType("audio") }]; + } + navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, [options]).then( + keySystemAccess => { + keySystemAccess + .createMediaKeys() + .then(p.resolve, () => + p.reject(`${token} Failed to create MediaKeys object.`) + ); + }, + () => p.reject(`${token} Failed to request key system access.`) + ); + } + + v.addEventListener("encrypted", onencrypted, { once: true }); + return p.promise; +} + +/* + * Create a new MediaKeys object and provide it to the media element. + * Return a promise which will be resolved if succeeded, or will be rejected + * with a string that describes the failure. + */ +function CreateAndSetMediaKeys(v, test, token) { + let p = new EMEPromise(); + + CreateMediaKeys(v, test, token).then(mediaKeys => { + v.setMediaKeys(mediaKeys).then(p.resolve, () => + p.reject(`${token} Failed to set MediaKeys on <video> element.`) + ); + }, p.reject); + + return p.promise; +} + +/* + * Collect the init data from 'encrypted' events. + * Return a promise which will be resolved with the init data when collection + * is completed (specified by test.sessionCount). + */ +function LoadInitData(v, test, token) { + let p = new EMEPromise(); + let initDataQueue = []; + + // Call SimpleTest._originalSetTimeout() to bypass the flaky timeout checker. + let timer = SimpleTest._originalSetTimeout.call( + window, + () => { + p.reject(`${token} Timed out in waiting for the init data.`); + }, + 60000 + ); + + function onencrypted(ev) { + initDataQueue.push(ev); + Log( + token, + `got encrypted(${ev.initDataType}, ` + + `${StringToHex(ArrayBufferToString(ev.initData))}) event.` + ); + if (test.sessionCount == initDataQueue.length) { + p.resolve(initDataQueue); + clearTimeout(timer); + } + } + + v.addEventListener("encrypted", onencrypted); + return p.promise; +} + +/* + * Generate a license request and update the session. + * Return a promsise which will be resolved with the updated session + * or rejected with a string that describes the failure. + */ +function MakeRequest(test, token, ev, session, sessionType) { + sessionType = sessionType || "temporary"; + let p = new EMEPromise(); + let str = + `session[${session.sessionId}].generateRequest(` + + `${ev.initDataType}, ${StringToHex(ArrayBufferToString(ev.initData))})`; + + session.addEventListener( + "message", + UpdateSessionFunc(test, token, sessionType, p.resolve, p.reject) + ); + + Log(token, str); + session.generateRequest(ev.initDataType, ev.initData).catch(reason => { + // Reject the promise if generateRequest() failed. + // Otherwise it will be resolved in UpdateSessionFunc(). + p.reject(`${token}: ${str} failed; ${reason}`); + }); + + return p.promise; +} + +/* + * Process the init data by calling MakeRequest(). + * Return a promise which will be resolved with the updated sessions + * when all init data are processed or rejected if any failure. + */ +function ProcessInitData(v, test, token, initData, sessionType) { + return Promise.all( + initData.map(ev => { + let session = v.mediaKeys.createSession(sessionType); + return MakeRequest(test, token, ev, session, sessionType); + }) + ); +} + +/* + * Clean up the |v| element. + */ +function CleanUpMedia(v) { + v.setMediaKeys(null); + v.remove(); + v.removeAttribute("src"); + v.load(); +} + +/* + * Close all sessions and clean up the |v| element. + */ +function CloseSessions(v, sessions) { + return Promise.all(sessions.map(s => s.close())).then(CleanUpMedia(v)); +} + +/* + * Set up media keys and source buffers for the media element. + * Return a promise resolved when all key sessions are updated or rejected + * if any failure. + */ +function SetupEME(v, test, token) { + let p = new EMEPromise(); + + v.onerror = function() { + p.reject(`${token} got an error event.`); + }; + + Promise.all([ + LoadInitData(v, test, token), + CreateAndSetMediaKeys(v, test, token), + LoadTest(test, v, token), + ]) + .then(values => { + let initData = values[0]; + return ProcessInitData(v, test, token, initData); + }) + .then(p.resolve, p.reject); + + return p.promise; +} + +function SetupEMEPref(callback) { + var prefs = [ + ["media.mediasource.enabled", true], + ["media.mediasource.webm.enabled", true], + ]; + + if ( + SpecialPowers.Services.appinfo.name == "B2G" || + !manifestVideo().canPlayType("video/mp4") + ) { + // XXX remove once we have mp4 PlatformDecoderModules on all platforms. + prefs.push(["media.use-blank-decoder", true]); + } + + SpecialPowers.pushPrefEnv({ set: prefs }, callback); +} + +function fetchWithXHR(uri, onLoadFunction) { + var p = new Promise(function(resolve, reject) { + var xhr = new XMLHttpRequest(); + xhr.open("GET", uri, true); + xhr.responseType = "arraybuffer"; + xhr.addEventListener("load", function() { + is( + xhr.status, + 200, + "fetchWithXHR load uri='" + uri + "' status=" + xhr.status + ); + resolve(xhr.response); + }); + xhr.send(); + }); + + if (onLoadFunction) { + p.then(onLoadFunction); + } + + return p; +} + +function once(target, name, cb) { + var p = new Promise(function(resolve, reject) { + target.addEventListener( + name, + function(arg) { + resolve(arg); + }, + { once: true } + ); + }); + if (cb) { + p.then(cb); + } + return p; +} diff --git a/dom/media/test/empty_size.mp3 b/dom/media/test/empty_size.mp3 Binary files differnew file mode 100644 index 0000000000..0c208a2959 --- /dev/null +++ b/dom/media/test/empty_size.mp3 diff --git a/dom/media/test/file_access_controls.html b/dom/media/test/file_access_controls.html new file mode 100644 index 0000000000..2f7bc360ed --- /dev/null +++ b/dom/media/test/file_access_controls.html @@ -0,0 +1,160 @@ +<html> +<head> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="setTimeout(load, 0);"> +<script> + +// Page URL: http://example.org/tests/dom/media/test/file_access_controls.html + +var gResource = getPlayableVideo(gSmallTests).name; + +var gTests = [ + { + // Test 0 + url: "redirect.sjs?domain=example.com&file="+ gResource, + result: "error", + description: "Won't load when redirected to different domain", + },{ + // Test 1 + url: "redirect.sjs?domain=example.com&allowed&file=" + gResource, + result: "loadeddata", + description: "Can load when redirected to different domain with allow-origin", + },{ + // Test 2 + url: "redirect.sjs?domain=test1.example.org&file=" + gResource, + result: "error", + description: "Won't load when redirected to subdomain", + },{ + // Test 3 + url: "redirect.sjs?domain=test1.example.org&allowed&file=" + gResource, + result: "loadeddata", + description: "Can load when redirected to subdomain with allow-origin", + },{ + // Test 4 + url: "redirect.sjs?domain=example.org&file=" + gResource, + result: "loadeddata", + description: "Can load when redirected to same domain", + },{ + // Test 5 + url: "http://example.org/tests/dom/media/test/" + gResource, + result: "loadeddata", + description: "Can load from same domain" + },{ + // Test 6 + url: "http://example.org:8000/tests/dom/media/test/" + gResource, + result: "error", + description: "Won't load from different port on same domain" + },{ + // Test 7 + url: "http://example.org:8000/tests/dom/media/test/allowed.sjs?" + gResource, + result: "loadeddata", + description: "Can load from different port on same domain with allow-origin", + },{ + // Test 8 + url: "http://example.com/tests/dom/media/test/" + gResource, + result: "error", + description: "Won't load cross domain", + },{ + // Test 9 + url: "http://example.com/tests/dom/media/test/allowed.sjs?" + gResource, + result: "loadeddata", + description: "Can load cross domain with allow-origin", + },{ + // Test 10 + url: "http://test1.example.org/tests/dom/media/test/allowed.sjs?" + gResource, + result: "loadeddata", + description: "Can load from subdomain with allow-origin", + },{ + // Test 11 + url: "http://test1.example.org/tests/dom/media/test/" + gResource, + result: "error", + description: "Won't load from subdomain", + } +]; + +var gTestNum = 0; +var gVideo = null; +var gTestedRemoved = false; + +function eventHandler(event) { + //dump((gTestNum - 1) + ": " + event.type + "\n"); + var video = event.target; + opener.postMessage({"result": (event.type == video.expectedResult), + "message": video.testDescription + (gTestedRemoved ? " (element not in document)" : " (element in document)")}, + "http://mochi.test:8888"); + // Make sure any extra events cause an error + video.expectedResult = "<none>"; + nextTest(); +} + +function createVideo() { + var v = document.createElement('video'); + v.addEventListener('loadeddata', eventHandler); + v.addEventListener('error', eventHandler); + v.crossOrigin = 'anonymous'; + return v; +} + +function load() { + opener.postMessage({"result": (window.location.href == "http://example.org/tests/dom/media/test/file_access_controls.html"), + "message": "We must be on a example.org:80"}, + "http://mochi.test:8888"); + + nextTest(); +} + +function nextTest() { + //dump("nextTest() called, gTestNum="+gTestNum+" gTestedRemoved="+gTestedRemoved+"\n"); + if (gTestNum == gTests.length) { + //dump("gTestNum == gTests.length\n"); + if (!gTestedRemoved) { + // Repeat all tests with element removed from doc, should get same result. + gTestedRemoved = true; + gTestNum = 0; + } else { + //dump("Exiting...\n"); + // We're done, exit the test. + done(); + window.close(); + return; + } + } + + if (gVideo) { + gVideo.remove(); + gVideo.removeAttribute("src"); + gVideo.load(); + } + + gVideo = null; + SpecialPowers.forceGC(); + + gVideo = createVideo(); + gVideo.expectedResult = gTests[gTestNum].result; + gVideo.testDescription = gTests[gTestNum].description; + // Uniquify the resource URL to ensure that the resources loaded by earlier or subsequent tests + // don't overlap with the resources we load here, which are loaded with non-default preferences set. + // We also want to make sure that an HTTP fetch actually happens for each testcase. + var url = gTests[gTestNum].url; + var random = Math.floor(Math.random()*1000000000); + url += (url.search(/\?/) < 0 ? "?" : "&") + "rand=" + random; + gVideo.src = url; + //dump("Starting test " + gTestNum + " at " + gVideo.src + " expecting:" + gVideo.expectedResult + "\n"); + if (!gTestedRemoved) { + document.body.appendChild(gVideo); + // Will cause load() to be invoked. + } else { + gVideo.load(); + } + gTestNum++; +} + +function done() { + opener.postMessage({"done": "true"}, "http://mochi.test:8888"); +} + +</script> +</body> +</html> + diff --git a/dom/media/test/file_eme_createMediaKeys.html b/dom/media/test/file_eme_createMediaKeys.html new file mode 100644 index 0000000000..3ff782b1bf --- /dev/null +++ b/dom/media/test/file_eme_createMediaKeys.html @@ -0,0 +1,47 @@ +<!DOCTYPE html> +<html> +<head> +<title>Eme createMediaKeys helper page</title> +<script> +// This script waits for a message then attempts to requestMediaKeySystemAccess +// then createMediaKeys. On success posts 'successCreatingMediaKeys' to the +// source of the message, on failure posts 'failureCreatingMediaKeys' and a +// description of the failure to the source of the message. + +async function createMediaKeys() { + const clearKeyOptions = [ + { + initDataTypes: ["webm"], + videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }], + }, + ]; + + let access = await navigator.requestMediaKeySystemAccess( + "org.w3.clearkey", + clearKeyOptions + ); + + return access.createMediaKeys(); +} +function setupMessageListener() { + window.onmessage = async event => { + // We don't bother checking the message data since it should always be + // telling us to create media keys. + try { + let keys = await createMediaKeys(); + if (!keys) { + event.source.postMessage("failureCreatingMediaKeys no keys", "*"); + return; + } + event.source.postMessage("successCreatingMediaKeys", "*"); + } catch (e) { + event.source.postMessage(`failureCreatingMediaKeys ${e}`, "*"); + } + }; +} +window.onload = setupMessageListener; +</script> +</head> +<body> +</body> +</html> diff --git a/dom/media/test/flac-noheader-s16.flac b/dom/media/test/flac-noheader-s16.flac Binary files differnew file mode 100644 index 0000000000..01152142a9 --- /dev/null +++ b/dom/media/test/flac-noheader-s16.flac diff --git a/dom/media/test/flac-noheader-s16.flac^headers^ b/dom/media/test/flac-noheader-s16.flac^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/flac-noheader-s16.flac^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/flac-s24.flac b/dom/media/test/flac-s24.flac Binary files differnew file mode 100644 index 0000000000..1ba5e27a15 --- /dev/null +++ b/dom/media/test/flac-s24.flac diff --git a/dom/media/test/flac-s24.flac^headers^ b/dom/media/test/flac-s24.flac^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/flac-s24.flac^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/flac-sample-cenc.mp4 b/dom/media/test/flac-sample-cenc.mp4 Binary files differnew file mode 100644 index 0000000000..c89190387a --- /dev/null +++ b/dom/media/test/flac-sample-cenc.mp4 diff --git a/dom/media/test/flac-sample-cenc.mp4^headers^ b/dom/media/test/flac-sample-cenc.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/flac-sample-cenc.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/flac-sample.mp4 b/dom/media/test/flac-sample.mp4 Binary files differnew file mode 100644 index 0000000000..d39c94a7b2 --- /dev/null +++ b/dom/media/test/flac-sample.mp4 diff --git a/dom/media/test/flac-sample.mp4^headers^ b/dom/media/test/flac-sample.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/flac-sample.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/fragment_noplay.js b/dom/media/test/fragment_noplay.js new file mode 100644 index 0000000000..247641763f --- /dev/null +++ b/dom/media/test/fragment_noplay.js @@ -0,0 +1,19 @@ +function test_fragment_noplay(v, start, end, is, ok, finish) { + function onLoadedMetadata() { + var s = start == null ? 0 : start; + var e = end == null ? v.duration : end; + var a = s - 0.15; + var b = s + 0.15; + ok( + v.currentTime >= a && v.currentTime <= b, + "loadedmetadata currentTime is " + a + " < " + v.currentTime + " < " + b + ); + ok( + v.mozFragmentEnd == e, + "mozFragmentEnd (" + v.mozFragmentEnd + ") == end Time (" + e + ")" + ); + finish(); + } + + v.addEventListener("loadedmetadata", onLoadedMetadata); +} diff --git a/dom/media/test/fragment_play.js b/dom/media/test/fragment_play.js new file mode 100644 index 0000000000..ec0fe7952a --- /dev/null +++ b/dom/media/test/fragment_play.js @@ -0,0 +1,92 @@ +function test_fragment_play(v, start, end, is, ok, finish) { + var completed = false; + var loadedMetadataRaised = false; + var seekedRaised = false; + var pausedRaised = false; + + function onLoadedMetadata() { + var s = start == null ? 0 : start; + var e = end == null ? v.duration : end; + ok( + v.currentTime == s, + "loadedmetadata currentTime is " + v.currentTime + " != " + s + ); + ok( + v.mozFragmentEnd == e, + "mozFragmentEnd (" + v.mozFragmentEnd + ") == end Time (" + e + ")" + ); + loadedMetadataRaised = true; + v.play(); + } + + function onSeeked() { + if (completed) { + return; + } + + var s = start == null ? 0 : start; + ok( + v.currentTime - s < 0.1, + "seeked currentTime is " + + v.currentTime + + " != " + + s + + " (fuzzy compare +-0.1)" + ); + + seekedRaised = true; + } + + function onTimeUpdate() { + if (completed) { + return; + } + + v._lastTimeUpdate = v.currentTime; + } + + function onPause() { + if (completed) { + return; + } + + var e = end == null ? v.duration : end; + var a = e - 0.05; + var b = e + 0.05; + ok( + v.currentTime >= a && v.currentTime <= b, + "paused currentTime is " + + a + + " < " + + v.currentTime + + " < " + + b + + " ? " + + v._lastTimeUpdate + ); + pausedRaised = true; + v.play(); + } + + function onEnded() { + if (completed) { + return; + } + + completed = true; + ok(loadedMetadataRaised, "loadedmetadata event"); + if (start) { + ok(seekedRaised, "seeked event"); + } + if (end) { + ok(pausedRaised, "paused event: " + end + " " + v.duration); + } + finish(); + } + + v.addEventListener("ended", onEnded); + v.addEventListener("loadedmetadata", onLoadedMetadata); + v.addEventListener("seeked", onSeeked); + v.addEventListener("pause", onPause); + v.addEventListener("timeupdate", onTimeUpdate); +} diff --git a/dom/media/test/gUM_support.js b/dom/media/test/gUM_support.js new file mode 100644 index 0000000000..2be885abf4 --- /dev/null +++ b/dom/media/test/gUM_support.js @@ -0,0 +1,106 @@ +// Support script for test that use getUserMedia. This allows explicit +// configuration of prefs which affect gUM. See also +// `testing/mochitest/runtests.py` for how the harness configures values. + +// Setup preconditions for tests using getUserMedia. This functions helps +// manage different prefs that affect gUM calls in tests and makes explicit +// the expected state before test runs. +async function pushGetUserMediaTestPrefs({ + fakeAudio = false, + fakeVideo = false, + loopbackAudio = false, + loopbackVideo = false, +}) { + // Make sure we have sensical arguments + if (!fakeAudio && !loopbackAudio) { + throw new Error( + "pushGetUserMediaTestPrefs: Should have fake or loopback audio!" + ); + } else if (fakeAudio && loopbackAudio) { + throw new Error( + "pushGetUserMediaTestPrefs: Should not have both fake and loopback audio!" + ); + } + if (!fakeVideo && !loopbackVideo) { + throw new Error( + "pushGetUserMediaTestPrefs: Should have fake or loopback video!" + ); + } else if (fakeVideo && loopbackVideo) { + throw new Error( + "pushGetUserMediaTestPrefs: Should not have both fake and loopback video!" + ); + } + + let testPrefs = []; + if (fakeAudio) { + // Unset the loopback device so it doesn't take precedence + testPrefs.push(["media.audio_loopback_dev", ""]); + // Setup fake streams pref + testPrefs.push(["media.navigator.streams.fake", true]); + } + if (loopbackAudio) { + // If audio loopback is requested we expect the test harness to have set + // the loopback device pref, make sure it's set + let audioLoopDev = SpecialPowers.getCharPref( + "media.audio_loopback_dev", + "" + ); + if (!audioLoopDev) { + throw new Error( + "pushGetUserMediaTestPrefs: Loopback audio requested but " + + "media.audio_loopback_dev does not appear to be set!" + ); + } + } + if (fakeVideo) { + // Unset the loopback device so it doesn't take precedence + testPrefs.push(["media.video_loopback_dev", ""]); + // Setup fake streams pref + testPrefs.push(["media.navigator.streams.fake", true]); + } + if (loopbackVideo) { + // If video loopback is requested we expect the test harness to have set + // the loopback device pref, make sure it's set + let videoLoopDev = SpecialPowers.getCharPref( + "media.video_loopback_dev", + "" + ); + if (!videoLoopDev) { + throw new Error( + "pushGetUserMediaTestPrefs: Loopback video requested but " + + "media.video_loopback_dev does not appear to be set!" + ); + } + } + if (loopbackAudio || loopbackVideo) { + // Prevent gUM permission prompt. Since loopback devices are considered + // real devices we need to set prefs so the gUM prompt isn't presented. + testPrefs.push(["media.navigator.permission.disabled", true]); + } + return SpecialPowers.pushPrefEnv({ set: testPrefs }); +} + +// Setup preconditions for tests using getUserMedia. This function will +// configure prefs to select loopback device(s) if it can find loopback device +// names already set in the prefs. If no loopback device name can be found then +// prefs are setup such that a fake device is used. +async function setupGetUserMediaTestPrefs() { + let prefRequests = {}; + let audioLoopDev = SpecialPowers.getCharPref("media.audio_loopback_dev", ""); + if (audioLoopDev) { + prefRequests.fakeAudio = false; + prefRequests.loopbackAudio = true; + } else { + prefRequests.fakeAudio = true; + prefRequests.loopbackAudio = false; + } + let videoLoopDev = SpecialPowers.getCharPref("media.video_loopback_dev", ""); + if (videoLoopDev) { + prefRequests.fakeVideo = false; + prefRequests.loopbackVideo = true; + } else { + prefRequests.fakeVideo = true; + prefRequests.loopbackVideo = false; + } + return pushGetUserMediaTestPrefs(prefRequests); +} diff --git a/dom/media/test/gizmo-frag.mp4 b/dom/media/test/gizmo-frag.mp4 Binary files differnew file mode 100644 index 0000000000..f6980663c2 --- /dev/null +++ b/dom/media/test/gizmo-frag.mp4 diff --git a/dom/media/test/gizmo-noaudio.mp4 b/dom/media/test/gizmo-noaudio.mp4 Binary files differnew file mode 100644 index 0000000000..24732a4064 --- /dev/null +++ b/dom/media/test/gizmo-noaudio.mp4 diff --git a/dom/media/test/gizmo-noaudio.mp4^headers^ b/dom/media/test/gizmo-noaudio.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/gizmo-noaudio.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/gizmo-noaudio.webm b/dom/media/test/gizmo-noaudio.webm Binary files differnew file mode 100644 index 0000000000..9f412cb6e3 --- /dev/null +++ b/dom/media/test/gizmo-noaudio.webm diff --git a/dom/media/test/gizmo-noaudio.webm^headers^ b/dom/media/test/gizmo-noaudio.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/gizmo-noaudio.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/gizmo-short.mp4 b/dom/media/test/gizmo-short.mp4 Binary files differnew file mode 100644 index 0000000000..f8caec741e --- /dev/null +++ b/dom/media/test/gizmo-short.mp4 diff --git a/dom/media/test/gizmo-short.mp4^headers^ b/dom/media/test/gizmo-short.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/gizmo-short.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/gizmo.mp4 b/dom/media/test/gizmo.mp4 Binary files differnew file mode 100644 index 0000000000..87efad5ade --- /dev/null +++ b/dom/media/test/gizmo.mp4 diff --git a/dom/media/test/gizmo.mp4^headers^ b/dom/media/test/gizmo.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/gizmo.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/gizmo.webm b/dom/media/test/gizmo.webm Binary files differnew file mode 100644 index 0000000000..518531a93f --- /dev/null +++ b/dom/media/test/gizmo.webm diff --git a/dom/media/test/gizmo.webm^headers^ b/dom/media/test/gizmo.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/gizmo.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/gzipped_mp4.sjs b/dom/media/test/gzipped_mp4.sjs new file mode 100644 index 0000000000..132b87e851 --- /dev/null +++ b/dom/media/test/gzipped_mp4.sjs @@ -0,0 +1,27 @@ +function getGzippedFileBytes()
+{
+ var file;
+ getObjectState("SERVER_ROOT", function(serverRoot) {
+ file = serverRoot.getFile("tests/dom/media/test/short.mp4.gz");
+ });
+ var fileInputStream =
+ Components.classes['@mozilla.org/network/file-input-stream;1']
+ .createInstance(Components.interfaces.nsIFileInputStream);
+ var binaryInputStream =
+ Components.classes["@mozilla.org/binaryinputstream;1"]
+ .createInstance(Components.interfaces.nsIBinaryInputStream);
+ fileInputStream.init(file, -1, -1, 0);
+ binaryInputStream.setInputStream(fileInputStream);
+ return binaryInputStream.readBytes(binaryInputStream.available());
+}
+
+function handleRequest(request, response)
+{
+ var bytes = getGzippedFileBytes();
+ response.setHeader("Content-Length", String(bytes.length), false);
+ response.setHeader("Content-Type", "video/mp4", false);
+ response.setHeader("Access-Control-Allow-Origin", "*", false);
+ response.setHeader("Content-Encoding", "gzip", false);
+ response.setHeader("Cache-Control", "no-cache", false);
+ response.write(bytes, bytes.length);
+}
diff --git a/dom/media/test/hls/400x300_prog_index.m3u8 b/dom/media/test/hls/400x300_prog_index.m3u8 new file mode 100644 index 0000000000..3252eb178f --- /dev/null +++ b/dom/media/test/hls/400x300_prog_index.m3u8 @@ -0,0 +1,10 @@ +#EXTM3U +#EXT-X-TARGETDURATION:10 +#EXT-X-VERSION:3 +#EXT-X-MEDIA-SEQUENCE:0 +#EXT-X-PLAYLIST-TYPE:VOD +#EXTINF:9.97667, +400x300_seg0.ts +#EXTINF:9.97667, +400x300_seg1.ts +#EXT-X-ENDLIST diff --git a/dom/media/test/hls/400x300_prog_index_5s.m3u8 b/dom/media/test/hls/400x300_prog_index_5s.m3u8 new file mode 100644 index 0000000000..8e9d19f764 --- /dev/null +++ b/dom/media/test/hls/400x300_prog_index_5s.m3u8 @@ -0,0 +1,8 @@ +#EXTM3U +#EXT-X-TARGETDURATION:4.00 +#EXT-X-VERSION:3 +#EXT-X-MEDIA-SEQUENCE:0 +#EXT-X-PLAYLIST-TYPE:VOD +#EXTINF:4.00, +400x300_seg0_5s.ts +#EXT-X-ENDLIST diff --git a/dom/media/test/hls/400x300_seg0.ts b/dom/media/test/hls/400x300_seg0.ts Binary files differnew file mode 100644 index 0000000000..b17b0a88ff --- /dev/null +++ b/dom/media/test/hls/400x300_seg0.ts diff --git a/dom/media/test/hls/400x300_seg0_5s.ts b/dom/media/test/hls/400x300_seg0_5s.ts Binary files differnew file mode 100644 index 0000000000..9504d5d889 --- /dev/null +++ b/dom/media/test/hls/400x300_seg0_5s.ts diff --git a/dom/media/test/hls/400x300_seg1.ts b/dom/media/test/hls/400x300_seg1.ts Binary files differnew file mode 100644 index 0000000000..d751091e68 --- /dev/null +++ b/dom/media/test/hls/400x300_seg1.ts diff --git a/dom/media/test/hls/416x243_prog_index_5s.m3u8 b/dom/media/test/hls/416x243_prog_index_5s.m3u8 new file mode 100644 index 0000000000..ae518e7890 --- /dev/null +++ b/dom/media/test/hls/416x243_prog_index_5s.m3u8 @@ -0,0 +1,8 @@ +#EXTM3U +#EXT-X-TARGETDURATION:4.04 +#EXT-X-VERSION:4 +#EXT-X-MEDIA-SEQUENCE:0 +#EXT-X-PLAYLIST-TYPE:VOD +#EXTINF:4.04, +416x243_seg0_5s.ts +#EXT-X-ENDLIST diff --git a/dom/media/test/hls/416x243_seg0_5s.ts b/dom/media/test/hls/416x243_seg0_5s.ts Binary files differnew file mode 100644 index 0000000000..48e5473276 --- /dev/null +++ b/dom/media/test/hls/416x243_seg0_5s.ts diff --git a/dom/media/test/hls/640x480_prog_index.m3u8 b/dom/media/test/hls/640x480_prog_index.m3u8 new file mode 100644 index 0000000000..ac1212e2e7 --- /dev/null +++ b/dom/media/test/hls/640x480_prog_index.m3u8 @@ -0,0 +1,10 @@ +#EXTM3U +#EXT-X-TARGETDURATION:10 +#EXT-X-VERSION:3 +#EXT-X-MEDIA-SEQUENCE:0 +#EXT-X-PLAYLIST-TYPE:VOD +#EXTINF:9.97667, +640x480_seg0.ts +#EXTINF:9.97667, +640x480_seg1.ts +#EXT-X-ENDLIST diff --git a/dom/media/test/hls/640x480_seg0.ts b/dom/media/test/hls/640x480_seg0.ts Binary files differnew file mode 100644 index 0000000000..9bf0f0454a --- /dev/null +++ b/dom/media/test/hls/640x480_seg0.ts diff --git a/dom/media/test/hls/640x480_seg1.ts b/dom/media/test/hls/640x480_seg1.ts Binary files differnew file mode 100644 index 0000000000..c1ed938f44 --- /dev/null +++ b/dom/media/test/hls/640x480_seg1.ts diff --git a/dom/media/test/hls/960x720_prog_index.m3u8 b/dom/media/test/hls/960x720_prog_index.m3u8 new file mode 100644 index 0000000000..8ff18b089c --- /dev/null +++ b/dom/media/test/hls/960x720_prog_index.m3u8 @@ -0,0 +1,10 @@ +#EXTM3U +#EXT-X-TARGETDURATION:10 +#EXT-X-VERSION:3 +#EXT-X-MEDIA-SEQUENCE:0 +#EXT-X-PLAYLIST-TYPE:VOD +#EXTINF:9.97667, +960x720_seg0.ts +#EXTINF:9.97667, +960x720_seg1.ts +#EXT-X-ENDLIST diff --git a/dom/media/test/hls/960x720_seg0.ts b/dom/media/test/hls/960x720_seg0.ts Binary files differnew file mode 100644 index 0000000000..031bfe30d6 --- /dev/null +++ b/dom/media/test/hls/960x720_seg0.ts diff --git a/dom/media/test/hls/960x720_seg1.ts b/dom/media/test/hls/960x720_seg1.ts Binary files differnew file mode 100644 index 0000000000..63f15cb0cb --- /dev/null +++ b/dom/media/test/hls/960x720_seg1.ts diff --git a/dom/media/test/hls/bipbop_16x9_single.m3u8 b/dom/media/test/hls/bipbop_16x9_single.m3u8 new file mode 100644 index 0000000000..dce6a76c7b --- /dev/null +++ b/dom/media/test/hls/bipbop_16x9_single.m3u8 @@ -0,0 +1,5 @@ +#EXTM3U + +#EXT-X-STREAM-INF:BANDWIDTH=263851,CODECS="mp4a.40.2, avc1.4d400d" +416x243_prog_index_5s.m3u8 + diff --git a/dom/media/test/hls/bipbop_4x3_single.m3u8 b/dom/media/test/hls/bipbop_4x3_single.m3u8 new file mode 100644 index 0000000000..8f354ff011 --- /dev/null +++ b/dom/media/test/hls/bipbop_4x3_single.m3u8 @@ -0,0 +1,4 @@ +#EXTM3U + +#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=232370,CODECS="mp4a.40.2, avc1.4d4015" +400x300_prog_index_5s.m3u8
\ No newline at end of file diff --git a/dom/media/test/hls/bipbop_4x3_variant.m3u8 b/dom/media/test/hls/bipbop_4x3_variant.m3u8 new file mode 100644 index 0000000000..8a9a100dba --- /dev/null +++ b/dom/media/test/hls/bipbop_4x3_variant.m3u8 @@ -0,0 +1,10 @@ +#EXTM3U + +#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=232370,CODECS="mp4a.40.2, avc1.4d4015" +400x300_prog_index.m3u8 + +#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=649879,CODECS="mp4a.40.2, avc1.4d401e" +640x480_prog_index.m3u8 + +#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=649879,CODECS="mp4a.40.2, avc1.4d401e" +960x720_prog_index.m3u8 diff --git a/dom/media/test/huge-id3.mp3 b/dom/media/test/huge-id3.mp3 Binary files differnew file mode 100644 index 0000000000..41cb93d805 --- /dev/null +++ b/dom/media/test/huge-id3.mp3 diff --git a/dom/media/test/huge-id3.mp3^headers^ b/dom/media/test/huge-id3.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/huge-id3.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/id3tags.mp3 b/dom/media/test/id3tags.mp3 Binary files differnew file mode 100644 index 0000000000..bad506cf18 --- /dev/null +++ b/dom/media/test/id3tags.mp3 diff --git a/dom/media/test/id3tags.mp3^headers^ b/dom/media/test/id3tags.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/id3tags.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-cmap-s0c0.opus b/dom/media/test/invalid-cmap-s0c0.opus Binary files differnew file mode 100644 index 0000000000..0b99587865 --- /dev/null +++ b/dom/media/test/invalid-cmap-s0c0.opus diff --git a/dom/media/test/invalid-cmap-s0c0.opus^headers^ b/dom/media/test/invalid-cmap-s0c0.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-cmap-s0c0.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-cmap-s0c2.opus b/dom/media/test/invalid-cmap-s0c2.opus Binary files differnew file mode 100644 index 0000000000..a921894fee --- /dev/null +++ b/dom/media/test/invalid-cmap-s0c2.opus diff --git a/dom/media/test/invalid-cmap-s0c2.opus^headers^ b/dom/media/test/invalid-cmap-s0c2.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-cmap-s0c2.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-cmap-s1c2.opus b/dom/media/test/invalid-cmap-s1c2.opus Binary files differnew file mode 100644 index 0000000000..95a84f523c --- /dev/null +++ b/dom/media/test/invalid-cmap-s1c2.opus diff --git a/dom/media/test/invalid-cmap-s1c2.opus^headers^ b/dom/media/test/invalid-cmap-s1c2.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-cmap-s1c2.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-cmap-short.opus b/dom/media/test/invalid-cmap-short.opus Binary files differnew file mode 100644 index 0000000000..fcd7eb506a --- /dev/null +++ b/dom/media/test/invalid-cmap-short.opus diff --git a/dom/media/test/invalid-cmap-short.opus^headers^ b/dom/media/test/invalid-cmap-short.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-cmap-short.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-discard_on_multi_blocks.webm b/dom/media/test/invalid-discard_on_multi_blocks.webm Binary files differnew file mode 100644 index 0000000000..f39ab5bcb8 --- /dev/null +++ b/dom/media/test/invalid-discard_on_multi_blocks.webm diff --git a/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^ b/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-excess_discard.webm b/dom/media/test/invalid-excess_discard.webm Binary files differnew file mode 100644 index 0000000000..5b34aca1a7 --- /dev/null +++ b/dom/media/test/invalid-excess_discard.webm diff --git a/dom/media/test/invalid-excess_discard.webm^headers^ b/dom/media/test/invalid-excess_discard.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-excess_discard.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-excess_neg_discard.webm b/dom/media/test/invalid-excess_neg_discard.webm Binary files differnew file mode 100644 index 0000000000..2bfad6ed21 --- /dev/null +++ b/dom/media/test/invalid-excess_neg_discard.webm diff --git a/dom/media/test/invalid-excess_neg_discard.webm^headers^ b/dom/media/test/invalid-excess_neg_discard.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-excess_neg_discard.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m0c0.opus b/dom/media/test/invalid-m0c0.opus Binary files differnew file mode 100644 index 0000000000..86555f60eb --- /dev/null +++ b/dom/media/test/invalid-m0c0.opus diff --git a/dom/media/test/invalid-m0c0.opus^headers^ b/dom/media/test/invalid-m0c0.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m0c0.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m0c3.opus b/dom/media/test/invalid-m0c3.opus Binary files differnew file mode 100644 index 0000000000..2c681a8c03 --- /dev/null +++ b/dom/media/test/invalid-m0c3.opus diff --git a/dom/media/test/invalid-m0c3.opus^headers^ b/dom/media/test/invalid-m0c3.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m0c3.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m1c0.opus b/dom/media/test/invalid-m1c0.opus Binary files differnew file mode 100644 index 0000000000..c7728f79ee --- /dev/null +++ b/dom/media/test/invalid-m1c0.opus diff --git a/dom/media/test/invalid-m1c0.opus^headers^ b/dom/media/test/invalid-m1c0.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m1c0.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m1c9.opus b/dom/media/test/invalid-m1c9.opus Binary files differnew file mode 100644 index 0000000000..1ef6e9f9cf --- /dev/null +++ b/dom/media/test/invalid-m1c9.opus diff --git a/dom/media/test/invalid-m1c9.opus^headers^ b/dom/media/test/invalid-m1c9.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m1c9.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m2c0.opus b/dom/media/test/invalid-m2c0.opus Binary files differnew file mode 100644 index 0000000000..5c3f97e2ab --- /dev/null +++ b/dom/media/test/invalid-m2c0.opus diff --git a/dom/media/test/invalid-m2c0.opus^headers^ b/dom/media/test/invalid-m2c0.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m2c0.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-m2c1.opus b/dom/media/test/invalid-m2c1.opus Binary files differnew file mode 100644 index 0000000000..5ecb95ee25 --- /dev/null +++ b/dom/media/test/invalid-m2c1.opus diff --git a/dom/media/test/invalid-m2c1.opus^headers^ b/dom/media/test/invalid-m2c1.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-m2c1.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-neg_discard.webm b/dom/media/test/invalid-neg_discard.webm Binary files differnew file mode 100644 index 0000000000..3f665c0b59 --- /dev/null +++ b/dom/media/test/invalid-neg_discard.webm diff --git a/dom/media/test/invalid-neg_discard.webm^headers^ b/dom/media/test/invalid-neg_discard.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-neg_discard.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/invalid-preskip.webm b/dom/media/test/invalid-preskip.webm Binary files differnew file mode 100644 index 0000000000..99b4f2ca71 --- /dev/null +++ b/dom/media/test/invalid-preskip.webm diff --git a/dom/media/test/invalid-preskip.webm^headers^ b/dom/media/test/invalid-preskip.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/invalid-preskip.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/make-headers.sh b/dom/media/test/make-headers.sh new file mode 100644 index 0000000000..35d9bd90f8 --- /dev/null +++ b/dom/media/test/make-headers.sh @@ -0,0 +1,18 @@ +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +# Script to generate ^header^ files for all media files we use. +# This is to ensure that our media files are not cached by necko, +# so that our detection as to whether the server supports byte range +# requests is not interferred with by Necko's cache. See bug 977398 +# for details. Necko will fix this in bug 977314. + +FILES=(`ls *.ogg *.ogv *.webm *.mp3 *.opus *.mp4 *.m4s *.wav`) + +rm -f *.ogg^headers^ *.ogv^headers^ *.webm^headers^ *.mp3^headers^ *.opus^headers^ *.mp4^headers^ *.m4s^headers^ *.wav^headers^ + +for i in "${FILES[@]}" +do + echo "Cache-Control: no-store" >> $i^headers^ +done diff --git a/dom/media/test/manifest.js b/dom/media/test/manifest.js new file mode 100644 index 0000000000..0d8d5b9300 --- /dev/null +++ b/dom/media/test/manifest.js @@ -0,0 +1,2266 @@ +// In each list of tests below, test file types that are not supported should +// be ignored. To make sure tests respect that, we include a file of type +// "bogus/duh" in each list. + +// Make sure to not touch navigator in here, since we want to push prefs that +// will affect the APIs it exposes, but the set of exposed APIs is determined +// when Navigator.prototype is created. So if we touch navigator before pushing +// the prefs, the APIs it exposes will not take those prefs into account. We +// work around this by using a navigator object from a different global for our +// UA string testing. +var gManifestNavigatorSource = document.documentElement.appendChild( + document.createElement("iframe") +); +gManifestNavigatorSource.style.display = "none"; +function manifestNavigator() { + return gManifestNavigatorSource.contentWindow.navigator; +} + +// Similarly, use a <video> element from a different global for canPlayType or +// other feature testing. If we used one from our global and did so before our +// prefs are pushed, then we'd instantiate HTMLMediaElement.prototype before the +// prefs are pushed and APIs we expect to be on that object would not be there. +function manifestVideo() { + return gManifestNavigatorSource.contentDocument.createElement("video"); +} + +// Need to get the server url composed with ip:port instead of mochi.test. +// Since we will provide the url to Exoplayer which cannot recognize the domain +// name "mochi.test". +let serverUrl = SpecialPowers.Services.prefs.getCharPref( + "media.hls.server.url" +); +var gHLSTests = [ + { + name: serverUrl + "/bipbop_4x3_variant.m3u8", + type: "audio/x-mpegurl", + duration: 20.0, + }, +]; + +// These are small test files, good for just seeing if something loads. We +// really only need one test file per backend here. +var gSmallTests = [ + { name: "small-shot.ogg", type: "audio/ogg", duration: 0.276 }, + { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 }, + { name: "small-shot.mp3", type: "audio/mpeg", duration: 0.27 }, + { name: "small-shot-mp3.mp4", type: "audio/mp4; codecs=mp3", duration: 0.34 }, + { name: "small-shot.flac", type: "audio/flac", duration: 0.197 }, + { name: "r11025_s16_c1-short.wav", type: "audio/x-wav", duration: 0.37 }, + { + name: "320x240.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 0.266, + contentDuration: 0.133, + }, + { + name: "seek-short.webm", + type: "video/webm", + width: 320, + height: 240, + duration: 0.23, + }, + { + name: "vp9-short.webm", + type: "video/webm", + width: 320, + height: 240, + duration: 0.2, + }, + { + name: "detodos-short.opus", + type: "audio/ogg; codecs=opus", + duration: 0.22, + }, + { + name: "gizmo-short.mp4", + type: "video/mp4", + width: 560, + height: 320, + duration: 0.27, + }, + { name: "flac-s24.flac", type: "audio/flac", duration: 4.04 }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +var gFrameCountTests = [ + { name: "bipbop.mp4", type: "video/mp4", totalFrameCount: 297 }, + { name: "gizmo.mp4", type: "video/mp4", totalFrameCount: 166 }, + { name: "seek-short.webm", type: "video/webm", totalFrameCount: 8 }, + { name: "seek.webm", type: "video/webm", totalFrameCount: 120 }, + { name: "320x240.ogv", type: "video/ogg", totalFrameCount: 8 }, + { name: "av1.mp4", type: "video/mp4", totalFrameCount: 24 }, +]; + +gSmallTests = gSmallTests.concat([ + { name: "sample.3gp", type: "video/3gpp", duration: 4.933 }, + { name: "sample.3g2", type: "video/3gpp2", duration: 4.933 }, +]); + +// Used by test_bug654550.html, for videoStats preference +var gVideoTests = [ + { + name: "320x240.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 0.266, + }, + { + name: "seek-short.webm", + type: "video/webm", + width: 320, + height: 240, + duration: 0.23, + }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// Temp hack for trackIDs and captureStream() -- bug 1215769 +var gLongerTests = [ + { + name: "seek.webm", + type: "video/webm", + width: 320, + height: 240, + duration: 3.966, + }, + { + name: "gizmo.mp4", + type: "video/mp4", + width: 560, + height: 320, + duration: 5.56, + }, +]; + +// Used by test_progress to ensure we get the correct progress information +// during resource download. +var gProgressTests = [ + { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0, size: 11069 }, + { name: "big-short.wav", type: "audio/x-wav", duration: 1.11, size: 12366 }, + { name: "seek-short.ogv", type: "video/ogg", duration: 1.03, size: 79921 }, + { + name: "320x240.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 0.266, + size: 28942, + }, + { name: "seek-short.webm", type: "video/webm", duration: 0.23, size: 19267 }, + { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27, size: 29905 }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// Used by test_played.html +var gPlayedTests = [ + { name: "big-short.wav", type: "audio/x-wav", duration: 1.11 }, + { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 }, + { name: "seek-short.webm", type: "video/webm", duration: 0.23 }, + { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27 }, + { name: "owl-short.mp3", type: "audio/mpeg", duration: 0.52 }, + { name: "very-short.mp3", type: "audio/mpeg", duration: 0.07 }, + // Disable vbr.mp3 to see if it reduces the error of AUDCLNT_E_CPUUSAGE_EXCEEDED. + // See bug 1110922 comment 26. + //{ name:"vbr.mp3", type:"audio/mpeg", duration:10.0 }, + { name: "bug495794.ogg", type: "audio/ogg", duration: 0.3 }, +]; + +if ( + manifestNavigator().userAgent.includes("Windows") && + manifestVideo().canPlayType('video/mp4; codecs="avc1.42E01E"') +) { + gPlayedTests = gPlayedTests.concat( + { name: "red-46x48.mp4", type: "video/mp4", duration: 1.0 }, + { name: "red-48x46.mp4", type: "video/mp4", duration: 1.0 } + ); +} + +// Used by test_mozLoadFrom. Need one test file per decoder backend, plus +// anything for testing clone-specific bugs. +var cloneKey = Math.floor(Math.random() * 100000000); +var gCloneTests = [ + // short-video is more like 1s, so if you load this twice you'll get an unexpected duration + { + name: + "dynamic_resource.sjs?key=" + + cloneKey + + "&res1=320x240.ogv&res2=short-video.ogv", + type: "video/ogg", + duration: 0.266, + }, +]; + +// Used by test_play_twice. Need one test file per decoder backend, plus +// anything for testing bugs that occur when replying a played file. +var gReplayTests = gSmallTests.concat([ + { name: "bug533822.ogg", type: "audio/ogg" }, +]); + +// Used by test_paused_after_ended. Need one test file per decoder backend, plus +// anything for testing bugs that occur when replying a played file. +var gPausedAfterEndedTests = gSmallTests.concat([ + { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0 }, + { name: "small-shot.ogg", type: "video/ogg", duration: 0.276 }, +]); + +// Test the mozHasAudio property, and APIs that detect different kinds of +// tracks +var gTrackTests = [ + { + name: "big-short.wav", + type: "audio/x-wav", + duration: 1.11, + size: 12366, + hasAudio: true, + hasVideo: false, + }, + { + name: "320x240.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 0.266, + size: 28942, + hasAudio: false, + hasVideo: true, + }, + { + name: "short-video.ogv", + type: "video/ogg", + duration: 1.081, + hasAudio: true, + hasVideo: true, + }, + { + name: "seek-short.webm", + type: "video/webm", + duration: 0.23, + size: 19267, + hasAudio: false, + hasVideo: true, + }, + { + name: "flac-s24.flac", + type: "audio/flac", + duration: 4.04, + hasAudio: true, + hasVideo: false, + }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +var gClosingConnectionsTest = [ + { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 }, +]; + +// Used by any media recorder test. Need one test file per decoder backend +// currently supported by the media encoder. +var gMediaRecorderTests = [ + // Duration should be greater than 500ms because we will record 2 + // time slices (250ms per slice) + { + name: "detodos-recorder-test.opus", + type: "audio/ogg; codecs=opus", + duration: 0.62, + }, +]; + +// Used by video media recorder tests +var gMediaRecorderVideoTests = [ + { + name: "seek-short.webm", + type: "video/webm", + width: 320, + height: 240, + duration: 0.23, + }, +]; + +// These are files that we want to make sure we can play through. We can +// also check metadata. Put files of the same type together in this list so if +// something crashes we have some idea of which backend is responsible. +// Used by test_playback, which expects no error event and one ended event. +var gPlayTests = [ + // Test playback of a WebM file with vp9 video + { name: "vp9cake-short.webm", type: "video/webm", duration: 1.0 }, + // 8-bit samples + { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0 }, + // 8-bit samples, file is truncated + { name: "r11025_u8_c1_trunc.wav", type: "audio/x-wav", duration: 1.8 }, + // file has trailing non-PCM data + { name: "r11025_s16_c1_trailing.wav", type: "audio/x-wav", duration: 1.0 }, + // file with list chunk + { name: "r16000_u8_c1_list.wav", type: "audio/x-wav", duration: 4.2 }, + // file with 2 extra bytes of metadata + { + name: "16bit_wave_extrametadata.wav", + type: "audio/x-wav", + duration: 1.108, + }, + // IEEE float wave file + { name: "wavedata_float.wav", type: "audio/x-wav", duration: 1.0 }, + // 24-bit samples + { name: "wavedata_s24.wav", type: "audio/x-wav", duration: 1.0 }, + // aLaw compressed wave file + { name: "wavedata_alaw.wav", type: "audio/x-wav", duration: 1.0 }, + // uLaw compressed wave file + { name: "wavedata_ulaw.wav", type: "audio/x-wav", duration: 1.0 }, + // Data length 0xFFFFFFFF + { name: "bug1301226.wav", type: "audio/x-wav", duration: 0.003673 }, + // Data length 0xFFFFFFFF and odd chunk lengths. + { name: "bug1301226-odd.wav", type: "audio/x-wav", duration: 0.003673 }, + + // Ogg stream without eof marker + { name: "bug461281.ogg", type: "application/ogg", duration: 2.208 }, + + // oggz-chop stream + { name: "bug482461.ogv", type: "video/ogg", duration: 4.34 }, + // Theora only oggz-chop stream + { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 }, + // With first frame a "duplicate" (empty) frame. + { + name: "bug500311.ogv", + type: "video/ogg", + duration: 1.96, + contentDuration: 1.958, + }, + // Small audio file + { name: "small-shot.ogg", type: "audio/ogg", duration: 0.276 }, + // More audio in file than video. + { name: "short-video.ogv", type: "video/ogg", duration: 1.081 }, + // First Theora data packet is zero bytes. + { name: "bug504613.ogv", type: "video/ogg", duration: Number.NaN }, + // Multiple audio streams. + { name: "bug516323.ogv", type: "video/ogg", duration: 4.208 }, + // oggz-chop with non-keyframe as first frame + { + name: "bug556821.ogv", + type: "video/ogg", + duration: 2.936, + contentDuration: 2.903, + }, + + // Encoded with vorbis beta1, includes unusually sized codebooks + { name: "beta-phrasebook.ogg", type: "audio/ogg", duration: 4.01 }, + // Small file, only 1 frame with audio only. + { name: "bug520493.ogg", type: "audio/ogg", duration: 0.458 }, + // Small file with vorbis comments with 0 length values and names. + { name: "bug520500.ogg", type: "audio/ogg", duration: 0.123 }, + + // Various weirdly formed Ogg files + { + name: "bug499519.ogv", + type: "video/ogg", + duration: 0.24, + contentDuration: 0.22, + }, + { name: "bug506094.ogv", type: "video/ogg", duration: 0 }, + { name: "bug498855-1.ogv", type: "video/ogg", duration: 0.24 }, + { name: "bug498855-2.ogv", type: "video/ogg", duration: 0.24 }, + { name: "bug498855-3.ogv", type: "video/ogg", duration: 0.24 }, + { + name: "bug504644.ogv", + type: "video/ogg", + duration: 1.6, + contentDuration: 1.52, + }, + { + name: "chain.ogv", + type: "video/ogg", + duration: Number.NaN, + contentDuration: 0.266, + }, + { + name: "bug523816.ogv", + type: "video/ogg", + duration: 0.766, + contentDuration: 0, + }, + { name: "bug495129.ogv", type: "video/ogg", duration: 2.41 }, + { + name: "bug498380.ogv", + type: "video/ogg", + duration: 0.7663, + contentDuration: 0, + }, + { name: "bug495794.ogg", type: "audio/ogg", duration: 0.3 }, + { name: "bug557094.ogv", type: "video/ogg", duration: 0.24 }, + { name: "multiple-bos.ogg", type: "video/ogg", duration: 0.431 }, + { name: "audio-overhang.ogg", type: "video/ogg", duration: 2.3 }, + { name: "video-overhang.ogg", type: "video/ogg", duration: 3.966 }, + + // bug461281.ogg with the middle second chopped out. + { name: "audio-gaps.ogg", type: "audio/ogg", duration: 2.208 }, + + // Test playback/metadata work after a redirect + { + name: "redirect.sjs?domain=mochi.test:8888&file=320x240.ogv", + type: "video/ogg", + duration: 0.266, + }, + + // Test playback of a webm file + { name: "seek-short.webm", type: "video/webm", duration: 0.23 }, + + // Test playback of a webm file with 'matroska' doctype + { name: "bug1377278.webm", type: "video/webm", duration: 4.0 }, + + // Test playback of a WebM file with non-zero start time. + { name: "split.webm", type: "video/webm", duration: 1.967 }, + + // Test playback of a WebM file with resolution changes. + { name: "resolution-change.webm", type: "video/webm", duration: 6.533 }, + + // A really short, low sample rate, single channel file. This tests whether + // we can handle playing files when only push very little audio data to the + // hardware. + { name: "spacestorm-1000Hz-100ms.ogg", type: "audio/ogg", duration: 0.099 }, + + // Opus data in an ogg container + { + name: "detodos-short.opus", + type: "audio/ogg; codecs=opus", + duration: 0.22, + contentDuration: 0.2135, + }, + // Opus data in a webm container + { + name: "detodos-short.webm", + type: "audio/webm; codecs=opus", + duration: 0.26, + contentDuration: 0.2535, + }, + // Opus in webm channel mapping=2 sample file + { + name: "opus-mapping2.webm", + type: "audio/webm; codecs=opus", + duration: 10.01, + contentDuration: 9.99, + }, + { name: "bug1066943.webm", type: "audio/webm; codecs=opus", duration: 1.383 }, + + // Multichannel Opus in an ogg container + { name: "test-1-mono.opus", type: "audio/ogg; codecs=opus", duration: 1.044 }, + { + name: "test-2-stereo.opus", + type: "audio/ogg; codecs=opus", + duration: 2.925, + }, + { name: "test-3-LCR.opus", type: "audio/ogg; codecs=opus", duration: 4.214 }, + { name: "test-4-quad.opus", type: "audio/ogg; codecs=opus", duration: 6.234 }, + { name: "test-5-5.0.opus", type: "audio/ogg; codecs=opus", duration: 7.558 }, + { name: "test-6-5.1.opus", type: "audio/ogg; codecs=opus", duration: 10.333 }, + { name: "test-7-6.1.opus", type: "audio/ogg; codecs=opus", duration: 11.69 }, + { name: "test-8-7.1.opus", type: "audio/ogg; codecs=opus", duration: 13.478 }, + + { + name: "gizmo-short.mp4", + type: "video/mp4", + duration: 0.27, + contentDuration: 0.267, + }, + // Test playback of a MP4 file with a non-zero start time (and audio starting + // a second later). + { name: "bipbop-lateaudio.mp4", type: "video/mp4" }, + // Ambisonics AAC, requires AAC extradata to be set when creating decoder (see bug 1431169) + // Also test 4.0 decoding. + { name: "ambisonics.mp4", type: "audio/mp4", duration: 16.48 }, + // Opus in MP4 channel mapping=0 sample file (content shorter due to preskip) + { + name: "opus-sample.mp4", + type: "audio/mp4; codecs=opus", + duration: 10.92, + contentDuration: 10.09, + }, + // Opus in MP4 channel mapping=2 sample file + { name: "opus-mapping2.mp4", type: "audio/mp4; codecs=opus", duration: 10.0 }, + + { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 }, + { name: "small-shot.mp3", type: "audio/mpeg", duration: 0.27 }, + { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 }, + // owl.mp3 as above, but with something funny going on in the ID3v2 tag + // that caused DirectShow to fail. + { name: "owl-funny-id3.mp3", type: "audio/mpeg", duration: 3.343 }, + // owl.mp3 as above, but with something even funnier going on in the ID3v2 tag + // that caused DirectShow to fail. + { name: "owl-funnier-id3.mp3", type: "audio/mpeg", duration: 3.343 }, + // One second of silence with ~140KB of ID3 tags. Usually when the first MP3 + // frame is at such a high offset into the file, MP3FrameParser will give up + // and report that the stream is not MP3. However, it does not count ID3 tags + // in that offset. This test case makes sure that ID3 exclusion holds. + { name: "huge-id3.mp3", type: "audio/mpeg", duration: 1.0 }, + // A truncated VBR MP3 with just enough frames to keep most decoders happy. + // The Xing header reports the length of the file to be around 10 seconds, but + // there is really only one second worth of data. We want MP3FrameParser to + // trust the header, so this should be reported as 10 seconds. + { + name: "vbr-head.mp3", + type: "audio/mpeg", + duration: 10.0, + contentDuration: 1.019, + }, + + // A flac file where the STREAMINFO block was removed. + // It is necessary to parse the file to find an audio frame instead. + { name: "flac-noheader-s16.flac", type: "audio/flac", duration: 4.0 }, + { name: "flac-s24.flac", type: "audio/flac", duration: 4.04 }, + { + name: "flac-sample.mp4", + type: "audio/mp4; codecs=flac", + duration: 4.95, + contentDuration: 5.03, + }, + // Ogg with theora video and flac audio. + { + name: "A4.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 3.13, + }, + + // Invalid file + { name: "bogus.duh", type: "bogus/duh", duration: Number.NaN }, +]; + +const win32 = + SpecialPowers.Services.appinfo.OS == "WINNT" && + !SpecialPowers.Services.appinfo.is64Bit; +if (!win32) { + gPlayTests.push({ name: "av1.mp4", type: "video/mp4", duration: 1.0 }); +} + +var gSeekToNextFrameTests = [ + // Test playback of a WebM file with vp9 video + { name: "vp9-short.webm", type: "video/webm", duration: 0.2 }, + { name: "vp9cake-short.webm", type: "video/webm", duration: 1.0 }, + // oggz-chop stream + { name: "bug482461.ogv", type: "video/ogg", duration: 4.34 }, + // Theora only oggz-chop stream + { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 }, + // With first frame a "duplicate" (empty) frame. + { name: "bug500311.ogv", type: "video/ogg", duration: 1.96 }, + + // More audio in file than video. + { name: "short-video.ogv", type: "video/ogg", duration: 1.081 }, + // First Theora data packet is zero bytes. + { name: "bug504613.ogv", type: "video/ogg", duration: Number.NaN }, + // Multiple audio streams. + { name: "bug516323.ogv", type: "video/ogg", duration: 4.208 }, + // oggz-chop with non-keyframe as first frame + { name: "bug556821.ogv", type: "video/ogg", duration: 2.936 }, + // Various weirdly formed Ogg files + { name: "bug498855-1.ogv", type: "video/ogg", duration: 0.24 }, + { name: "bug498855-2.ogv", type: "video/ogg", duration: 0.24 }, + { name: "bug498855-3.ogv", type: "video/ogg", duration: 0.24 }, + { name: "bug504644.ogv", type: "video/ogg", duration: 1.6 }, + + { name: "bug523816.ogv", type: "video/ogg", duration: 0.766 }, + + { name: "bug498380.ogv", type: "video/ogg", duration: 0.2 }, + { name: "bug557094.ogv", type: "video/ogg", duration: 0.24 }, + { name: "multiple-bos.ogg", type: "video/ogg", duration: 0.431 }, + // Test playback/metadata work after a redirect + { + name: "redirect.sjs?domain=mochi.test:8888&file=320x240.ogv", + type: "video/ogg", + duration: 0.266, + }, + // Test playback of a webm file + { name: "seek-short.webm", type: "video/webm", duration: 0.23 }, + // Test playback of a WebM file with non-zero start time. + { name: "split.webm", type: "video/webm", duration: 1.967 }, + + { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27 }, + + // Test playback of a MP4 file with a non-zero start time (and audio starting + // a second later). + { name: "bipbop-lateaudio.mp4", type: "video/mp4" }, +]; + +// A file for each type we can support. +var gSnifferTests = [ + { name: "big.wav", type: "audio/x-wav", duration: 9.278982, size: 102444 }, + { + name: "320x240.ogv", + type: "video/ogg", + width: 320, + height: 240, + duration: 0.233, + size: 28942, + }, + { name: "seek.webm", type: "video/webm", duration: 3.966, size: 215529 }, + { name: "gizmo.mp4", type: "video/mp4", duration: 5.56, size: 383631 }, + // A mp3 file with id3 tags. + { name: "id3tags.mp3", type: "audio/mpeg", duration: 0.28, size: 3530 }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// Files that contain resolution changes +var gResolutionChangeTests = [ + { name: "resolution-change.webm", type: "video/webm", duration: 6.533 }, +]; + +// Files we must reject as invalid. +var gInvalidTests = [ + { name: "invalid-m0c0.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-m0c3.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-m1c0.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-m1c9.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-m2c0.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-m2c1.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-cmap-short.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-cmap-s0c0.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-cmap-s0c2.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-cmap-s1c2.opus", type: "audio/ogg; codecs=opus" }, + { name: "invalid-preskip.webm", type: "audio/webm; codecs=opus" }, +]; + +var gInvalidPlayTests = [ + { name: "invalid-excess_discard.webm", type: "audio/webm; codecs=opus" }, + { name: "invalid-excess_neg_discard.webm", type: "audio/webm; codecs=opus" }, + { name: "invalid-neg_discard.webm", type: "audio/webm; codecs=opus" }, + { + name: "invalid-discard_on_multi_blocks.webm", + type: "audio/webm; codecs=opus", + }, +]; + +// Files to check different cases of ogg skeleton information. +// sample-fisbone-skeleton4.ogv +// - Skeleton v4, w/ Content-Type,Role,Name,Language,Title for both theora/vorbis +// sample-fisbone-wrong-header.ogv +// - Skeleton v4, wrong message field sequence for vorbis +// multiple-bos-more-header-fields.ogg +// - Skeleton v3, w/ Content-Type,Role,Name,Language,Title for both theora/vorbis +// seek-short.ogv +// - No skeleton, but theora +// audio-gaps-short.ogg +// - No skeleton, but vorbis +var gMultitrackInfoOggPlayList = [ + { name: "sample-fisbone-skeleton4.ogv", type: "video/ogg", duration: 1.0 }, + { name: "sample-fisbone-wrong-header.ogv", type: "video/ogg", duration: 1.0 }, + { + name: "multiple-bos-more-header-fileds.ogg", + type: "video/ogg", + duration: 0.431, + }, + { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 }, + { name: "audio-gaps-short.ogg", type: "audio/ogg", duration: 0.5 }, +]; +// Pre-parsed results of gMultitrackInfoOggPlayList. +var gOggTrackInfoResults = { + "sample-fisbone-skeleton4.ogv": { + audio_id: " audio_1", + audio_kind: "main", + audio_language: " en-US", + audio_label: " Audio track for test", + video_id: " video_1", + video_kind: "main", + video_language: " fr", + video_label: " Video track for test", + }, + "sample-fisbone-wrong-header.ogv": { + audio_id: "1", + audio_kind: "main", + audio_language: "", + audio_label: "", + video_id: " video_1", + video_kind: "main", + video_language: " fr", + video_label: " Video track for test", + }, + "multiple-bos-more-header-fileds.ogg": { + audio_id: "1", + audio_kind: "main", + audio_language: "", + audio_label: "", + video_id: "2", + video_kind: "main", + video_language: "", + video_label: "", + }, + "seek-short.ogv": { + video_id: "2", + video_kind: "main", + video_language: "", + video_label: "", + }, + "audio-gaps-short.ogg": { + audio_id: "1", + audio_kind: "main", + audio_language: "", + audio_label: "", + }, +}; + +// Returns a promise that resolves to a function that converts +// relative paths to absolute, to test loading files from file: URIs. +// Optionally checks whether the file actually exists on disk at the location +// we've specified. +function makeAbsolutePathConverter() { + const url = SimpleTest.getTestFileURL("chromeHelper.js"); + const script = SpecialPowers.loadChromeScript(url); + return new Promise((resolve, reject) => { + script.addMessageListener("media-test:cwd", cwd => { + if (!cwd) { + ok(false, "Failed to find path to test files"); + } + + resolve((path, mustExist) => { + // android mochitest doesn't support file:// + if (manifestNavigator().appVersion.includes("Android")) { + return path; + } + + const { Ci, Cc } = SpecialPowers; + var f = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile); + f.initWithPath(cwd); + var split = path.split("/"); + for (var i = 0; i < split.length; ++i) { + f.append(split[i]); + } + if (mustExist && !f.exists()) { + ok(false, "We expected '" + path + "' to exist, but it doesn't!"); + } + return f.path; + }); + }); + script.sendAsyncMessage("media-test:getcwd"); + }); +} + +// Returns true if two TimeRanges are equal, false otherwise +function range_equals(r1, r2) { + if (r1.length != r2.length) { + return false; + } + for (var i = 0; i < r1.length; i++) { + if (r1.start(i) != r2.start(i) || r1.end(i) != r2.end(i)) { + return false; + } + } + return true; +} + +// These are URIs to files that we use to check that we don't leak any state +// or other information such that script can determine stuff about a user's +// environment. Used by test_info_leak. +function makeInfoLeakTests() { + return makeAbsolutePathConverter().then(fileUriToSrc => [ + { + type: "video/ogg", + src: fileUriToSrc("tests/dom/media/test/320x240.ogv", true), + }, + { + type: "video/ogg", + src: fileUriToSrc("tests/dom/media/test/404.ogv", false), + }, + { + type: "audio/x-wav", + src: fileUriToSrc("tests/dom/media/test/r11025_s16_c1.wav", true), + }, + { + type: "audio/x-wav", + src: fileUriToSrc("tests/dom/media/test/404.wav", false), + }, + { + type: "audio/ogg", + src: fileUriToSrc("tests/dom/media/test/bug461281.ogg", true), + }, + { + type: "audio/ogg", + src: fileUriToSrc("tests/dom/media/test/404.ogg", false), + }, + { + type: "video/webm", + src: fileUriToSrc("tests/dom/media/test/seek.webm", true), + }, + { + type: "video/webm", + src: fileUriToSrc("tests/dom/media/test/404.webm", false), + }, + { + type: "video/ogg", + src: "http://localhost/404.ogv", + }, + { + type: "audio/x-wav", + src: "http://localhost/404.wav", + }, + { + type: "video/webm", + src: "http://localhost/404.webm", + }, + { + type: "video/ogg", + src: "http://example.com/tests/dom/media/test/test_info_leak.html", + }, + { + type: "audio/ogg", + src: "http://example.com/tests/dom/media/test/test_info_leak.html", + }, + ]); +} + +// These are files that must fire an error during load or playback, and do not +// cause a crash. Used by test_playback_errors, which expects one error event +// and no ended event. Put files of the same type together in this list so if +// something crashes we have some idea of which backend is responsible. +var gErrorTests = [ + { name: "bogus.wav", type: "audio/x-wav" }, + { name: "bogus.ogv", type: "video/ogg" }, + { name: "448636.ogv", type: "video/ogg" }, + { name: "bug504843.ogv", type: "video/ogg" }, + { name: "bug501279.ogg", type: "audio/ogg" }, + { name: "bug603918.webm", type: "video/webm" }, + { name: "bug604067.webm", type: "video/webm" }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// These files would get error after receiving "loadedmetadata", we would like +// to check duration in "onerror" and make sure the duration is still available. +var gDurationTests = [ + { name: "bug603918.webm", duration: 6.076 }, + { name: "bug604067.webm", duration: 6.076 }, +]; + +// These are files that have nontrivial duration and are useful for seeking within. +var gSeekTests = [ + { name: "r11025_s16_c1.wav", type: "audio/x-wav", duration: 1.0 }, + { name: "audio.wav", type: "audio/x-wav", duration: 0.031247 }, + { name: "seek.ogv", type: "video/ogg", duration: 3.966 }, + { name: "320x240.ogv", type: "video/ogg", duration: 0.266 }, + { name: "seek.webm", type: "video/webm", duration: 3.966 }, + { name: "sine.webm", type: "audio/webm", duration: 4.001 }, + { name: "bug516323.indexed.ogv", type: "video/ogg", duration: 4.208333 }, + { name: "split.webm", type: "video/webm", duration: 1.967 }, + { name: "detodos.opus", type: "audio/ogg; codecs=opus", duration: 2.9135 }, + { name: "gizmo.mp4", type: "video/mp4", duration: 5.56 }, + { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 }, + { name: "bogus.duh", type: "bogus/duh", duration: 123 }, + + // Bug 1242338: hit a numerical problem while seeking to the duration. + { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 }, +]; + +var gFastSeekTests = [ + { + name: "gizmo.mp4", + type: "video/mp4", + keyframes: [0, 1.0, 2.0, 3.0, 4.0, 5.0], + }, + // Note: Not all keyframes in the file are actually referenced in the Cues in this file. + { name: "seek.webm", type: "video/webm", keyframes: [0, 0.8, 1.6, 2.4, 3.2] }, + // Note: the sync points are the points on both the audio and video streams + // before the keyframes. You can't just assume that the keyframes are the sync + // points, as the audio required for that sync point may be before the keyframe. + { + name: "bug516323.indexed.ogv", + type: "video/ogg", + keyframes: [0, 0.46, 3.06], + }, +]; + +// These files are WebMs without cues. They're seekable within their buffered +// ranges. If work renders WebMs fully seekable these files should be moved +// into gSeekTests +var gCuelessWebMTests = [ + { name: "no-cues.webm", type: "video/webm", duration: 3.967 }, +]; + +// These are files that are non seekable, due to problems with the media, +// for example broken or missing indexes. +var gUnseekableTests = [{ name: "bogus.duh", type: "bogus/duh" }]; + +var androidVersion = -1; // non-Android platforms +if ( + manifestNavigator().userAgent.includes("Mobile") || + manifestNavigator().userAgent.includes("Tablet") +) { + androidVersion = SpecialPowers.Services.sysinfo.getProperty("version"); +} + +function getAndroidVersion() { + return androidVersion; +} + +// These are files suitable for using with a "new Audio" constructor. +var gAudioTests = [ + { name: "r11025_s16_c1.wav", type: "audio/x-wav", duration: 1.0 }, + { name: "sound.ogg", type: "audio/ogg" }, + { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 }, + { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 }, + { name: "bogus.duh", type: "bogus/duh", duration: 123 }, + { name: "empty_size.mp3", type: "audio/mpeg", duration: 2.235 }, +]; + +// These files ensure our handling of 404 errors is consistent across the +// various backends. +var g404Tests = [ + { name: "404.wav", type: "audio/x-wav" }, + { name: "404.ogv", type: "video/ogg" }, + { name: "404.oga", type: "audio/ogg" }, + { name: "404.webm", type: "video/webm" }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// These are files suitable for testing various decoder failures that are +// expected to fire MEDIA_ERR_DECODE. Used by test_decode_error, which expects +// an error and emptied event, and no loadedmetadata or ended event. +var gDecodeErrorTests = [ + // Valid files with unsupported codecs + { name: "r11025_msadpcm_c1.wav", type: "audio/x-wav" }, + { name: "dirac.ogg", type: "video/ogg" }, + // Invalid files + { name: "bogus.wav", type: "audio/x-wav" }, + { name: "bogus.ogv", type: "video/ogg" }, + + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// These are files that are used for media fragments tests +var gFragmentTests = [ + { name: "big.wav", type: "audio/x-wav", duration: 9.278982, size: 102444 }, +]; + +// Used by test_chaining.html. The |links| attributes is the number of links in +// this file that we should be able to play. +var gChainingTests = [ + // Vorbis and Opus chained file. They have user comments |index=n| where `n` + // is the index of this segment in the file, 0 indexed. + { name: "chain.ogg", type: "audio/ogg", links: 4 }, + { name: "chain.opus", type: "audio/ogg; codec=opus", links: 4 }, + // Those files are chained files with a different number of channels in each + // part. This is not supported and should stop playing after the first part. + { name: "variable-channel.ogg", type: "audio/ogg", links: 1 }, + { name: "variable-channel.opus", type: "audio/ogg; codec=opus", links: 1 }, + // Those files are chained files with a different sample rate in each + // part. This is not supported and should stop playing after the first part. + { name: "variable-samplerate.ogg", type: "audio/ogg", links: 1 }, + // Opus decoding in Firefox outputs 48 kHz PCM despite having a different + // original sample rate, so we can safely play Opus chained media that have + // different samplerate accross links. + { name: "variable-samplerate.opus", type: "audio/ogg; codec=opus", links: 2 }, + // A chained video file. We don't support those, so only one link should be + // reported. + { name: "chained-video.ogv", type: "video/ogg", links: 1 }, + // A file that consist in 4 links of audio, then another link that has video. + // We should stop right after the 4 audio links. + { name: "chained-audio-video.ogg", type: "video/ogg", links: 4 }, + // An opus file that has two links, with a different preskip value for each + // link. We should be able to play both links. + { name: "variable-preskip.opus", type: "audio/ogg; codec=opus", links: 2 }, + { name: "bogus.duh", type: "bogus/duh" }, +]; + +// Videos with an aspect ratio. Used for testing that displaying frames +// on a canvas works correctly in the case of non-standard aspect ratios. +// See bug 874897 for an example. +var gAspectRatioTests = [ + { name: "VID_0001.ogg", type: "video/ogg", duration: 19.966 }, +]; + +// These are files with non-trivial tag sets. +// Used by test_metadata.html. +var gMetadataTests = [ + // Ogg Vorbis files + { + name: "short-video.ogv", + tags: { + TITLE: "Lepidoptera", + ARTIST: "Epoq", + ALBUM: "Kahvi Collective", + DATE: "2002", + COMMENT: "http://www.kahvi.org", + }, + }, + { + name: "bug516323.ogv", + tags: { + GENRE: "Open Movie", + ENCODER: "Audacity", + TITLE: "Elephants Dream", + ARTIST: "Silvia Pfeiffer", + COMMENTS: "Audio Description", + }, + }, + { + name: "bug516323.indexed.ogv", + tags: { + GENRE: "Open Movie", + ENCODER: "Audacity", + TITLE: "Elephants Dream", + ARTIST: "Silvia Pfeiffer", + COMMENTS: "Audio Description", + }, + }, + { + name: "detodos.opus", + tags: { + title: "De todos. Para todos.", + artist: "Mozilla.org", + }, + }, + { name: "sound.ogg", tags: {} }, + { + name: "small-shot.ogg", + tags: { + title: "Pew SFX", + }, + }, + { + name: "badtags.ogg", + tags: { + // We list only the valid tags here, and verify + // the invalid ones are filtered out. + title: "Invalid comments test file", + empty: "", + "": "empty", + "{- [(`!@\"#$%^&')] -}": "valid tag name, surprisingly", + // The file also includes the following invalid tags. + // "A description with no separator is a common problem.", + // "雨":"Likely, but an invalid key (non-ascii).", + // "not\nval\x1fid":"invalid tag name", + // "not~valid":"this isn't a valid name either", + // "not-utf-8":"invalid sequences: \xff\xfe\xfa\xfb\0eol" + }, + }, + { + name: "wave_metadata.wav", + tags: { + name: "Track Title", + artist: "Artist Name", + comments: "Comments", + }, + }, + { + name: "wave_metadata_utf8.wav", + tags: { + name: "歌曲名稱", + artist: "作曲者", + comments: "註解", + }, + }, + { + name: "wave_metadata_unknown_tag.wav", + tags: { + name: "Track Title", + comments: "Comments", + }, + }, + { + name: "wave_metadata_bad_len.wav", + tags: { + name: "Track Title", + artist: "Artist Name", + comments: "Comments", + }, + }, + { + name: "wave_metadata_bad_no_null.wav", + tags: { + name: "Track Title", + artist: "Artist Name", + comments: "Comments!!", + }, + }, + { + name: "wave_metadata_bad_utf8.wav", + tags: { + name: "歌曲名稱", + comments: "註解", + }, + }, + { name: "wavedata_u8.wav", tags: {} }, +]; + +// Now Fennec doesn't support flac, so only test it on non-android platforms. +if (getAndroidVersion() < 0) { + gMetadataTests = gMetadataTests.concat([ + { + name: "flac-s24.flac", + tags: { + ALBUM: "Seascapes", + TITLE: "(La Mer) - II. Jeux de vagues. Allegro", + COMPOSER: "Debussy, Claude", + TRACKNUMBER: "2/9", + DISCNUMBER: "1/1", + encoder: "Lavf57.41.100", + }, + }, + ]); +} + +// Test files for Encrypted Media Extensions +var gEMETests = [ + { + name: "vp9 in mp4", + tracks: [ + { + name: "video", + type: 'video/mp4; codecs="vp9.0"', + fragments: ["short-vp9-encrypted-video.mp4"], + }, + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: ["short-aac-encrypted-audio.mp4"], + }, + ], + keys: { + "2cdb0ed6119853e7850671c3e9906c3c": "808B9ADAC384DE1E4F56140F4AD76194", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 0.47, + }, + { + name: "video-only with 2 keys", + tracks: [ + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop-cenc-videoinit.mp4", + "bipbop-cenc-video1.m4s", + "bipbop-cenc-video2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333", + "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444", + }, + sessionType: "temporary", + sessionCount: 1, + duration: 1.6, + }, + { + name: "video-only with 2 keys, CORS", + tracks: [ + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop-cenc-videoinit.mp4", + "bipbop-cenc-video1.m4s", + "bipbop-cenc-video2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333", + "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444", + }, + sessionType: "temporary", + sessionCount: 1, + crossOrigin: true, + duration: 1.6, + }, + { + name: "audio&video tracks, both with all keys", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop-cenc-audioinit.mp4", + "bipbop-cenc-audio1.m4s", + "bipbop-cenc-audio2.m4s", + "bipbop-cenc-audio3.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop-cenc-videoinit.mp4", + "bipbop-cenc-video1.m4s", + "bipbop-cenc-video2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333", + "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "audio&video tracks, both with all keys, CORS", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop-cenc-audioinit.mp4", + "bipbop-cenc-audio1.m4s", + "bipbop-cenc-audio2.m4s", + "bipbop-cenc-audio3.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop-cenc-videoinit.mp4", + "bipbop-cenc-video1.m4s", + "bipbop-cenc-video2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333", + "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444", + }, + sessionType: "temporary", + sessionCount: 2, + crossOrigin: true, + duration: 1.6, + }, + { + name: "400x300 audio&video tracks, each with its key", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_300_215kbps-cenc-audio-key1-init.mp4", + "bipbop_300_215kbps-cenc-audio-key1-1.m4s", + "bipbop_300_215kbps-cenc-audio-key1-2.m4s", + "bipbop_300_215kbps-cenc-audio-key1-3.m4s", + "bipbop_300_215kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_300_215kbps-cenc-video-key1-init.mp4", + "bipbop_300_215kbps-cenc-video-key1-1.m4s", + "bipbop_300_215kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "640x480@624kbps audio&video tracks, each with its key", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_624kbps-cenc-audio-key1-init.mp4", + "bipbop_480_624kbps-cenc-audio-key1-1.m4s", + "bipbop_480_624kbps-cenc-audio-key1-2.m4s", + "bipbop_480_624kbps-cenc-audio-key1-3.m4s", + "bipbop_480_624kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_624kbps-cenc-video-key1-init.mp4", + "bipbop_480_624kbps-cenc-video-key1-1.m4s", + "bipbop_480_624kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "640x480@959kbps audio&video tracks, each with its key", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_959kbps-cenc-audio-key1-init.mp4", + "bipbop_480_959kbps-cenc-audio-key1-1.m4s", + "bipbop_480_959kbps-cenc-audio-key1-2.m4s", + "bipbop_480_959kbps-cenc-audio-key1-3.m4s", + "bipbop_480_959kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_959kbps-cenc-video-key1-init.mp4", + "bipbop_480_959kbps-cenc-video-key1-1.m4s", + "bipbop_480_959kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "640x480 then 400x300, same key (1st) per track", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_624kbps-cenc-audio-key1-init.mp4", + "bipbop_480_624kbps-cenc-audio-key1-1.m4s", + "bipbop_480_624kbps-cenc-audio-key1-2.m4s", + "bipbop_480_624kbps-cenc-audio-key1-3.m4s", + "bipbop_480_624kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_624kbps-cenc-video-key1-init.mp4", + "bipbop_480_624kbps-cenc-video-key1-1.m4s", + "bipbop_300_215kbps-cenc-video-key1-init.mp4", + "bipbop_300_215kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "640x480 then 400x300, same key (2nd) per track", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_624kbps-cenc-audio-key2-init.mp4", + "bipbop_480_624kbps-cenc-audio-key2-1.m4s", + "bipbop_480_624kbps-cenc-audio-key2-2.m4s", + "bipbop_480_624kbps-cenc-audio-key2-3.m4s", + "bipbop_480_624kbps-cenc-audio-key2-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_624kbps-cenc-video-key2-init.mp4", + "bipbop_480_624kbps-cenc-video-key2-1.m4s", + "bipbop_300_215kbps-cenc-video-key2-init.mp4", + "bipbop_300_215kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + "7e571d047e571d047e571d047e571d22": "7e5744447e5744447e5744447e574422", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "640x480 with 1st keys then 400x300 with 2nd keys", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_624kbps-cenc-audio-key1-init.mp4", + "bipbop_480_624kbps-cenc-audio-key1-1.m4s", + "bipbop_480_624kbps-cenc-audio-key1-2.m4s", + "bipbop_480_624kbps-cenc-audio-key1-3.m4s", + "bipbop_480_624kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_624kbps-cenc-video-key1-init.mp4", + "bipbop_480_624kbps-cenc-video-key1-1.m4s", + "bipbop_300_215kbps-cenc-video-key2-init.mp4", + "bipbop_300_215kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "400x300 with 1st keys then 640x480 with 2nd keys", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_300_215kbps-cenc-audio-key1-init.mp4", + "bipbop_300_215kbps-cenc-audio-key1-1.m4s", + "bipbop_300_215kbps-cenc-audio-key1-2.m4s", + "bipbop_300_215kbps-cenc-audio-key1-3.m4s", + "bipbop_300_215kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_300_215kbps-cenc-video-key1-init.mp4", + "bipbop_300_215kbps-cenc-video-key1-1.m4s", + "bipbop_480_624kbps-cenc-video-key2-init.mp4", + "bipbop_480_624kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "640x480@959kbps with 1st keys then 640x480@624kbps with 2nd keys", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_959kbps-cenc-audio-key1-init.mp4", + "bipbop_480_959kbps-cenc-audio-key1-1.m4s", + "bipbop_480_959kbps-cenc-audio-key1-2.m4s", + "bipbop_480_959kbps-cenc-audio-key1-3.m4s", + "bipbop_480_959kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_959kbps-cenc-video-key1-init.mp4", + "bipbop_480_959kbps-cenc-video-key1-1.m4s", + "bipbop_480_624kbps-cenc-video-key2-init.mp4", + "bipbop_480_624kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "640x480@624kbps with 1st keys then 640x480@959kbps with 2nd keys", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_480_624kbps-cenc-audio-key1-init.mp4", + "bipbop_480_624kbps-cenc-audio-key1-1.m4s", + "bipbop_480_624kbps-cenc-audio-key1-2.m4s", + "bipbop_480_624kbps-cenc-audio-key1-3.m4s", + "bipbop_480_624kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_480_624kbps-cenc-video-key1-init.mp4", + "bipbop_480_624kbps-cenc-video-key1-1.m4s", + "bipbop_480_959kbps-cenc-video-key2-init.mp4", + "bipbop_480_959kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "400x300 with presentation size 533x300", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4", + "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s", + "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s", + "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s", + "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_300wp_227kbps-cenc-video-key1-init.mp4", + "bipbop_300wp_227kbps-cenc-video-key1-1.m4s", + "bipbop_300wp_227kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "400x300 as-is then 400x300 presented as 533x300", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_300_215kbps-cenc-audio-key1-init.mp4", + "bipbop_300_215kbps-cenc-audio-key1-1.m4s", + "bipbop_300_215kbps-cenc-audio-key1-2.m4s", + "bipbop_300_215kbps-cenc-audio-key1-3.m4s", + "bipbop_300_215kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_300_215kbps-cenc-video-key1-init.mp4", + "bipbop_300_215kbps-cenc-video-key1-1.m4s", + "bipbop_300wp_227kbps-cenc-video-key1-init.mp4", + "bipbop_300wp_227kbps-cenc-video-key1-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "400x225", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_225w_175kbps-cenc-audio-key1-init.mp4", + "bipbop_225w_175kbps-cenc-audio-key1-1.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-2.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-3.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_225w_175kbps-cenc-video-key1-init.mp4", + "bipbop_225w_175kbps-cenc-video-key1-1.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "640x360", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_360w_253kbps-cenc-audio-key1-init.mp4", + "bipbop_360w_253kbps-cenc-audio-key1-1.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-2.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-3.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_360w_253kbps-cenc-video-key1-init.mp4", + "bipbop_360w_253kbps-cenc-video-key1-1.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "400x225 then 640x360", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_225w_175kbps-cenc-audio-key1-init.mp4", + "bipbop_225w_175kbps-cenc-audio-key1-1.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-2.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-3.m4s", + "bipbop_225w_175kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_225w_175kbps-cenc-video-key1-init.mp4", + "bipbop_225w_175kbps-cenc-video-key1-1.m4s", + "bipbop_360w_253kbps-cenc-video-key2-init.mp4", + "bipbop_360w_253kbps-cenc-video-key2-1.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + name: "640x360 then 640x480", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="mp4a.40.2"', + fragments: [ + "bipbop_360w_253kbps-cenc-audio-key1-init.mp4", + "bipbop_360w_253kbps-cenc-audio-key1-1.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-2.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-3.m4s", + "bipbop_360w_253kbps-cenc-audio-key1-4.m4s", + ], + }, + { + name: "video", + type: 'video/mp4; codecs="avc1.64000d"', + fragments: [ + "bipbop_360w_253kbps-cenc-video-key1-init.mp4", + "bipbop_360w_253kbps-cenc-video-key1-1.m4s", + "bipbop_480_624kbps-cenc-video-key2-init.mp4", + "bipbop_480_624kbps-cenc-video-key2-2.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311", + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312", + }, + sessionType: "temporary", + sessionCount: 3, + duration: 1.6, + }, + { + // File generated with shaka packager: + // packager-osx --enable_raw_key_encryption --keys label=:key_id=7e571d047e571d047e571d047e571d21:key=7e5744447e5744447e5744447e574421 --segment_duration 1 --clear_lead 0 in=test-flac.mp4,stream=audio,output=flac-sample-cenc.mp4 + name: "flac in mp4 clearkey", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="flac"', + fragments: ["flac-sample-cenc.mp4"], + }, + ], + keys: { + // "keyid" : "key" + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 1, + duration: 2.05, + }, + { + // File generated with shaka packager: + // packager-osx --enable_raw_key_encryption --keys label=:key_id=7e571d047e571d047e571d047e571d21:key=7e5744447e5744447e5744447e574421 --segment_duration 1 --clear_lead 0 in=test-opus.mp4,stream=audio,output=opus-sample-cenc.mp4 + name: "opus in mp4 clearkey", + tracks: [ + { + name: "audio", + type: 'audio/mp4; codecs="opus"', + fragments: ["opus-sample-cenc.mp4"], + }, + ], + keys: { + // "keyid" : "key" + "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421", + }, + sessionType: "temporary", + sessionCount: 1, + duration: 1.98, + }, + { + name: "WebM vorbis audio & vp8 video clearkey", + tracks: [ + { + name: "audio", + type: 'audio/webm; codecs="vorbis"', + fragments: ["bipbop_360w_253kbps-clearkey-audio.webm"], + }, + { + name: "video", + type: 'video/webm; codecs="vp8"', + fragments: ["bipbop_360w_253kbps-clearkey-video-vp8.webm"], + }, + ], + keys: { + // "keyid" : "key" + f1f3ee1790527e9de47217d43835f76a: "97b9ddc459c8d5ff23c1f2754c95abe8", + "8b5df745ad84145b5617c33116e35a67": "bddfd35dd9be033ee73bc18bc1885056", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "WebM vorbis audio & vp9 video clearkey", + tracks: [ + { + name: "audio", + type: 'audio/webm; codecs="vorbis"', + fragments: ["bipbop_360w_253kbps-clearkey-audio.webm"], + }, + { + name: "video", + type: 'video/webm; codecs="vp9"', + fragments: ["bipbop_360w_253kbps-clearkey-video-vp9.webm"], + }, + ], + keys: { + // "keyid" : "key" + f1f3ee1790527e9de47217d43835f76a: "97b9ddc459c8d5ff23c1f2754c95abe8", + eedf63a94fa7c398ee094f123a4ee709: "973b679a746c82f3acdb856b30e9378e", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 1.6, + }, + { + name: "WebM vorbis audio & vp9 video clearkey with subsample encryption", + tracks: [ + { + name: "audio", + type: 'audio/webm; codecs="vorbis"', + fragments: ["sintel-short-clearkey-subsample-encrypted-audio.webm"], + }, + { + name: "video", + type: 'video/webm; codecs="vp9"', + fragments: ["sintel-short-clearkey-subsample-encrypted-video.webm"], + }, + ], + keys: { + // "keyid" : "key" + "2cdb0ed6119853e7850671c3e9906c3c": "808B9ADAC384DE1E4F56140F4AD76194", + }, + sessionType: "temporary", + sessionCount: 2, + duration: 2.0, + }, + { + // Files adapted from testcase for bug 1560092. See bug 1630381 for a + // detailed explanation on how they were adapted. + name: "avc3 h264 video in mp4 using clearkey cenc encryption", + tracks: [ + { + name: "video", + type: 'video/mp4; codecs="avc3.640015"', + fragments: [ + "big-buck-bunny-cenc-avc3-init.mp4", + "big-buck-bunny-cenc-avc3-1.m4s", + ], + }, + ], + keys: { + // "keyid" : "key" + "10000000100010001000100000000001": "3A2A1B68DD2BD9B2EEB25E84C4776668", + }, + sessionType: "temporary", + sessionCount: 1, + duration: 2.08, + }, +]; + +var gEMENonMSEFailTests = [ + { + name: "short-cenc.mp4", + audioType: 'audio/mp4; codecs="mp4a.40.2"', + videoType: 'video/mp4; codecs="avc1.64000d"', + duration: 0.47, + }, +]; + +// These are files that are used for video decode suspend in +// background tabs tests. +var gDecodeSuspendTests = [ + { name: "gizmo.mp4", type: "video/mp4", duration: 5.56 }, + { name: "gizmo-noaudio.mp4", type: "video/mp4", duration: 5.56 }, + { name: "gizmo.webm", type: 'video/webm; codecs="vp9,opus"', duration: 5.56 }, + { + name: "gizmo-noaudio.webm", + type: 'video/webm; codecs="vp9"', + duration: 5.56, + }, +]; + +function checkMetadata(msg, e, test) { + if (test.width) { + is(e.videoWidth, test.width, msg + " video width"); + } + if (test.height) { + is(e.videoHeight, test.height, msg + " video height"); + } + if (test.duration) { + ok( + Math.abs(e.duration - test.duration) < 0.1, + msg + " duration (" + e.duration + ") should be around " + test.duration + ); + } + is( + !!test.keys, + SpecialPowers.do_lookupGetter(e, "isEncrypted").apply(e), + msg + " isEncrypted should be true if we have decryption keys" + ); +} + +// Returns the first test from candidates array which we can play with the +// installed video backends. +function getPlayableVideo(candidates) { + var resources = getPlayableVideos(candidates); + if (resources.length > 0) { + return resources[0]; + } + return null; +} + +function getPlayableVideos(candidates) { + var v = manifestVideo(); + return candidates.filter(function(x) { + return /^video/.test(x.type) && v.canPlayType(x.type); + }); +} + +function getPlayableAudio(candidates) { + var v = manifestVideo(); + var resources = candidates.filter(function(x) { + return /^audio/.test(x.type) && v.canPlayType(x.type); + }); + if (resources.length > 0) { + return resources[0]; + } + return null; +} + +// Returns the type of element that should be created for the given mimetype. +function getMajorMimeType(mimetype) { + if (/^video/.test(mimetype)) { + return "video"; + } + return "audio"; +} + +// Force releasing decoder to avoid timeout in waiting for decoding resource. +function removeNodeAndSource(n) { + n.remove(); + // reset |srcObject| first since it takes precedence over |src|. + n.srcObject = null; + n.removeAttribute("src"); + n.load(); + while (n.firstChild) { + n.firstChild.remove(); + } +} + +function once(target, name, cb) { + var p = new Promise(function(resolve, reject) { + target.addEventListener( + name, + function() { + resolve(); + }, + { once: true } + ); + }); + if (cb) { + p.then(cb); + } + return p; +} + +/** + * @param {HTMLMediaElement} video target of interest. + * @param {string} eventName the event to wait on. + * @returns {Promise} A promise that is resolved when event happens. + */ +function nextEvent(video, eventName) { + return new Promise(function(resolve, reject) { + let f = function(event) { + video.removeEventListener(eventName, f); + resolve(event); + }; + video.addEventListener(eventName, f); + }); +} + +function TimeStamp(token) { + function pad(x) { + return x < 10 ? "0" + x : x; + } + var now = new Date(); + var ms = now.getMilliseconds(); + var time = + "[" + + pad(now.getHours()) + + ":" + + pad(now.getMinutes()) + + ":" + + pad(now.getSeconds()) + + "." + + ms + + "]" + + // eslint-disable-next-line no-nested-ternary + (ms < 10 ? " " : ms < 100 ? " " : ""); + return token ? time + " " + token : time; +} + +function Log(token, msg) { + info(TimeStamp(token) + " " + msg); +} + +// Number of tests to run in parallel. +var PARALLEL_TESTS = 2; + +// Prefs to set before running tests. Use this to improve coverage of +// conditions that might not otherwise be encountered on the test data. +var gTestPrefs = [ + ["media.recorder.max_memory", 1024], + ["media.audio-max-decode-error", 0], + ["media.video-max-decode-error", 0], +]; + +// When true, we'll loop forever on whatever test we run. Use this to debug +// intermittent test failures. +const DEBUG_TEST_LOOP_FOREVER = false; + +// Manages a run of media tests. Runs them in chunks in order to limit +// the number of media elements/threads running in parallel. This limits peak +// memory use, particularly on Linux x86 where thread stacks use 10MB of +// virtual address space. +// Usage: +// 1. Create a new MediaTestManager object. +// 2. Create a test startTest function. This takes a test object and a token, +// and performs anything necessary to start the test. The test object is an +// element in one of the g*Tests above. Your startTest function must call +// MediaTestManager.start(token) if it starts a test. The test object is +// guaranteed to be playable by our supported decoders; you don't need to +// check canPlayType. +// 3. When your tests finishes, call MediaTestManager.finished(), passing +// the token back to the manager. The manager may either start the next run +// or end the mochitest if all the tests are done. +function MediaTestManager() { + // Set a very large timeout to prevent Mochitest timeout. + // Instead MediaTestManager will manage timeout of each test. + SimpleTest.requestLongerTimeout(1000); + + // Return how many seconds elapsed since |begin|. + function elapsedTime(begin) { + var end = new Date(); + return (end.getTime() - begin.getTime()) / 1000; + } + // Sets up a MediaTestManager to runs through the 'tests' array, which needs + // to be one of, or have the same fields as, the g*Test arrays of tests. Uses + // the user supplied 'startTest' function to initialize the test. This + // function must accept two arguments, the test entry from the 'tests' array, + // and a token. Call MediaTestManager.started(token) if you start the test, + // and MediaTestManager.finished(token) when the test finishes. You don't have + // to start every test, but if you call started() you *must* call finish() + // else you'll timeout. + this.runTests = function(tests, startTest) { + this.startTime = new Date(); + SimpleTest.info( + "Started " + + this.startTime + + " (" + + this.startTime.getTime() / 1000 + + "s)" + ); + this.testNum = 0; + this.tests = tests; + this.startTest = startTest; + this.tokens = []; + this.isShutdown = false; + this.numTestsRunning = 0; + this.handlers = {}; + this.timers = {}; + + // Always wait for explicit finish. + SimpleTest.waitForExplicitFinish(); + SpecialPowers.pushPrefEnv({ set: gTestPrefs }, () => { + this.nextTest(); + }); + + SimpleTest.registerCleanupFunction(() => { + if (this.tokens.length > 0) { + info("Test timed out. Remaining tests=" + this.tokens); + } + for (var token of this.tokens) { + var handler = this.handlers[token]; + if (handler && handler.ontimeout) { + handler.ontimeout(); + } + } + }); + }; + + // Registers that the test corresponding to 'token' has been started. + // Don't call more than once per token. + this.started = function(token, handler) { + this.tokens.push(token); + this.numTestsRunning++; + this.handlers[token] = handler; + + var onTimeout = async () => { + ok(false, "Test timed out!"); + info(`${token} timed out!`); + await dumpDebugInfoForToken(token); + this.finished(token); + }; + // Default timeout to 180s for each test. + // Call SimpleTest._originalSetTimeout() to bypass the flaky timeout checker. + this.timers[token] = SimpleTest._originalSetTimeout.call( + window, + onTimeout, + 180000 + ); + + is( + this.numTestsRunning, + this.tokens.length, + "[started " + + token + + " t=" + + elapsedTime(this.startTime) + + "] Length of array should match number of running tests" + ); + }; + + // Registers that the test corresponding to 'token' has finished. Call when + // you've finished your test. If all tests are complete this will finish the + // run, otherwise it may start up the next run. It's ok to call multiple times + // per token. + this.finished = function(token) { + var i = this.tokens.indexOf(token); + if (i != -1) { + // Remove the element from the list of running tests. + this.tokens.splice(i, 1); + } + + if (this.timers[token]) { + // Cancel the timer when the test finishes. + clearTimeout(this.timers[token]); + this.timers[token] = null; + } + + info("[finished " + token + "] remaining= " + this.tokens); + this.numTestsRunning--; + is( + this.numTestsRunning, + this.tokens.length, + "[finished " + + token + + " t=" + + elapsedTime(this.startTime) + + "] Length of array should match number of running tests" + ); + if (this.tokens.length < PARALLEL_TESTS) { + this.nextTest(); + } + }; + + // Starts the next batch of tests, or finishes if they're all done. + // Don't call this directly, call finished(token) when you're done. + this.nextTest = function() { + while ( + this.testNum < this.tests.length && + this.tokens.length < PARALLEL_TESTS + ) { + var test = this.tests[this.testNum]; + var token = (test.name ? test.name + "-" : "") + this.testNum; + this.testNum++; + + if (DEBUG_TEST_LOOP_FOREVER && this.testNum == this.tests.length) { + this.testNum = 0; + } + + // Ensure we can play the resource type. + if ( + test.type && + !document.createElement("video").canPlayType(test.type) + ) { + continue; + } + + // Do the init. This should start the test. + this.startTest(test, token); + } + + if ( + this.testNum == this.tests.length && + !DEBUG_TEST_LOOP_FOREVER && + this.tokens.length == 0 && + !this.isShutdown + ) { + this.isShutdown = true; + if (this.onFinished) { + this.onFinished(); + } + var onCleanup = () => { + var end = new Date(); + SimpleTest.info( + "Finished at " + end + " (" + end.getTime() / 1000 + "s)" + ); + SimpleTest.info("Running time: " + elapsedTime(this.startTime) + "s"); + SimpleTest.finish(); + }; + mediaTestCleanup(onCleanup); + } + }; +} + +// Ensures we've got no active video or audio elements in the document, and +// forces a GC to release the address space reserved by the decoders' threads' +// stacks. +function mediaTestCleanup(callback) { + var V = document.getElementsByTagName("video"); + for (let i = 0; i < V.length; i++) { + removeNodeAndSource(V[i]); + V[i] = null; + } + var A = document.getElementsByTagName("audio"); + for (let i = 0; i < A.length; i++) { + removeNodeAndSource(A[i]); + A[i] = null; + } + SpecialPowers.exactGC(callback); +} + +async function dumpDebugInfoForToken(token) { + for (let v of document.getElementsByTagName("video")) { + if (token === v.token) { + info(JSON.stringify(await SpecialPowers.wrap(v).mozRequestDebugInfo())); + return; + } + } + for (let a of document.getElementsByTagName("audio")) { + if (token === a.token) { + info(JSON.stringify(await SpecialPowers.wrap(a).mozRequestDebugInfo())); + return; + } + } +} + +// Could be undefined in a page opened by the parent test page +// like file_access_controls.html. +if ("SimpleTest" in window) { + SimpleTest.requestFlakyTimeout("untriaged"); + + // Register timeout function to dump debugging logs. + SimpleTest.registerTimeoutFunction(async function() { + for (const v of document.getElementsByTagName("video")) { + SimpleTest.info( + JSON.stringify(await SpecialPowers.wrap(v).mozRequestDebugInfo()) + ); + } + for (const a of document.getElementsByTagName("audio")) { + SimpleTest.info( + JSON.stringify(await SpecialPowers.wrap(a).mozRequestDebugInfo()) + ); + } + }); +} diff --git a/dom/media/test/midflight-redirect.sjs b/dom/media/test/midflight-redirect.sjs new file mode 100644 index 0000000000..d4dae537a0 --- /dev/null +++ b/dom/media/test/midflight-redirect.sjs @@ -0,0 +1,78 @@ +function parseQuery(query, key) { + for (let p of query.split('&')) { + if (p == key) { + return true; + } + if (p.startsWith(key + "=")) { + return p.substring(key.length + 1); + } + } +} + +// Return the first few bytes in a short byte range response. When Firefox +// requests subsequent bytes in a second range request, respond with a +// redirect. Requests after the first redirected are serviced as expected. +function handleRequest(request, response) +{ + var query = request.queryString; + var resource = parseQuery(query, "resource"); + var type = parseQuery(query, "type") || "application/octet-stream"; + var redirected = parseQuery(query, "redirected") || false; + var useCors = parseQuery(query, "cors") || false; + + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/" + resource; + var split = paths.split("/"); + for (var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + + bis.setInputStream(fis); + var bytes = bis.readBytes(bis.available()); + let [from, to] = request.getHeader("range").split("=")[1].split("-").map(s => parseInt(s)); + + if (!redirected && from > 0) { + var origin = request.host == "mochi.test" ? "example.org" : "mochi.test:8888"; + response.setStatusLine(request.httpVersion, 303, "See Other"); + let url = "http://" + origin + + "/tests/dom/media/test/midflight-redirect.sjs?redirected&" + query; + response.setHeader("Location", url); + response.setHeader("Content-Type", "text/html"); + return; + } + + if (isNaN(to)) { + to = bytes.length - 1; + } + + if (from == 0 && !redirected) { + to = parseInt(parseQuery(query, "redirectAt")) || Math.floor(bytes.length / 4); + } + to = Math.min(to, bytes.length - 1); + + // Note: 'to' is the first index *excluded*, so we need (to + 1) + // in the substring end here. + byterange = bytes.substring(from, to + 1); + + let contentRange = "bytes " + from + "-" + to + "/" + bytes.length; + let contentLength = byterange.length.toString(); + + response.setStatusLine(request.httpVersion, 206, "Partial Content"); + response.setHeader("Content-Range", contentRange); + response.setHeader("Content-Length", contentLength, false); + response.setHeader("Content-Type", type, false); + response.setHeader("Accept-Ranges", "bytes", false); + response.setHeader("Cache-Control", "no-cache, no-store, must-revalidate"); + if (redirected && useCors) { + response.setHeader("Access-Control-Allow-Origin", "*"); + } + response.write(byterange, byterange.length); + bis.close(); +} diff --git a/dom/media/test/mochitest.ini b/dom/media/test/mochitest.ini new file mode 100644 index 0000000000..6561f1e614 --- /dev/null +++ b/dom/media/test/mochitest.ini @@ -0,0 +1,1172 @@ +# Media tests should be backend independent, i.e., not conditioned on ogg, +# wave etc. (The only exception is the can_play_type tests, which +# necessarily depend on the backend(s) configured.) As far as possible, each +# test should work with any resource type. This makes it easy to add new +# backends and reduces the amount of test duplication. + +# For each supported backend, resources that can be played by that backend +# should be added to the lists in manifest.js. Media tests that aren't +# testing for a bug in handling a specific resource type should pick one of +# the lists in manifest.js and run the test for each resource in the list +# that is supported in the current build (the canPlayType API is useful for +# this). + +# To test whether a valid resource can simply be played through correctly, +# and optionally that its metadata is read correctly, just add it to +# gPlayTests in manifest.js. To test whether an invalid resource correctly +# throws an error (and does not cause a crash or hang), just add it to +# gErrorTests in manifest.js. + +# To test for a specific bug in handling a specific resource type, make the +# test first check canPlayType for the type, and if it's not supported, just +# do ok(true, "Type not supported") and stop the test. + +[DEFAULT] +subsuite = media +skip-if = (os == "win" && processor == "aarch64") # aarch64 due to 1536604 +support-files = + 16bit_wave_extrametadata.wav + 16bit_wave_extrametadata.wav^headers^ + 320x240.ogv + 320x240.ogv^headers^ + 448636.ogv + 448636.ogv^headers^ + A4.ogv + A4.ogv^headers^ + VID_0001.ogg + VID_0001.ogg^headers^ + allowed.sjs + ambisonics.mp4 + ambisonics.mp4^headers^ + audio-gaps.ogg + audio-gaps.ogg^headers^ + audio-gaps-short.ogg + audio-gaps-short.ogg^headers^ + audio-overhang.ogg + audio-overhang.ogg^headers^ + audio.wav + audio.wav^headers^ + av1.mp4 + av1.mp4^headers^ + background_video.js + badtags.ogg + badtags.ogg^headers^ + bear-640x360-v_frag-cenc-key_rotation.mp4 + bear-640x360-a_frag-cenc-key_rotation.mp4 + beta-phrasebook.ogg + beta-phrasebook.ogg^headers^ + big.wav + big.wav^headers^ + big-buck-bunny-cenc-avc3-1.m4s + big-buck-bunny-cenc-avc3-1.m4s^headers^ + big-buck-bunny-cenc-avc3-init.mp4 + big-buck-bunny-cenc-avc3-init.mp4^headers^ + big-short.wav + big-short.wav^headers^ + bipbop.mp4 + bipbop-cenc-audio1.m4s + bipbop-cenc-audio1.m4s^headers^ + bipbop-cenc-audio2.m4s + bipbop-cenc-audio2.m4s^headers^ + bipbop-cenc-audio3.m4s + bipbop-cenc-audio3.m4s^headers^ + bipbop-cenc-audioinit.mp4 + bipbop-cenc-audioinit.mp4^headers^ + bipbop-cenc-video1.m4s + bipbop-cenc-video1.m4s^headers^ + bipbop-cenc-video2.m4s + bipbop-cenc-video2.m4s^headers^ + bipbop-cenc-videoinit.mp4 + bipbop-cenc-videoinit.mp4^headers^ + bipbop-cenc-video-10s.mp4 + bipbop-cenc-video-10s.mp4^headers^ + bipbop-clearkey-keyrotation-clear-lead-audio.mp4 + bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^ + bipbop-clearkey-keyrotation-clear-lead-video.mp4 + bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^ + bipbop_225w_175kbps.mp4 + bipbop_225w_175kbps.mp4^headers^ + bipbop_225w_175kbps-cenc-audio-key1-1.m4s + bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key1-2.m4s + bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key1-3.m4s + bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key1-4.m4s + bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key1-init.mp4 + bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_225w_175kbps-cenc-audio-key2-1.m4s + bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key2-2.m4s + bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key2-3.m4s + bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key2-4.m4s + bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_225w_175kbps-cenc-audio-key2-init.mp4 + bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_225w_175kbps-cenc-video-key1-1.m4s + bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^ + bipbop_225w_175kbps-cenc-video-key1-init.mp4 + bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^ + bipbop_225w_175kbps-cenc-video-key2-1.m4s + bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^ + bipbop_225w_175kbps-cenc-video-key2-init.mp4 + bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^ + bipbop_300_215kbps-cenc-audio-key1-1.m4s + bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key1-2.m4s + bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key1-3.m4s + bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key1-4.m4s + bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key1-init.mp4 + bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_300_215kbps-cenc-audio-key2-1.m4s + bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key2-2.m4s + bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key2-3.m4s + bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key2-4.m4s + bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_300_215kbps-cenc-audio-key2-init.mp4 + bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_300_215kbps-cenc-video-key1-1.m4s + bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^ + bipbop_300_215kbps-cenc-video-key1-2.m4s + bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^ + bipbop_300_215kbps-cenc-video-key1-init.mp4 + bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^ + bipbop_300_215kbps-cenc-video-key2-1.m4s + bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^ + bipbop_300_215kbps-cenc-video-key2-2.m4s + bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^ + bipbop_300_215kbps-cenc-video-key2-init.mp4 + bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^ + bipbop_300wp_227kbps-cenc-audio-key1-1.m4s + bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key1-2.m4s + bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key1-3.m4s + bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key1-4.m4s + bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key1-init.mp4 + bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_300wp_227kbps-cenc-audio-key2-1.m4s + bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key2-2.m4s + bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key2-3.m4s + bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key2-4.m4s + bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_300wp_227kbps-cenc-audio-key2-init.mp4 + bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_300wp_227kbps-cenc-video-key1-1.m4s + bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^ + bipbop_300wp_227kbps-cenc-video-key1-2.m4s + bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^ + bipbop_300wp_227kbps-cenc-video-key1-init.mp4 + bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^ + bipbop_300wp_227kbps-cenc-video-key2-1.m4s + bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^ + bipbop_300wp_227kbps-cenc-video-key2-2.m4s + bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^ + bipbop_300wp_227kbps-cenc-video-key2-init.mp4 + bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^ + bipbop_360w_253kbps-cenc-audio-key1-1.m4s + bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key1-2.m4s + bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key1-3.m4s + bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key1-4.m4s + bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key1-init.mp4 + bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_360w_253kbps-cenc-audio-key2-1.m4s + bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key2-2.m4s + bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key2-3.m4s + bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key2-4.m4s + bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_360w_253kbps-cenc-audio-key2-init.mp4 + bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_360w_253kbps-cenc-video-key1-1.m4s + bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^ + bipbop_360w_253kbps-cenc-video-key1-init.mp4 + bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^ + bipbop_360w_253kbps-cenc-video-key2-1.m4s + bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^ + bipbop_360w_253kbps-cenc-video-key2-init.mp4 + bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^ + bipbop_360w_253kbps-clearkey-audio.webm + bipbop_360w_253kbps-clearkey-audio.webm^headers^ + bipbop_360w_253kbps-clearkey-video-vp8.webm + bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^ + bipbop_360w_253kbps-clearkey-video-vp9.webm + bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^ + bipbop_480_624kbps-cenc-audio-key1-1.m4s + bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key1-2.m4s + bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key1-3.m4s + bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key1-4.m4s + bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key1-init.mp4 + bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_480_624kbps-cenc-audio-key2-1.m4s + bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key2-2.m4s + bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key2-3.m4s + bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key2-4.m4s + bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_480_624kbps-cenc-audio-key2-init.mp4 + bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_480_624kbps-cenc-video-key1-1.m4s + bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^ + bipbop_480_624kbps-cenc-video-key1-2.m4s + bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^ + bipbop_480_624kbps-cenc-video-key1-init.mp4 + bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^ + bipbop_480_624kbps-cenc-video-key2-1.m4s + bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^ + bipbop_480_624kbps-cenc-video-key2-2.m4s + bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^ + bipbop_480_624kbps-cenc-video-key2-init.mp4 + bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^ + bipbop_480_959kbps-cenc-audio-key1-1.m4s + bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key1-2.m4s + bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key1-3.m4s + bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key1-4.m4s + bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key1-init.mp4 + bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_480_959kbps-cenc-audio-key2-1.m4s + bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key2-2.m4s + bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key2-3.m4s + bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key2-4.m4s + bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_480_959kbps-cenc-audio-key2-init.mp4 + bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_480_959kbps-cenc-video-key1-1.m4s + bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^ + bipbop_480_959kbps-cenc-video-key1-2.m4s + bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^ + bipbop_480_959kbps-cenc-video-key1-init.mp4 + bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^ + bipbop_480_959kbps-cenc-video-key2-1.m4s + bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^ + bipbop_480_959kbps-cenc-video-key2-2.m4s + bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^ + bipbop_480_959kbps-cenc-video-key2-init.mp4 + bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^ + bipbop_480wp_663kbps-cenc-audio-key1-1.m4s + bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key1-2.m4s + bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key1-3.m4s + bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key1-4.m4s + bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key1-init.mp4 + bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_480wp_663kbps-cenc-audio-key2-1.m4s + bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key2-2.m4s + bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key2-3.m4s + bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key2-4.m4s + bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_480wp_663kbps-cenc-audio-key2-init.mp4 + bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_480wp_663kbps-cenc-video-key1-1.m4s + bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^ + bipbop_480wp_663kbps-cenc-video-key1-2.m4s + bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^ + bipbop_480wp_663kbps-cenc-video-key1-init.mp4 + bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^ + bipbop_480wp_663kbps-cenc-video-key2-1.m4s + bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^ + bipbop_480wp_663kbps-cenc-video-key2-2.m4s + bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^ + bipbop_480wp_663kbps-cenc-video-key2-init.mp4 + bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^ + bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s + bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s + bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s + bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s + bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4 + bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^ + bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s + bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s + bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s + bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s + bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^ + bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4 + bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^ + bipbop_480wp_1001kbps-cenc-video-key1-1.m4s + bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^ + bipbop_480wp_1001kbps-cenc-video-key1-2.m4s + bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^ + bipbop_480wp_1001kbps-cenc-video-key1-init.mp4 + bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^ + bipbop_480wp_1001kbps-cenc-video-key2-1.m4s + bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^ + bipbop_480wp_1001kbps-cenc-video-key2-2.m4s + bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^ + bipbop_480wp_1001kbps-cenc-video-key2-init.mp4 + bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^ + bipbop-lateaudio.mp4 + bipbop-lateaudio.mp4^headers^ + black100x100-aspect3to2.ogv + black100x100-aspect3to2.ogv^headers^ + bogus.duh + bogus.ogv + bogus.ogv^headers^ + bogus.wav + bogus.wav^headers^ + bug461281.ogg + bug461281.ogg^headers^ + bug482461-theora.ogv + bug482461-theora.ogv^headers^ + bug482461.ogv + bug482461.ogv^headers^ + bug495129.ogv + bug495129.ogv^headers^ + bug495794.ogg + bug495794.ogg^headers^ + bug498380.ogv + bug498380.ogv^headers^ + bug498855-1.ogv + bug498855-1.ogv^headers^ + bug498855-2.ogv + bug498855-2.ogv^headers^ + bug498855-3.ogv + bug498855-3.ogv^headers^ + bug499519.ogv + bug499519.ogv^headers^ + bug500311.ogv + bug500311.ogv^headers^ + bug501279.ogg + bug501279.ogg^headers^ + bug504613.ogv + bug504613.ogv^headers^ + bug504644.ogv + bug504644.ogv^headers^ + bug504843.ogv + bug504843.ogv^headers^ + bug506094.ogv + bug506094.ogv^headers^ + bug516323.indexed.ogv + bug516323.indexed.ogv^headers^ + bug516323.ogv + bug516323.ogv^headers^ + bug520493.ogg + bug520493.ogg^headers^ + bug520500.ogg + bug520500.ogg^headers^ + bug520908.ogv + bug520908.ogv^headers^ + bug523816.ogv + bug523816.ogv^headers^ + bug533822.ogg + bug533822.ogg^headers^ + bug556821.ogv + bug556821.ogv^headers^ + bug557094.ogv + bug557094.ogv^headers^ + bug603918.webm + bug603918.webm^headers^ + bug604067.webm + bug604067.webm^headers^ + bug1066943.webm + bug1066943.webm^headers^ + bug1301226.wav + bug1301226.wav^headers^ + bug1301226-odd.wav + bug1301226-odd.wav^headers^ + bug1377278.webm + bug1377278.webm^headers^ + bunny.webm + can_play_type_dash.js + can_play_type_ogg.js + can_play_type_wave.js + can_play_type_webm.js + cancellable_request.sjs + chain.ogg + chain.ogg^headers^ + chain.ogv + chain.ogv^headers^ + chain.opus + chain.opus^headers^ + chained-audio-video.ogg + chained-audio-video.ogg^headers^ + chained-video.ogv + chained-video.ogv^headers^ + chromeHelper.js + cloneElementVisually_helpers.js + contentType.sjs + detodos.opus + detodos.opus^headers^ + detodos.webm + detodos.webm^headers^ + detodos-short.webm + detodos-short.webm^headers^ + detodos-recorder-test.opus + detodos-recorder-test.opus^headers^ + detodos-short.opus + detodos-short.opus^headers^ + dirac.ogg + dirac.ogg^headers^ + dynamic_resource.sjs + eme.js + empty_size.mp3 + file_access_controls.html + file_eme_createMediaKeys.html + flac-s24.flac + flac-s24.flac^headers^ + flac-noheader-s16.flac + flac-noheader-s16.flac^headers^ + flac-sample.mp4 + flac-sample.mp4^headers^ + flac-sample-cenc.mp4 + flac-sample-cenc.mp4^headers^ + fragment_noplay.js + fragment_play.js + gizmo.mp4 + gizmo.mp4^headers^ + gizmo-noaudio.mp4 + gizmo-noaudio.mp4^headers^ + gizmo-short.mp4 + gizmo-short.mp4^headers^ + gizmo.webm + gizmo.webm^headers^ + gizmo-noaudio.webm + gizmo-noaudio.webm^headers^ + gUM_support.js + gzipped_mp4.sjs + huge-id3.mp3 + huge-id3.mp3^headers^ + id3tags.mp3 + id3tags.mp3^headers^ + invalid-cmap-s0c0.opus + invalid-cmap-s0c0.opus^headers^ + invalid-cmap-s0c2.opus + invalid-cmap-s0c2.opus^headers^ + invalid-cmap-s1c2.opus + invalid-cmap-s1c2.opus^headers^ + invalid-cmap-short.opus + invalid-cmap-short.opus^headers^ + invalid-discard_on_multi_blocks.webm + invalid-discard_on_multi_blocks.webm^headers^ + invalid-excess_discard.webm + invalid-excess_discard.webm^headers^ + invalid-excess_neg_discard.webm + invalid-excess_neg_discard.webm^headers^ + invalid-m0c0.opus + invalid-m0c0.opus^headers^ + invalid-m0c3.opus + invalid-m0c3.opus^headers^ + invalid-m1c0.opus + invalid-m1c0.opus^headers^ + invalid-m1c9.opus + invalid-m1c9.opus^headers^ + invalid-m2c0.opus + invalid-m2c0.opus^headers^ + invalid-m2c1.opus + invalid-m2c1.opus^headers^ + invalid-neg_discard.webm + invalid-neg_discard.webm^headers^ + invalid-preskip.webm + invalid-preskip.webm^headers^ + manifest.js + midflight-redirect.sjs + multiple-bos.ogg + multiple-bos.ogg^headers^ + multiple-bos-more-header-fileds.ogg + multiple-bos-more-header-fileds.ogg^headers^ + multi_id3v2.mp3 + no-cues.webm + no-cues.webm^headers^ + notags.mp3 + notags.mp3^headers^ + opus-mapping2.mp4 + opus-mapping2.mp4^headers^ + opus-mapping2.webm + opus-mapping2.webm^headers^ + opus-sample.mp4 + opus-sample.mp4^headers^ + opus-sample-cenc.mp4 + opus-sample-cenc.mp4^headers^ + owl-funnier-id3.mp3 + owl-funnier-id3.mp3^headers^ + owl-funny-id3.mp3 + owl-funny-id3.mp3^headers^ + owl.mp3 + owl.mp3^headers^ + owl-short.mp3 + owl-short.mp3^headers^ + pixel_aspect_ratio.mp4 + play_promise.js + poster-test.jpg + r11025_msadpcm_c1.wav + r11025_msadpcm_c1.wav^headers^ + r11025_s16_c1.wav + r11025_s16_c1.wav^headers^ + r11025_s16_c1_trailing.wav + r11025_s16_c1_trailing.wav^headers^ + r11025_s16_c1-short.wav + r11025_s16_c1-short.wav^headers^ + r11025_u8_c1.wav + r11025_u8_c1.wav^headers^ + r11025_u8_c1_trunc.wav + r11025_u8_c1_trunc.wav^headers^ + r16000_u8_c1_list.wav + r16000_u8_c1_list.wav^headers^ + reactivate_helper.html + red-46x48.mp4 + red-46x48.mp4^headers^ + red-48x46.mp4 + red-48x46.mp4^headers^ + redirect.sjs + referer.sjs + resolution-change.webm + resolution-change.webm^headers^ + sample.3gp + sample.3g2 + sample-encrypted-sgpdstbl-sbgptraf.mp4 + sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^ + sample-fisbone-skeleton4.ogv + sample-fisbone-skeleton4.ogv^headers^ + sample-fisbone-wrong-header.ogv + sample-fisbone-wrong-header.ogv^headers^ + seek.ogv + seek.ogv^headers^ + seek-short.ogv + seek-short.ogv^headers^ + seek.webm + seek.webm^headers^ + seek-short.webm + seek-short.webm^headers^ + seek_support.js + seekLies.sjs + seek_with_sound.ogg^headers^ + short-cenc.mp4 + sine.webm + sine.webm^headers^ + sintel-short-clearkey-subsample-encrypted-audio.webm + sintel-short-clearkey-subsample-encrypted-audio.webm^headers^ + sintel-short-clearkey-subsample-encrypted-video.webm + sintel-short-clearkey-subsample-encrypted-video.webm^headers^ + short.mp4 + short.mp4.gz + short.mp4^headers^ + short-aac-encrypted-audio.mp4 + short-aac-encrypted-audio.mp4^headers^ + short-audio-fragmented-cenc-without-pssh.mp4 + short-audio-fragmented-cenc-without-pssh.mp4^headers^ + short-video.ogv + short-video.ogv^headers^ + short-vp9-encrypted-video.mp4 + short-vp9-encrypted-video.mp4^headers^ + small-shot-mp3.mp4 + small-shot-mp3.mp4^headers^ + small-shot.m4a + small-shot.mp3 + small-shot.mp3^headers^ + small-shot.ogg + small-shot.ogg^headers^ + small-shot.flac + sound.ogg + sound.ogg^headers^ + spacestorm-1000Hz-100ms.ogg + spacestorm-1000Hz-100ms.ogg^headers^ + split.webm + split.webm^headers^ + street.mp4 + street.mp4^headers^ + test-1-mono.opus + test-1-mono.opus^headers^ + test-2-stereo.opus + test-2-stereo.opus^headers^ + test-3-LCR.opus + test-3-LCR.opus^headers^ + test-4-quad.opus + test-4-quad.opus^headers^ + test-5-5.0.opus + test-5-5.0.opus^headers^ + test-6-5.1.opus + test-6-5.1.opus^headers^ + test-7-6.1.opus + test-7-6.1.opus^headers^ + test-8-7.1.opus + test-8-7.1.opus^headers^ + test-stereo-phase-inversion-180.opus + test-stereo-phase-inversion-180.opus^headers^ + variable-channel.ogg + variable-channel.ogg^headers^ + variable-channel.opus + variable-channel.opus^headers^ + variable-preskip.opus + variable-preskip.opus^headers^ + variable-samplerate.ogg + variable-samplerate.ogg^headers^ + variable-samplerate.opus + variable-samplerate.opus^headers^ + vbr-head.mp3 + vbr-head.mp3^headers^ + vbr.mp3 + vbr.mp3^headers^ + very-short.mp3 + video-overhang.ogg + video-overhang.ogg^headers^ + vp9-superframes.webm + vp9-superframes.webm^headers^ + vp9.webm + vp9.webm^headers^ + vp9-short.webm + vp9-short.webm^headers^ + vp9cake.webm + vp9cake.webm^headers^ + vp9cake-short.webm + vp9cake-short.webm^headers^ + wave_metadata.wav + wave_metadata.wav^headers^ + wave_metadata_bad_len.wav + wave_metadata_bad_len.wav^headers^ + wave_metadata_bad_no_null.wav + wave_metadata_bad_no_null.wav^headers^ + wave_metadata_bad_utf8.wav + wave_metadata_bad_utf8.wav^headers^ + wave_metadata_unknown_tag.wav + wave_metadata_unknown_tag.wav^headers^ + wave_metadata_utf8.wav + wave_metadata_utf8.wav^headers^ + wavedata_alaw.wav + wavedata_alaw.wav^headers^ + wavedata_float.wav + wavedata_float.wav^headers^ + wavedata_s24.wav + wavedata_s24.wav^headers^ + wavedata_s16.wav + wavedata_s16.wav^headers^ + wavedata_u8.wav + wavedata_u8.wav^headers^ + wavedata_ulaw.wav + wavedata_ulaw.wav^headers^ + !/dom/canvas/test/captureStream_common.js + !/dom/html/test/reflect.js + !/dom/media/webrtc/tests/mochitests/head.js + hls/bipbop_16x9_single.m3u8 + hls/bipbop_4x3_single.m3u8 + hls/bipbop_4x3_variant.m3u8 + hls/400x300_prog_index.m3u8 + hls/400x300_prog_index_5s.m3u8 + hls/416x243_prog_index_5s.m3u8 + hls/640x480_prog_index.m3u8 + hls/960x720_prog_index.m3u8 + hls/400x300_seg0.ts + hls/400x300_seg0_5s.ts + hls/400x300_seg1.ts + hls/416x243_seg0_5s.ts + hls/640x480_seg0.ts + hls/640x480_seg1.ts + hls/960x720_seg0.ts + hls/960x720_seg1.ts + +[test_access_control.html] +[test_arraybuffer.html] +[test_aspectratio_mp4.html] +[test_audio1.html] +[test_audio2.html] +[test_audioDocumentTitle.html] +skip-if = true # bug 475110 - disabled since we don't play Wave files standalone +[test_buffered.html] +[test_bug448534.html] +[test_bug463162.xhtml] +[test_bug465498.html] +[test_bug495145.html] +skip-if = os == "win" #Bug 1404373 +[test_bug495300.html] +[test_bug654550.html] +[test_bug686942.html] +[test_bug726904.html] +[test_bug874897.html] +[test_bug879717.html] +skip-if = toolkit == 'android' # bug 1285441, android(bug 1232305) +tags=capturestream +[test_bug895305.html] +skip-if = (android_version == '25' && debug) # android(bug 1232305) +[test_bug919265.html] +skip-if = (android_version == '25' && debug) # android(bug 1232305) +[test_bug1113600.html] +tags=capturestream +[test_bug1242338.html] +[test_bug1248229.html] +tags=capturestream +[test_bug1512958.html] +tags=mtg capturestream +[test_bug1553262.html] +tags=mtg capturestream +[test_can_play_type.html] +skip-if = (android_version == '25' && debug) # android(bug 1232305) +[test_can_play_type_mpeg.html] +[test_can_play_type_no_ogg.html] +skip-if = (android_version == '25' && debug) # android(bug 1232305) +[test_can_play_type_ogg.html] +skip-if = (android_version == '25' && debug) # android(bug 1232305) +[test_chaining.html] +[test_clone_media_element.html] +skip-if = toolkit == 'android' # bug 1108558, android(bug 1232305) +[test_closing_connections.html] +[test_constants.html] +[test_controls.html] +[test_cueless_webm_seek-1.html] +[test_cueless_webm_seek-2.html] +[test_cueless_webm_seek-3.html] +[test_currentTime.html] +[test_decode_error.html] +[test_decode_error_crossorigin.html] +[test_decoder_disable.html] +[test_defaultMuted.html] +[test_delay_load.html] +[test_duration_after_error.html] +[test_eme_autoplay.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_pssh_in_moof.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_session_callable_value.html] +skip-if = verify && debug && (os == 'linux') +scheme=https +[test_eme_canvas_blocked.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_createMediaKeys_iframes.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_detach_media_keys.html] +skip-if = toolkit == 'android' || (verify && debug && (os == 'linux' || os == 'win')) # bug 1149374 +scheme=https +[test_eme_detach_reattach_same_mediakeys_during_playback.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_initDataTypes.html] +skip-if = toolkit == 'android' || (verify && debug && (os == 'linux' || os == 'mac')) # bug 1149374 +scheme=https +[test_eme_missing_pssh.html] +skip-if = toolkit == 'android' || (verify && debug && (os == 'mac')) # bug 1149374 +scheme=https +[test_eme_non_mse_fails.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_request_notifications.html] +skip-if = toolkit == 'android' || (verify && debug && (os == 'linux')) # bug 1149374 +scheme=https +[test_eme_playback.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_requestKeySystemAccess.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_requestMediaKeySystemAccess_with_app_approval.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_sample_groups_playback.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_setMediaKeys_before_attach_MediaSource.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_stream_capture_blocked_case1.html] +tags=mtg capturestream +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_stream_capture_blocked_case2.html] +tags=mtg capturestream +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_stream_capture_blocked_case3.html] +tags=mtg capturestream +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_unsetMediaKeys_then_capture.html] +skip-if = xorigin || toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_waitingforkey.html] +skip-if = xorigin || toolkit == 'android' # bug 1149374 +scheme=https +[test_eme_getstatusforpolicy.html] +skip-if = toolkit == 'android' # bug 1149374 +scheme=https +[test_empty_resource.html] +[test_error_in_video_document.html] +[test_error_on_404.html] +[test_fastSeek.html] +[test_fastSeek-forwards.html] +[test_imagecapture.html] +scheme=https +[test_info_leak.html] +[test_invalid_reject.html] +[test_invalid_reject_play.html] +[test_invalid_seek.html] +[test_load.html] +[test_load_candidates.html] +[test_load_same_resource.html] +[test_load_source.html] +[test_load_source_empty_type.html] +[test_loop.html] +[test_looping_eventsOrder.html] +[test_media_selection.html] +[test_media_sniffer.html] +[test_mediacapabilities_resistfingerprinting.html] +[test_mediarecorder_avoid_recursion.html] +skip-if = os == 'win' && !debug +scheme=https +tags=mtg +[test_mediarecorder_bitrate.html] +skip-if = toolkit == 'android' # bug 1297432, android(bug 1232305) +tags=mtg +[test_mediarecorder_creation.html] +tags=mtg capturestream +[test_mediarecorder_creation_fail.html] +tags=mtg +[test_mediarecorder_fires_start_event_once_when_erroring.html] +tags=mtg +[test_mediarecorder_onerror_pause.html] +scheme=https +tags=mtg +[test_mediarecorder_pause_resume_video.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_mediarecorder_playback_can_repeat.html] +tags=mtg +[test_mediarecorder_principals.html] +skip-if = (os == 'win' && os_version == '10.0' && webrender) # Bug 1453375 +tags=mtg +[test_mediarecorder_record_4ch_audiocontext.html] +tags=mtg +skip-if = os == "linux" && bits == 64 #Bug 1598101 +[test_mediarecorder_record_addtracked_stream.html] +skip-if = toolkit == 'android' # Bug 1408241 +tags=mtg capturestream +[test_mediarecorder_record_audiocontext.html] +tags=mtg +[test_mediarecorder_record_audiocontext_mlk.html] +tags=mtg +[test_mediarecorder_record_audionode.html] +tags=mtg +[test_mediarecorder_record_canvas_captureStream.html] +skip-if = toolkit == 'android' # android(bug 1232305) +tags=mtg +[test_mediarecorder_record_changing_video_resolution.html] +skip-if = toolkit == 'android' # android(bug 1232305) +tags=mtg +[test_mediarecorder_record_upsize_resolution.html] +skip-if = toolkit == 'android' # android(bug 1232305) +tags=mtg +[test_mediarecorder_record_downsize_resolution.html] +skip-if = toolkit == 'android' # android(bug 1232305) +tags=mtg +[test_mediarecorder_record_gum_video_timeslice.html] +scheme=https +tags=mtg +[test_mediarecorder_record_gum_video_timeslice_mixed.html] +scheme=https +tags=mtg +[test_mediarecorder_record_immediate_stop.html] +tags=mtg capturestream +[test_mediarecorder_record_no_timeslice.html] +tags=mtg capturestream +[test_mediarecorder_record_session.html] +tags=mtg capturestream +[test_mediarecorder_record_startstopstart.html] +tags=mtg +[test_mediarecorder_record_timeslice.html] +tags=mtg capturestream +[test_mediarecorder_reload_crash.html] +tags=mtg capturestream +[test_mediarecorder_state_transition.html] +tags=mtg capturestream +[test_mediarecorder_state_event_order.html] +tags=mtg capturestream +[test_mediarecorder_webm_support.html] +tags=mtg +[test_mediarecorder_record_getdata_afterstart.html] +tags=mtg capturestream +[test_mediatrack_consuming_mediaresource.html] +[test_mediatrack_consuming_mediastream.html] +scheme=https +tags=mtg +[test_mediatrack_events.html] +scheme=https +[test_mediatrack_parsing_ogg.html] +[test_mediatrack_replay_from_end.html] +[test_metadata.html] +[test_midflight_redirect_blocked.html] +[test_mixed_principals.html] +skip-if = toolkit == 'android' # bug 1309814, android(bug 1232305) +[test_mozHasAudio.html] +[test_mp3_with_multiple_ID3v2.html] +[test_multiple_mediastreamtracks.html] +scheme=https +[test_networkState.html] +[test_new_audio.html] +[test_no_load_event.html] +[test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html] +[test_paused.html] +[test_paused_after_ended.html] +[test_play_events.html] +[test_play_events_2.html] +[test_play_promise_1.html] +tags=promise-play +[test_play_promise_2.html] +tags=promise-play +[test_play_promise_3.html] +tags=promise-play +[test_play_promise_4.html] +tags=promise-play +[test_play_promise_5.html] +tags=promise-play +[test_play_promise_6.html] +tags=promise-play +[test_play_promise_7.html] +tags=promise-play +[test_play_promise_8.html] +tags=promise-play +[test_play_promise_9.html] +tags=promise-play +[test_play_promise_10.html] +tags=promise-play +[test_play_promise_11.html] +tags=promise-play +[test_play_promise_12.html] +tags=promise-play +[test_play_promise_13.html] +tags=promise-play +[test_play_promise_14.html] +tags=promise-play +[test_play_promise_15.html] +tags=promise-play +[test_play_promise_16.html] +tags=promise-play +[test_play_promise_17.html] +tags=promise-play +[test_play_promise_18.html] +tags=promise-play +[test_play_twice.html] +skip-if = appname == "seamonkey" # Seamonkey: Bug 598252, bug 1307337, bug 1143695 +[test_playback.html] +skip-if = toolkit == 'android' || (debug && os == "mac") # bug 1316177, 1484451 +[test_playback_errors.html] +[test_playback_rate.html] +[test_playback_rate_playpause.html] +[test_playback_reactivate.html] +[test_played.html] +skip-if = toolkit == 'android' && is_emulator # Times out on android-em, Bug 1613946 +[test_preload_actions.html] +[test_preload_attribute.html] +[test_preload_suspend.html] +[test_preserve_playbackrate_after_ui_play.html] +[test_progress.html] +[test_reactivate.html] +skip-if = true # see bug 1319725 +[test_readyState.html] +[test_referer.html] +skip-if = android_version == '25' && debug # android(bug 1232305) +[test_replay_metadata.html] +[test_reset_events_async.html] +[test_reset_src.html] +skip-if = (verify && debug && os == 'win') +[test_video_dimensions.html] +[test_resolution_change.html] +tags=capturestream +[test_resume.html] +skip-if = true # bug 1021673 +[test_seamless_looping.html] +[test_seek_negative.html] +[test_seek_nosrc.html] +[test_seek_out_of_range.html] +skip-if = toolkit == 'android' # bug 1299382, android(bug 1232305) +[test_seek_promise_bug1344357.html] +skip-if = toolkit == 'android' # bug 1299382, android(bug 1232305) +[test_seek-1.html] +skip-if = toolkit == 'android' # bug 1322806, android(bug 1232305) +[test_seek-2.html] +skip-if = toolkit == 'android' # bug 1309778, android(bug 1232305) +[test_seek-3.html] +skip-if = toolkit == 'android' # bug 1321082, android(bug 1232305) +[test_seek-4.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seek-5.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seek-6.html] +skip-if = toolkit == 'android' # bug 1336629, bug 1324482, android(bug 1232305) +[test_seek-7.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seek-8.html] +skip-if = toolkit == 'android' # bug 1310584, android(bug 1232305) +[test_seek-9.html] +skip-if = toolkit == 'android' # bug 1332019, android(bug 1232305) +[test_seek-10.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seek-11.html] +skip-if = toolkit == 'android' # bug 1323133, android(bug 1232305) +[test_seek-12.html] +skip-if = toolkit == 'android' # bug 1321081, android(bug 1232305) +[test_seek-13.html] +skip-if = toolkit == 'android' # bug 1299174, android(bug 1232305) +[test_seek-14.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seekable1.html] +skip-if = toolkit == 'android' # android(bug 1232305) +[test_seekLies.html] +[test_seekToNextFrame.html] +skip-if = toolkit == 'android' # bug 1329391, android(bug 1232305) +tags=seektonextframe +[test_seek_duration.html] +[test_source.html] +[test_source_null.html] +[test_source_write.html] +[test_standalone.html] +[test_streams_capture_origin.html] +tags=mtg capturestream +[test_streams_element_capture.html] +skip-if = true # bug 1372457 # bug 1557901 # bug 1554808 +tags=mtg capturestream +[test_streams_element_capture_mediatrack.html] +tags=mtg capturestream +[test_streams_element_capture_playback.html] +tags=mtg capturestream +[test_streams_element_capture_reset.html] +tags=mtg capturestream +[test_streams_element_capture_twice.html] +tags=mtg capturestream +[test_streams_firstframe.html] +tags=mtg capturestream +[test_streams_gc.html] +tags=mtg capturestream +[test_streams_individual_pause.html] +scheme=https +tags=mtg +[test_streams_srcObject.html] +skip-if = toolkit == 'android' # bug 1300443, android(bug 1232305) +tags=mtg capturestream +[test_streams_tracks.html] +skip-if = toolkit == 'android' # android(bug 1232305) +tags=mtg capturestream +[test_suspend_media_by_inactive_docshell.html] +[test_timeupdate_small_files.html] +[test_unseekable.html] +[test_video_to_canvas.html] +skip-if = toolkit == 'android' # android(bug 1232305), bugs 1320418,1347953,1347954,1348140,1348386 +[test_video_in_audio_element.html] +[test_video_stats_resistfingerprinting.html] +tags = resistfingerprinting +[test_videoDocumentTitle.html] +[test_VideoPlaybackQuality.html] +[test_VideoPlaybackQuality_disabled.html] +[test_volume.html] +[test_vp9_superframes.html] +skip-if = os == 'mac' && os_version == '10.14' # mac due to bug 1545737 +# The tests below contain backend-specific tests. Write backend independent +# tests rather than adding to this list. +[test_can_play_type_webm.html] +[test_can_play_type_wave.html] +[test_fragment_noplay.html] +[test_fragment_play.html] +[test_background_video_cancel_suspend_taint.html] +skip-if = toolkit == 'android' # bug 1346705 +tags = suspend +[test_background_video_cancel_suspend_visible.html] +tags = suspend +[test_background_video_no_suspend_disabled.html] +tags = suspend +[test_background_video_no_suspend_short_vid.html] +tags = suspend +[test_background_video_no_suspend_not_in_tree.html] +tags = suspend +[test_background_video_resume_after_end_show_last_frame.html] +skip-if = toolkit == 'android' # bug 1346705 +tags = suspend +[test_background_video_resume_looping_video_without_audio.html] +tags = suspend +[test_background_video_suspend.html] +skip-if = os == 'android' #Bug 1304480 +tags = suspend +[test_background_video_suspend_ends.html] +tags = suspend +[test_background_video_tainted_by_capturestream.html] +tags = suspend +[test_background_video_tainted_by_createimagebitmap.html] +tags = suspend +[test_background_video_tainted_by_drawimage.html] +skip-if = toolkit == 'android' # bug 1346705 +tags = suspend +[test_background_video_drawimage_with_suspended_video.html] +skip-if = toolkit == 'android' # bug 1346705 +tags = suspend +[test_background_video_ended_event.html] +skip-if = toolkit == 'android' # bug 1346705 +tags = suspend + +[test_temporary_file_blob_video_plays.html] +skip-if = toolkit == 'android' || (os == 'win' && processor == 'aarch64') # bug 1533534 # android(bug 1232305) +[test_videoPlaybackQuality_totalFrames.html] +skip-if = os == 'win' || (os == 'mac' && os_version == '10.14') # bug 1374189, mac due to bug 1544938 + +[test_video_gzip_encoding.html] + +[test_playback_hls.html] +# HLS is only supported on Fennec with API level >= 16 +# TODO: This test is similar to test_playback.html, will remove the +# redundant code once test_playback.html is enabled on Fennec. +skip-if = toolkit != 'android' +tags = hls + +[test_hls_player_independency.html] +# There's a limit for creating decoder when API lever < 18(Bug 1278574) +# We could skip the test in that case as we cannot play 2 video at a time. +skip-if = toolkit != 'android' || android_version < '18' +tags = hls + +[test_bug1431810_opus_downmix_to_mono.html] + +[test_cloneElementVisually_paused.html] +tags = cloneelementvisually +[test_cloneElementVisually_mediastream.html] +tags = cloneelementvisually +[test_cloneElementVisually_mediastream_multitrack.html] +tags = cloneelementvisually +[test_cloneElementVisually_resource_change.html] +tags = cloneelementvisually +[test_cloneElementVisually_no_suspend.html] +tags = cloneelementvisually +[test_cloneElementVisually_poster.html] +tags = cloneelementvisually +[test_cloneElementVisually_ended_video.html] +tags = cloneelementvisually diff --git a/dom/media/test/multi_id3v2.mp3 b/dom/media/test/multi_id3v2.mp3 Binary files differnew file mode 100644 index 0000000000..253f19a9b6 --- /dev/null +++ b/dom/media/test/multi_id3v2.mp3 diff --git a/dom/media/test/multiple-bos-more-header-fileds.ogg b/dom/media/test/multiple-bos-more-header-fileds.ogg Binary files differnew file mode 100644 index 0000000000..c9721cb98e --- /dev/null +++ b/dom/media/test/multiple-bos-more-header-fileds.ogg diff --git a/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^ b/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/multiple-bos.ogg b/dom/media/test/multiple-bos.ogg Binary files differnew file mode 100644 index 0000000000..193200868e --- /dev/null +++ b/dom/media/test/multiple-bos.ogg diff --git a/dom/media/test/multiple-bos.ogg^headers^ b/dom/media/test/multiple-bos.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/multiple-bos.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/no-cues.webm b/dom/media/test/no-cues.webm Binary files differnew file mode 100644 index 0000000000..8ed761099e --- /dev/null +++ b/dom/media/test/no-cues.webm diff --git a/dom/media/test/no-cues.webm^headers^ b/dom/media/test/no-cues.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/no-cues.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/notags.mp3 b/dom/media/test/notags.mp3 Binary files differnew file mode 100644 index 0000000000..7f298131aa --- /dev/null +++ b/dom/media/test/notags.mp3 diff --git a/dom/media/test/notags.mp3^headers^ b/dom/media/test/notags.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/notags.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/opus-mapping2.mp4 b/dom/media/test/opus-mapping2.mp4 Binary files differnew file mode 100644 index 0000000000..72401a9c0b --- /dev/null +++ b/dom/media/test/opus-mapping2.mp4 diff --git a/dom/media/test/opus-mapping2.mp4^headers^ b/dom/media/test/opus-mapping2.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/opus-mapping2.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/opus-mapping2.webm b/dom/media/test/opus-mapping2.webm Binary files differnew file mode 100644 index 0000000000..4379f2534a --- /dev/null +++ b/dom/media/test/opus-mapping2.webm diff --git a/dom/media/test/opus-mapping2.webm^headers^ b/dom/media/test/opus-mapping2.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/opus-mapping2.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/opus-sample-cenc.mp4 b/dom/media/test/opus-sample-cenc.mp4 Binary files differnew file mode 100644 index 0000000000..22bb787540 --- /dev/null +++ b/dom/media/test/opus-sample-cenc.mp4 diff --git a/dom/media/test/opus-sample-cenc.mp4^headers^ b/dom/media/test/opus-sample-cenc.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/opus-sample-cenc.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/opus-sample.mp4 b/dom/media/test/opus-sample.mp4 Binary files differnew file mode 100644 index 0000000000..80329ce14b --- /dev/null +++ b/dom/media/test/opus-sample.mp4 diff --git a/dom/media/test/opus-sample.mp4^headers^ b/dom/media/test/opus-sample.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/opus-sample.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/owl-funnier-id3.mp3 b/dom/media/test/owl-funnier-id3.mp3 Binary files differnew file mode 100644 index 0000000000..05ec507530 --- /dev/null +++ b/dom/media/test/owl-funnier-id3.mp3 diff --git a/dom/media/test/owl-funnier-id3.mp3^headers^ b/dom/media/test/owl-funnier-id3.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/owl-funnier-id3.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/owl-funny-id3.mp3 b/dom/media/test/owl-funny-id3.mp3 Binary files differnew file mode 100644 index 0000000000..6533755a32 --- /dev/null +++ b/dom/media/test/owl-funny-id3.mp3 diff --git a/dom/media/test/owl-funny-id3.mp3^headers^ b/dom/media/test/owl-funny-id3.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/owl-funny-id3.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/owl-short.mp3 b/dom/media/test/owl-short.mp3 Binary files differnew file mode 100644 index 0000000000..9b31531f22 --- /dev/null +++ b/dom/media/test/owl-short.mp3 diff --git a/dom/media/test/owl-short.mp3^headers^ b/dom/media/test/owl-short.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/owl-short.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/owl.mp3 b/dom/media/test/owl.mp3 Binary files differnew file mode 100644 index 0000000000..9fafa32f93 --- /dev/null +++ b/dom/media/test/owl.mp3 diff --git a/dom/media/test/owl.mp3^headers^ b/dom/media/test/owl.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/owl.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/pixel_aspect_ratio.mp4 b/dom/media/test/pixel_aspect_ratio.mp4 Binary files differnew file mode 100644 index 0000000000..fce12cc03e --- /dev/null +++ b/dom/media/test/pixel_aspect_ratio.mp4 diff --git a/dom/media/test/play_promise.js b/dom/media/test/play_promise.js new file mode 100644 index 0000000000..7051fedc19 --- /dev/null +++ b/dom/media/test/play_promise.js @@ -0,0 +1,3 @@ +function getNotSupportedFile(name) { + return name + ".bad"; +} diff --git a/dom/media/test/poster-test.jpg b/dom/media/test/poster-test.jpg Binary files differnew file mode 100644 index 0000000000..595a5315f8 --- /dev/null +++ b/dom/media/test/poster-test.jpg diff --git a/dom/media/test/r11025_msadpcm_c1.wav b/dom/media/test/r11025_msadpcm_c1.wav Binary files differnew file mode 100644 index 0000000000..2e883ba5ed --- /dev/null +++ b/dom/media/test/r11025_msadpcm_c1.wav diff --git a/dom/media/test/r11025_msadpcm_c1.wav^headers^ b/dom/media/test/r11025_msadpcm_c1.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_msadpcm_c1.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r11025_s16_c1-short.wav b/dom/media/test/r11025_s16_c1-short.wav Binary files differnew file mode 100644 index 0000000000..e08d5bbdc0 --- /dev/null +++ b/dom/media/test/r11025_s16_c1-short.wav diff --git a/dom/media/test/r11025_s16_c1-short.wav^headers^ b/dom/media/test/r11025_s16_c1-short.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_s16_c1-short.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r11025_s16_c1.wav b/dom/media/test/r11025_s16_c1.wav Binary files differnew file mode 100644 index 0000000000..ab2e08befb --- /dev/null +++ b/dom/media/test/r11025_s16_c1.wav diff --git a/dom/media/test/r11025_s16_c1.wav^headers^ b/dom/media/test/r11025_s16_c1.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_s16_c1.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r11025_s16_c1_trailing.wav b/dom/media/test/r11025_s16_c1_trailing.wav Binary files differnew file mode 100644 index 0000000000..af53beaf25 --- /dev/null +++ b/dom/media/test/r11025_s16_c1_trailing.wav diff --git a/dom/media/test/r11025_s16_c1_trailing.wav^headers^ b/dom/media/test/r11025_s16_c1_trailing.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_s16_c1_trailing.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r11025_u8_c1.wav b/dom/media/test/r11025_u8_c1.wav Binary files differnew file mode 100644 index 0000000000..97dc453b9e --- /dev/null +++ b/dom/media/test/r11025_u8_c1.wav diff --git a/dom/media/test/r11025_u8_c1.wav^headers^ b/dom/media/test/r11025_u8_c1.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_u8_c1.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r11025_u8_c1_trunc.wav b/dom/media/test/r11025_u8_c1_trunc.wav Binary files differnew file mode 100644 index 0000000000..4d2db39777 --- /dev/null +++ b/dom/media/test/r11025_u8_c1_trunc.wav diff --git a/dom/media/test/r11025_u8_c1_trunc.wav^headers^ b/dom/media/test/r11025_u8_c1_trunc.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r11025_u8_c1_trunc.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/r16000_u8_c1_list.wav b/dom/media/test/r16000_u8_c1_list.wav Binary files differnew file mode 100644 index 0000000000..afde32e9a3 --- /dev/null +++ b/dom/media/test/r16000_u8_c1_list.wav diff --git a/dom/media/test/r16000_u8_c1_list.wav^headers^ b/dom/media/test/r16000_u8_c1_list.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/r16000_u8_c1_list.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/reactivate_helper.html b/dom/media/test/reactivate_helper.html new file mode 100644 index 0000000000..1834131559 --- /dev/null +++ b/dom/media/test/reactivate_helper.html @@ -0,0 +1,57 @@ +<!DOCTYPE HTML> +<html> +<body> +<script> +var loadsWaiting = 0; +var elements = []; + +function checkAllLoaded() { + --loadsWaiting; + if (loadsWaiting == 0) { + parent.loadedAll(elements); + } +} + +function loadedData(event) { + var e = event.target; + parent.ok(!elements.includes(e), "Element already loaded: " + e._name); + parent.info("Loaded " + e._name); + elements.push(e); + // Reset "onerror" handler to avoid triggering another error in removeNodeAndSource(). + e.onerror = null; + checkAllLoaded(); + +} + +function error(event) { + var e = event.target; + parent.info("Error " + e._name); + // Don't wait for the element encounting errors. + checkAllLoaded(); +} + +for (var i = 0; i < parent.gSmallTests.length; ++i) { + var test = parent.gSmallTests[i]; + var elemType = /^audio/.test(test.type) ? "audio" : "video"; + // Associate these elements with the subframe's document + var e = document.createElement(elemType); + e.preload = "metadata"; + if (e.canPlayType(test.type)) { + e.src = test.name; + e._name = test.name; + e.onloadeddata = loadedData; + e.onerror = error; + e.load(); + ++loadsWaiting; + parent.info("Loading " + e._name); + } +} + +if (loadsWaiting == 0) { + parent.todo(false, "Can't play anything"); +} else { + parent.SimpleTest.waitForExplicitFinish(); +} +</script> +</body> +</html> diff --git a/dom/media/test/red-46x48.mp4 b/dom/media/test/red-46x48.mp4 Binary files differnew file mode 100644 index 0000000000..0760cc1c16 --- /dev/null +++ b/dom/media/test/red-46x48.mp4 diff --git a/dom/media/test/red-46x48.mp4^headers^ b/dom/media/test/red-46x48.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/red-46x48.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/red-48x46.mp4 b/dom/media/test/red-48x46.mp4 Binary files differnew file mode 100644 index 0000000000..d83de4027d --- /dev/null +++ b/dom/media/test/red-48x46.mp4 diff --git a/dom/media/test/red-48x46.mp4^headers^ b/dom/media/test/red-48x46.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/red-48x46.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/redirect.sjs b/dom/media/test/redirect.sjs new file mode 100644 index 0000000000..1653e8efc0 --- /dev/null +++ b/dom/media/test/redirect.sjs @@ -0,0 +1,26 @@ +function parseQuery(request, key) { + var params = request.queryString.split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +// Return file content for the first request with a given key. +// All subsequent requests return a redirect to a different-origin resource. +function handleRequest(request, response) +{ + var domain = parseQuery(request, "domain"); + var file = parseQuery(request, "file"); + var allowed = parseQuery(request, "allowed"); + + response.setStatusLine(request.httpVersion, 303, "See Other"); + response.setHeader("Location", "http://" + domain + "/tests/dom/media/test/" + (allowed ? "allowed.sjs?" : "") + file); + response.setHeader("Content-Type", "text/html"); +} diff --git a/dom/media/test/referer.sjs b/dom/media/test/referer.sjs new file mode 100644 index 0000000000..69c27afe9a --- /dev/null +++ b/dom/media/test/referer.sjs @@ -0,0 +1,45 @@ +function parseQuery(request, key) { + var params = request.queryString.split('&'); + for (var j = 0; j < params.length; ++j) { + var p = params[j]; + if (p == key) + return true; + if (p.indexOf(key + "=") == 0) + return p.substring(key.length + 1); + if (p.indexOf("=") < 0 && key == "") + return p; + } + return false; +} + +function handleRequest(request, response) +{ + var referer = request.hasHeader("Referer") ? request.getHeader("Referer") + : undefined; + if (referer == "http://mochi.test:8888/tests/dom/media/test/test_referer.html") { + var name = parseQuery(request, "name"); + var type = parseQuery(request, "type"); + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/" + name; + var split = paths.split("/"); + for(var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + bis.setInputStream(fis); + var bytes = bis.readBytes(bis.available()); + response.setHeader("Content-Length", ""+bytes.length, false); + response.setHeader("Content-Type", type, false); + response.write(bytes, bytes.length); + bis.close(); + } + else { + response.setStatusLine(request.httpVersion, 404, "Not found"); + } +} diff --git a/dom/media/test/reftest/av1hdr2020.mp4 b/dom/media/test/reftest/av1hdr2020.mp4 Binary files differnew file mode 100644 index 0000000000..295bec8a3c --- /dev/null +++ b/dom/media/test/reftest/av1hdr2020.mp4 diff --git a/dom/media/test/reftest/av1hdr2020.png b/dom/media/test/reftest/av1hdr2020.png Binary files differnew file mode 100644 index 0000000000..c5d3344a80 --- /dev/null +++ b/dom/media/test/reftest/av1hdr2020.png diff --git a/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html new file mode 100644 index 0000000000..575acb107d --- /dev/null +++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" +src="" +> diff --git a/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html new file mode 100644 index 0000000000..600b04a4f0 --- /dev/null +++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "../bipbop_300_215kbps.mp4"; + video.play(); + video.addEventListener("ended", function() { + document.documentElement.removeAttribute('class'); + }, {once: true}); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/generateREF.html b/dom/media/test/reftest/generateREF.html new file mode 100644 index 0000000000..7249ef2579 --- /dev/null +++ b/dom/media/test/reftest/generateREF.html @@ -0,0 +1,99 @@ +<!DOCTYPE HTML> +<html> +<head> +<script type="application/javascript"> +</script> +</head> +<body> +<p id="out"></p> +<video id="v1" style="position:absolute; left:0; top:0"></video> +<canvas id="canvas"></canvas> +<script type="application/javascript"> +/* READ ME first. +The script is trying to make a reftest sample for reftest. +HOW TO USE: +1. Choose the first or last frame you want to generate. And set +window.onload function to dumpFirstFrame/dumpLastFrame. +2. Set the video.src in dumpFirstFrame/dumpLastFrame. +3. Run the script on browser. +4. Copy the base64 image url to your xxx-ref.html(short.mp4.firstframe-ref.html). +You might hit security error if the video.src cross origin. +Enable "media.seekToNextFrame.enabled" for the seekToNextFrame function +or using nightly, the seekToNextFrame() ensure the ended event fired. +*/ + +//window.onload = function() { setTimeout(dumpFirstFrame, 0); }; +//window.onload = function() { setTimeout(dumpLastFrame, 0); }; +window.onload = function() { setTimeout(function(){dumpNthFrame(15);}, 0); }; + +function drawVideoToInnerHTML(v) { + var canvas = document.getElementById("canvas"); + canvas.width = v.videoWidth; + canvas.height = v.videoHeight; + var ctx = canvas.getContext("2d"); + ctx.drawImage(v, 0, 0, v.videoWidth, v.videoHeight); + var dataURL = canvas.toDataURL(); + document.getElementById("out").innerHTML=dataURL; +} + +function dumpFirstFrame() { + var video = document.getElementById("v1"); + video.src = "short.mp4"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + drawVideoToInnerHTML(video); + }); +} + +function dumpNthFrame(n) { + var video = document.getElementById("v1"); + video.src = "street.mp4"; + video.preload = "metadata"; + + function checkNthFrame() { + console.log((15-n+1)+"th Frame time is " + video.currentTime); + n--; + if (n == 0) { + drawVideoToInnerHTML(video); + } else { + video.seekToNextFrame(); + } + } + video.addEventListener("loadeddata", checkNthFrame); + video.addEventListener("seeked", checkNthFrame); +} + +function dumpLastFrame() { + var video = document.getElementById("v1"); + video.src = "short.mp4"; + video.preload = "metadata"; + video.seenEnded = false; + // Seek to the end + video.addEventListener("loadeddata", function() { + video.currentTime = video.duration; + video.onseeked = () => { + video.onseeked = null; + callSeekToNextFrame(); + }; + }); + + function callSeekToNextFrame() { + video.seekToNextFrame().then( + () => { + if (!video.seenEnded) + callSeekToNextFrame(); + }, + () => { + // Reach the end, do nothing. + } + ); + } + + video.addEventListener("ended", function() { + video.seenEnded = true; + drawVideoToInnerHTML(video); + }); +} +</script> +</body> +</html> diff --git a/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html b/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html new file mode 100644 index 0000000000..28a93cc268 --- /dev/null +++ b/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html @@ -0,0 +1,7 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" +src=" + + +" +> diff --git a/dom/media/test/reftest/gizmo.mp4.seek.html b/dom/media/test/reftest/gizmo.mp4.seek.html new file mode 100644 index 0000000000..e4c1fe9515 --- /dev/null +++ b/dom/media/test/reftest/gizmo.mp4.seek.html @@ -0,0 +1,36 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<!--This testing should match the 55th frame of gizmo.mp4. The +55th frame's time is 1.8s, so seek to a time which is a little +greater than 1.8s, the display frame should be the 55th frame. +--> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "../gizmo.mp4"; + video.preload = "metadata"; + + video.currentTime = 1.801; + + video.addEventListener("seeked", function() { + // Since the our media pipeline send the frame to imageBridge, then fire + // seeked event, the target frame may not be shown on the screen. + // So using canvas to access the target frame in the imageContainer in + // videoElement. + var canvas = document.getElementById("canvas"); + canvas.width = video.videoWidth; + canvas.height = video.videoHeight; + var ctx = canvas.getContext("2d"); + ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight); + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +<canvas id="canvas" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html b/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html new file mode 100644 index 0000000000..5e9a8e9b4c --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);" + src="vp9hdr2020.png" +> diff --git a/dom/media/test/reftest/image-10bits-rendering-720-90-video.html b/dom/media/test/reftest/image-10bits-rendering-720-90-video.html new file mode 100644 index 0000000000..890aee1c50 --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-720-90-video.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "vp9hdr2020.webm"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"></video> +</body> +</html> diff --git a/dom/media/test/reftest/image-10bits-rendering-720-ref.html b/dom/media/test/reftest/image-10bits-rendering-720-ref.html new file mode 100644 index 0000000000..1ae393031a --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-720-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" + src="vp9hdr2020.png" +> diff --git a/dom/media/test/reftest/image-10bits-rendering-720-video.html b/dom/media/test/reftest/image-10bits-rendering-720-video.html new file mode 100644 index 0000000000..93d2651ffc --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-720-video.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "vp9hdr2020.webm"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/image-10bits-rendering-720.video.html b/dom/media/test/reftest/image-10bits-rendering-720.video.html new file mode 100644 index 0000000000..93d2651ffc --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-720.video.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "vp9hdr2020.webm"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/image-10bits-rendering-90-ref.html b/dom/media/test/reftest/image-10bits-rendering-90-ref.html new file mode 100644 index 0000000000..38f032f0fd --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-90-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);" + src="av1hdr2020.png" +> diff --git a/dom/media/test/reftest/image-10bits-rendering-90-video.html b/dom/media/test/reftest/image-10bits-rendering-90-video.html new file mode 100644 index 0000000000..d8a4eca5a9 --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-90-video.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "av1hdr2020.mp4"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"></video> +</body> +</html> diff --git a/dom/media/test/reftest/image-10bits-rendering-ref.html b/dom/media/test/reftest/image-10bits-rendering-ref.html new file mode 100644 index 0000000000..de7fbbfc95 --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" + src="av1hdr2020.png" +> diff --git a/dom/media/test/reftest/image-10bits-rendering-video.html b/dom/media/test/reftest/image-10bits-rendering-video.html new file mode 100644 index 0000000000..e6b3fda335 --- /dev/null +++ b/dom/media/test/reftest/image-10bits-rendering-video.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "av1hdr2020.mp4"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/reftest.list b/dom/media/test/reftest/reftest.list new file mode 100644 index 0000000000..5aeaec96e6 --- /dev/null +++ b/dom/media/test/reftest/reftest.list @@ -0,0 +1,8 @@ +skip-if(Android) fuzzy-if(OSX,0-80,0-76800) fuzzy-if(winWidget,0-62,0-76799) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-70,0-600) HTTP(..) == short.mp4.firstframe.html short.mp4.firstframe-ref.html +skip-if(Android) fuzzy-if(OSX,0-87,0-76797) fuzzy-if(winWidget,0-60,0-76797) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-60,0-1800) HTTP(..) == short.mp4.lastframe.html short.mp4.lastframe-ref.html +skip-if(Android) skip-if(winWidget) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-57,0-4281) fuzzy-if(OSX,55-80,4173-4417) HTTP(..) == bipbop_300_215kbps.mp4.lastframe.html bipbop_300_215kbps.mp4.lastframe-ref.html +skip-if(Android) fuzzy-if(OSX,0-25,0-175921) fuzzy-if(winWidget,0-71,0-179198) fuzzy-if((/^Windows\x20NT\x2010\.0/.test(http.oscpu))&&(/^aarch64-msvc/.test(xulRuntime.XPCOMABI)),0-255,0-179500) HTTP(..) == gizmo.mp4.seek.html gizmo.mp4.55thframe-ref.html +skip-if(Android) skip-if(MinGW) skip-if((/^Windows\x20NT\x2010\.0/.test(http.oscpu))&&(/^aarch64-msvc/.test(xulRuntime.XPCOMABI))) fuzzy(0-10,0-778236) == image-10bits-rendering-video.html image-10bits-rendering-ref.html +skip-if(Android) skip-if(MinGW) skip-if((/^Windows\x20NT\x2010\.0/.test(http.oscpu))&&(/^aarch64-msvc/.test(xulRuntime.XPCOMABI))) fuzzy(0-10,0-778536) == image-10bits-rendering-90-video.html image-10bits-rendering-90-ref.html +skip-if(Android) fuzzy(0-26,0-567562) == image-10bits-rendering-720-video.html image-10bits-rendering-720-ref.html +skip-if(Android) fuzzy(0-27,0-573249) == image-10bits-rendering-720-90-video.html image-10bits-rendering-720-90-ref.html diff --git a/dom/media/test/reftest/short.mp4.firstframe-ref.html b/dom/media/test/reftest/short.mp4.firstframe-ref.html new file mode 100644 index 0000000000..d80f2f985f --- /dev/null +++ b/dom/media/test/reftest/short.mp4.firstframe-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" + src="" +> diff --git a/dom/media/test/reftest/short.mp4.firstframe.html b/dom/media/test/reftest/short.mp4.firstframe.html new file mode 100644 index 0000000000..759bdc5ede --- /dev/null +++ b/dom/media/test/reftest/short.mp4.firstframe.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "../short.mp4"; + video.preload = "metadata"; + video.addEventListener("loadeddata", function() { + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/short.mp4.lastframe-ref.html b/dom/media/test/reftest/short.mp4.lastframe-ref.html new file mode 100644 index 0000000000..7474b9039e --- /dev/null +++ b/dom/media/test/reftest/short.mp4.lastframe-ref.html @@ -0,0 +1,4 @@ +<!DOCTYPE HTML> +<img style="position:absolute; left:0; top:0" +src="" +> diff --git a/dom/media/test/reftest/short.mp4.lastframe.html b/dom/media/test/reftest/short.mp4.lastframe.html new file mode 100644 index 0000000000..abd27c5c8e --- /dev/null +++ b/dom/media/test/reftest/short.mp4.lastframe.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html class="reftest-wait"> +<head> +<script type="text/javascript"> +function doTest() { + var video = document.getElementById("v1"); + video.src = "../short.mp4"; + video.preload = "metadata"; + video.seenEnded = false; + // Seek to the end + video.addEventListener("loadeddata", function() { + video.currentTime = video.duration; + video.onseeked = () => { + video.onseeked = null; + callSeekToNextFrame(); + }; + }); + + function callSeekToNextFrame() { + video.seekToNextFrame().then( + () => { + if (!video.seenEnded) + callSeekToNextFrame(); + }, + () => { + // Reach the end, do nothing. + } + ); + } + + video.addEventListener("ended", function() { + video.seenEnded = true; + document.documentElement.removeAttribute('class'); + }); +} +window.addEventListener("MozReftestInvalidate", doTest); +</script> +</head> +<body> +<video id="v1" style="position:absolute; left:0; top:0"></video> +</body> +</html> diff --git a/dom/media/test/reftest/vp9hdr2020.png b/dom/media/test/reftest/vp9hdr2020.png Binary files differnew file mode 100644 index 0000000000..afb68d9e0a --- /dev/null +++ b/dom/media/test/reftest/vp9hdr2020.png diff --git a/dom/media/test/reftest/vp9hdr2020.webm b/dom/media/test/reftest/vp9hdr2020.webm Binary files differnew file mode 100644 index 0000000000..516f62093a --- /dev/null +++ b/dom/media/test/reftest/vp9hdr2020.webm diff --git a/dom/media/test/resolution-change.webm b/dom/media/test/resolution-change.webm Binary files differnew file mode 100644 index 0000000000..29aad93b96 --- /dev/null +++ b/dom/media/test/resolution-change.webm diff --git a/dom/media/test/resolution-change.webm^headers^ b/dom/media/test/resolution-change.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/resolution-change.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4 b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4 Binary files differnew file mode 100644 index 0000000000..720339bdc2 --- /dev/null +++ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4 diff --git a/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sample-fisbone-skeleton4.ogv b/dom/media/test/sample-fisbone-skeleton4.ogv Binary files differnew file mode 100644 index 0000000000..8afe0be7a4 --- /dev/null +++ b/dom/media/test/sample-fisbone-skeleton4.ogv diff --git a/dom/media/test/sample-fisbone-skeleton4.ogv^headers^ b/dom/media/test/sample-fisbone-skeleton4.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sample-fisbone-skeleton4.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sample-fisbone-wrong-header.ogv b/dom/media/test/sample-fisbone-wrong-header.ogv Binary files differnew file mode 100644 index 0000000000..46c3933da5 --- /dev/null +++ b/dom/media/test/sample-fisbone-wrong-header.ogv diff --git a/dom/media/test/sample-fisbone-wrong-header.ogv^headers^ b/dom/media/test/sample-fisbone-wrong-header.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sample-fisbone-wrong-header.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sample.3g2 b/dom/media/test/sample.3g2 Binary files differnew file mode 100644 index 0000000000..769cb01dbd --- /dev/null +++ b/dom/media/test/sample.3g2 diff --git a/dom/media/test/sample.3gp b/dom/media/test/sample.3gp Binary files differnew file mode 100644 index 0000000000..4a3d8ea66f --- /dev/null +++ b/dom/media/test/sample.3gp diff --git a/dom/media/test/seek-short.ogv b/dom/media/test/seek-short.ogv Binary files differnew file mode 100644 index 0000000000..a5ca6951d0 --- /dev/null +++ b/dom/media/test/seek-short.ogv diff --git a/dom/media/test/seek-short.ogv^headers^ b/dom/media/test/seek-short.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/seek-short.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/seek-short.webm b/dom/media/test/seek-short.webm Binary files differnew file mode 100644 index 0000000000..36abd1570e --- /dev/null +++ b/dom/media/test/seek-short.webm diff --git a/dom/media/test/seek-short.webm^headers^ b/dom/media/test/seek-short.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/seek-short.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/seek.ogv b/dom/media/test/seek.ogv Binary files differnew file mode 100644 index 0000000000..ac7ece3519 --- /dev/null +++ b/dom/media/test/seek.ogv diff --git a/dom/media/test/seek.ogv^headers^ b/dom/media/test/seek.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/seek.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/seek.webm b/dom/media/test/seek.webm Binary files differnew file mode 100644 index 0000000000..72b0297233 --- /dev/null +++ b/dom/media/test/seek.webm diff --git a/dom/media/test/seek.webm^headers^ b/dom/media/test/seek.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/seek.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/seekLies.sjs b/dom/media/test/seekLies.sjs new file mode 100644 index 0000000000..3277950b08 --- /dev/null +++ b/dom/media/test/seekLies.sjs @@ -0,0 +1,23 @@ +function handleRequest(request, response) +{ + var file = Components.classes["@mozilla.org/file/directory_service;1"]. + getService(Components.interfaces.nsIProperties). + get("CurWorkD", Components.interfaces.nsIFile); + var fis = Components.classes['@mozilla.org/network/file-input-stream;1']. + createInstance(Components.interfaces.nsIFileInputStream); + var bis = Components.classes["@mozilla.org/binaryinputstream;1"]. + createInstance(Components.interfaces.nsIBinaryInputStream); + var paths = "tests/dom/media/test/seek.ogv"; + var split = paths.split("/"); + for(var i = 0; i < split.length; ++i) { + file.append(split[i]); + } + fis.init(file, -1, -1, false); + bis.setInputStream(fis); + var bytes = bis.readBytes(bis.available()); + response.setHeader("Content-Length", ""+bytes.length, false); + response.setHeader("Content-Type", "video/ogg", false); + response.setHeader("Accept-Ranges", "bytes", false); + response.write(bytes, bytes.length); + bis.close(); +} diff --git a/dom/media/test/seek_support.js b/dom/media/test/seek_support.js new file mode 100644 index 0000000000..b100572a5a --- /dev/null +++ b/dom/media/test/seek_support.js @@ -0,0 +1,61 @@ +// This file expects manifest.js to be included in the same scope. +/* import-globals-from manifest.js */ +// This file expects SEEK_TEST_NUMBER to be defined by the test. +/* global SEEK_TEST_NUMBER */ +var manager = new MediaTestManager(); + +function createTestArray() { + var tests = []; + var tmpVid = document.createElement("video"); + + for (var testNum = 0; testNum < gSeekTests.length; testNum++) { + var test = gSeekTests[testNum]; + if (!tmpVid.canPlayType(test.type)) { + continue; + } + + var t = {}; + t.name = test.name; + t.type = test.type; + t.duration = test.duration; + t.number = SEEK_TEST_NUMBER; + tests.push(t); + } + return tests; +} + +function startTest(test, token) { + var video = document.createElement("video"); + video.token = token += "-seek" + test.number + ".js"; + manager.started(video.token); + video.src = test.name; + video.preload = "metadata"; + document.body.appendChild(video); + var name = test.name + " seek test " + test.number; + var localIs = (function(n) { + return function(a, b, msg) { + is(a, b, n + ": " + msg); + }; + })(name); + var localOk = (function(n) { + return function(a, msg) { + ok(a, n + ": " + msg); + }; + })(name); + var localFinish = (function(v, m) { + return function() { + v.onerror = null; + removeNodeAndSource(v); + dump("SEEK-TEST: Finished " + name + " token: " + v.token + "\n"); + m.finished(v.token); + }; + })(video, manager); + dump("SEEK-TEST: Started " + name + "\n"); + window["test_seek" + test.number]( + video, + test.duration / 2, + localIs, + localOk, + localFinish + ); +} diff --git a/dom/media/test/seek_with_sound.ogg b/dom/media/test/seek_with_sound.ogg Binary files differnew file mode 100644 index 0000000000..c86d9946bd --- /dev/null +++ b/dom/media/test/seek_with_sound.ogg diff --git a/dom/media/test/seek_with_sound.ogg^headers^ b/dom/media/test/seek_with_sound.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/seek_with_sound.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/short-aac-encrypted-audio.mp4 b/dom/media/test/short-aac-encrypted-audio.mp4 Binary files differnew file mode 100644 index 0000000000..c1ba8d37e4 --- /dev/null +++ b/dom/media/test/short-aac-encrypted-audio.mp4 diff --git a/dom/media/test/short-aac-encrypted-audio.mp4^headers^ b/dom/media/test/short-aac-encrypted-audio.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/short-aac-encrypted-audio.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4 b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4 Binary files differnew file mode 100644 index 0000000000..bc58623999 --- /dev/null +++ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4 diff --git a/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/short-cenc-pssh-in-moof.mp4 b/dom/media/test/short-cenc-pssh-in-moof.mp4 Binary files differnew file mode 100644 index 0000000000..aa44c3b9a1 --- /dev/null +++ b/dom/media/test/short-cenc-pssh-in-moof.mp4 diff --git a/dom/media/test/short-cenc.mp4 b/dom/media/test/short-cenc.mp4 Binary files differnew file mode 100644 index 0000000000..aa44c3b9a1 --- /dev/null +++ b/dom/media/test/short-cenc.mp4 diff --git a/dom/media/test/short-cenc.xml b/dom/media/test/short-cenc.xml new file mode 100644 index 0000000000..9658c3e32f --- /dev/null +++ b/dom/media/test/short-cenc.xml @@ -0,0 +1,37 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- + This XML file describes the encryption applied to short-cenc.mp4. To generate + short-cenc, run the following command: + + MP4Box -crypt short-cenc.xml -out short-cenc.mp4 short.mp4 +--> + +<GPACDRM type="CENC AES-CTR"> + + <DRMInfo type="pssh" version="1"> + <!-- + SystemID specified in + https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html + --> + <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" /> + <!-- Number of KeyIDs = 2 --> + <BS bits="32" value="2" /> + <!-- KeyID --> + <BS ID128="0x7e571d017e571d017e571d017e571d01" /> + <BS ID128="0x7e571d027e571d027e571d027e571d02" /> + </DRMInfo> + + <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d017e571d017e571d017e571d01" + value="0x7e5711117e5711117e5711117e571111" /> + </CrypTrack> + + <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc" + first_IV="0x00000000000000000000000000000000"> + <key KID="0x7e571d027e571d027e571d027e571d02" + value="0x7e5722227e5722227e5722227e572222" /> + </CrypTrack> + +</GPACDRM> diff --git a/dom/media/test/short-video.ogv b/dom/media/test/short-video.ogv Binary files differnew file mode 100644 index 0000000000..68dee3cf2b --- /dev/null +++ b/dom/media/test/short-video.ogv diff --git a/dom/media/test/short-video.ogv^headers^ b/dom/media/test/short-video.ogv^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/short-video.ogv^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/short-vp9-encrypted-video.mp4 b/dom/media/test/short-vp9-encrypted-video.mp4 Binary files differnew file mode 100644 index 0000000000..23d1083d18 --- /dev/null +++ b/dom/media/test/short-vp9-encrypted-video.mp4 diff --git a/dom/media/test/short-vp9-encrypted-video.mp4^headers^ b/dom/media/test/short-vp9-encrypted-video.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/short-vp9-encrypted-video.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/short.mp4 b/dom/media/test/short.mp4 Binary files differnew file mode 100644 index 0000000000..802da047bc --- /dev/null +++ b/dom/media/test/short.mp4 diff --git a/dom/media/test/short.mp4.gz b/dom/media/test/short.mp4.gz Binary files differnew file mode 100644 index 0000000000..efb95e38e3 --- /dev/null +++ b/dom/media/test/short.mp4.gz diff --git a/dom/media/test/short.mp4^headers^ b/dom/media/test/short.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/short.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sine.webm b/dom/media/test/sine.webm Binary files differnew file mode 100644 index 0000000000..3913ffa874 --- /dev/null +++ b/dom/media/test/sine.webm diff --git a/dom/media/test/sine.webm^headers^ b/dom/media/test/sine.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sine.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm Binary files differnew file mode 100644 index 0000000000..7497096ba5 --- /dev/null +++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm Binary files differnew file mode 100644 index 0000000000..0f7609439d --- /dev/null +++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/small-shot-mp3.mp4 b/dom/media/test/small-shot-mp3.mp4 Binary files differnew file mode 100644 index 0000000000..61fe0ac719 --- /dev/null +++ b/dom/media/test/small-shot-mp3.mp4 diff --git a/dom/media/test/small-shot-mp3.mp4^headers^ b/dom/media/test/small-shot-mp3.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/small-shot-mp3.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/small-shot.flac b/dom/media/test/small-shot.flac Binary files differnew file mode 100644 index 0000000000..0da7c9044e --- /dev/null +++ b/dom/media/test/small-shot.flac diff --git a/dom/media/test/small-shot.m4a b/dom/media/test/small-shot.m4a Binary files differnew file mode 100644 index 0000000000..51a23c5b49 --- /dev/null +++ b/dom/media/test/small-shot.m4a diff --git a/dom/media/test/small-shot.mp3 b/dom/media/test/small-shot.mp3 Binary files differnew file mode 100644 index 0000000000..f9397a5106 --- /dev/null +++ b/dom/media/test/small-shot.mp3 diff --git a/dom/media/test/small-shot.mp3^headers^ b/dom/media/test/small-shot.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/small-shot.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/small-shot.ogg b/dom/media/test/small-shot.ogg Binary files differnew file mode 100644 index 0000000000..1a41623f81 --- /dev/null +++ b/dom/media/test/small-shot.ogg diff --git a/dom/media/test/small-shot.ogg^headers^ b/dom/media/test/small-shot.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/small-shot.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/sound.ogg b/dom/media/test/sound.ogg Binary files differnew file mode 100644 index 0000000000..edda4e9128 --- /dev/null +++ b/dom/media/test/sound.ogg diff --git a/dom/media/test/sound.ogg^headers^ b/dom/media/test/sound.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/sound.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/spacestorm-1000Hz-100ms.ogg b/dom/media/test/spacestorm-1000Hz-100ms.ogg Binary files differnew file mode 100644 index 0000000000..994041e1b0 --- /dev/null +++ b/dom/media/test/spacestorm-1000Hz-100ms.ogg diff --git a/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^ b/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/split.webm b/dom/media/test/split.webm Binary files differnew file mode 100644 index 0000000000..9207017fb6 --- /dev/null +++ b/dom/media/test/split.webm diff --git a/dom/media/test/split.webm^headers^ b/dom/media/test/split.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/split.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/street.mp4 b/dom/media/test/street.mp4 Binary files differnew file mode 100644 index 0000000000..837d23b383 --- /dev/null +++ b/dom/media/test/street.mp4 diff --git a/dom/media/test/street.mp4^headers^ b/dom/media/test/street.mp4^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/street.mp4^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-1-mono.opus b/dom/media/test/test-1-mono.opus Binary files differnew file mode 100644 index 0000000000..d5198e9ceb --- /dev/null +++ b/dom/media/test/test-1-mono.opus diff --git a/dom/media/test/test-1-mono.opus^headers^ b/dom/media/test/test-1-mono.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-1-mono.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-2-stereo.opus b/dom/media/test/test-2-stereo.opus Binary files differnew file mode 100644 index 0000000000..7115cac243 --- /dev/null +++ b/dom/media/test/test-2-stereo.opus diff --git a/dom/media/test/test-2-stereo.opus^headers^ b/dom/media/test/test-2-stereo.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-2-stereo.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-3-LCR.opus b/dom/media/test/test-3-LCR.opus Binary files differnew file mode 100644 index 0000000000..145536f3e7 --- /dev/null +++ b/dom/media/test/test-3-LCR.opus diff --git a/dom/media/test/test-3-LCR.opus^headers^ b/dom/media/test/test-3-LCR.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-3-LCR.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-4-quad.opus b/dom/media/test/test-4-quad.opus Binary files differnew file mode 100644 index 0000000000..731b867b2e --- /dev/null +++ b/dom/media/test/test-4-quad.opus diff --git a/dom/media/test/test-4-quad.opus^headers^ b/dom/media/test/test-4-quad.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-4-quad.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-5-5.0.opus b/dom/media/test/test-5-5.0.opus Binary files differnew file mode 100644 index 0000000000..7eb2faa812 --- /dev/null +++ b/dom/media/test/test-5-5.0.opus diff --git a/dom/media/test/test-5-5.0.opus^headers^ b/dom/media/test/test-5-5.0.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-5-5.0.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-6-5.1.opus b/dom/media/test/test-6-5.1.opus Binary files differnew file mode 100644 index 0000000000..526515eb7b --- /dev/null +++ b/dom/media/test/test-6-5.1.opus diff --git a/dom/media/test/test-6-5.1.opus^headers^ b/dom/media/test/test-6-5.1.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-6-5.1.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-7-6.1.opus b/dom/media/test/test-7-6.1.opus Binary files differnew file mode 100644 index 0000000000..8b6a7ce329 --- /dev/null +++ b/dom/media/test/test-7-6.1.opus diff --git a/dom/media/test/test-7-6.1.opus^headers^ b/dom/media/test/test-7-6.1.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-7-6.1.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-8-7.1.opus b/dom/media/test/test-8-7.1.opus Binary files differnew file mode 100644 index 0000000000..e56176a30f --- /dev/null +++ b/dom/media/test/test-8-7.1.opus diff --git a/dom/media/test/test-8-7.1.opus^headers^ b/dom/media/test/test-8-7.1.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-8-7.1.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test-stereo-phase-inversion-180.opus b/dom/media/test/test-stereo-phase-inversion-180.opus Binary files differnew file mode 100644 index 0000000000..ce70290002 --- /dev/null +++ b/dom/media/test/test-stereo-phase-inversion-180.opus diff --git a/dom/media/test/test-stereo-phase-inversion-180.opus^headers^ b/dom/media/test/test-stereo-phase-inversion-180.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/test-stereo-phase-inversion-180.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/test_VideoPlaybackQuality.html b/dom/media/test/test_VideoPlaybackQuality.html new file mode 100644 index 0000000000..67636f9ea6 --- /dev/null +++ b/dom/media/test/test_VideoPlaybackQuality.html @@ -0,0 +1,61 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test basic functionality of VideoPlaybackQuality</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function test() { + var video = document.createElement("video"); + ok(video.getVideoPlaybackQuality, "getVideoPlaybackQuality should be exposed with pref set"); + + var vpq = video.getVideoPlaybackQuality(); + ok(vpq, "getVideoPlaybackQuality should return an object"); + ok(vpq.creationTime <= performance.now(), "creationTime should be in the past"); + is(vpq.totalVideoFrames, 0, "totalVideoFrames should be 0"); + is(vpq.droppedVideoFrames, 0, "droppedVideoFrames should be 0"); + + var vpq2 = video.getVideoPlaybackQuality(); + ok(vpq !== vpq2, "getVideoPlaybackQuality should return a new object"); + ok(vpq.creationTime <= vpq2.creationTime, "VideoPlaybackQuality objects should have increasing creationTime"); + + var audio = document.createElement("audio"); + ok(!audio.getVideoPlaybackQuality, "getVideoPlaybackQuality should not be available on Audio elements"); + + video.src = "seek.webm"; + video.play(); + video.addEventListener("ended", function () { + vpq = video.getVideoPlaybackQuality(); + ok(vpq.creationTime <= performance.now(), "creationTime should be in the past"); + ok(vpq.totalVideoFrames > 0, "totalVideoFrames should be > 0"); + ok(vpq.droppedVideoFrames >= 0, "droppedVideoFrames should be >= 0"); + ok(vpq.droppedVideoFrames <= vpq.totalVideoFrames, "droppedVideoFrames should be <= totalVideoFrames"); + + SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", false]]}, function () { + vpq = video.getVideoPlaybackQuality(); + is(vpq.creationTime, 0, "creationTime should be 0"); + is(vpq.totalVideoFrames, 0, "totalVideoFrames should be 0"); + is(vpq.droppedVideoFrames, 0, "droppedVideoFrames should be 0"); + + SimpleTest.finish(); + }); + }); +} + +addLoadEvent(function() { + SpecialPowers.pushPrefEnv({"set": + [ + ["media.mediasource.enabled", true], + ] + }, test); +}); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_VideoPlaybackQuality_disabled.html b/dom/media/test/test_VideoPlaybackQuality_disabled.html new file mode 100644 index 0000000000..1c41b79d8b --- /dev/null +++ b/dom/media/test/test_VideoPlaybackQuality_disabled.html @@ -0,0 +1,37 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test basic functionality of VideoPlaybackQuality</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function test() { + var video = document.createElement("video"); + ok(!video.getVideoPlaybackQuality, "getVideoPlaybackQuality should be hidden behind a pref"); + var accessThrows = false; + try { + video.getVideoPlaybackQuality(); + } catch (e) { + accessThrows = true; + } + ok(accessThrows, "getVideoPlaybackQuality should be hidden behind a pref"); + SimpleTest.finish(); +} + +addLoadEvent(function() { + SpecialPowers.pushPrefEnv({"set": + [ + ["media.mediasource.enabled", false], + ] + }, test); +}); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_access_control.html b/dom/media/test/test_access_control.html new file mode 100644 index 0000000000..816563f9f5 --- /dev/null +++ b/dom/media/test/test_access_control.html @@ -0,0 +1,54 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=451958 +--> +<head> + <title>Test for Bug 451958</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=451958">Mozilla Bug 451958</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 451958 **/ + +function run() { + window.open("http://example.org:80/tests/dom/media/test/file_access_controls.html", "", "width=500,height=500"); +} + +function done() { + mediaTestCleanup(); + SimpleTest.finish(); +} + +addLoadEvent(run); +SimpleTest.waitForExplicitFinish(); + + +window.addEventListener("message", receiveMessage); + +function receiveMessage(event) +{ + if (event.origin !== "http://example.org") { + ok(false, "Received message from wrong domain"); + return; + } + + if (event.data.done == "true") { + done(); + return; + } + + ok(event.data.result, event.data.message); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_arraybuffer.html b/dom/media/test/test_arraybuffer.html new file mode 100644 index 0000000000..9ef84c53dc --- /dev/null +++ b/dom/media/test/test_arraybuffer.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=1462967 +--> +<head> + <title>Test for Bug 1457661</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1457661">Mozilla Bug 1457661</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Test for Bug 1457661; Ensure that readyState properly move to 4 when using arrayBuffer obtained from XMLHttpRequest + +var manager = new MediaTestManager; + +function getBlob(url, callback) { + const req = new XMLHttpRequest(); + req.addEventListener("load", function() { + callback(req.response); + }); + req.open("GET", url, true); + req.responseType = "arraybuffer"; + req.send(); +} + +function startTest(test, token) { + manager.started(token); + // Fetch the media resource using XHR so we can be sure the entire + // resource is loaded before we test buffered ranges. This ensures + // we have deterministic behaviour. + getBlob(test.name, function(arr) { + const v = document.createElement("video"); + const events = [ "suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause", "durationchange", "seeking", "seeked" ]; + function logEvent(e) { + info(test.name + ": got " + e.type + " event"); + } + events.forEach(function(e) { + v.addEventListener(e, logEvent); + }); + once(v, "stalled", function(e) { + // Resource fetch algorithm in local mode should never fire stalled event. + // https://html.spec.whatwg.org/multipage/media.html#concept-media-load-resource + ok(false, test.name + ": got stalled"); + removeNodeAndSource(v); + manager.finished(token); + }); + once(v, "canplaythrough", function(e) { + ok(true, test.name + ": got canplaythrough"); + is(v.readyState, v.HAVE_ENOUGH_DATA, test.name + ": readyState is HAVE_ENOUGH_DATA"); + removeNodeAndSource(v); + manager.finished(token); + }); + const blob = new Blob([arr], {type:test.type}); + const blobUrl = URL.createObjectURL(blob); + + document.body.appendChild(v); + v.preload = "auto"; + v.src = blobUrl; + v.load(); + }); +} + +SimpleTest.waitForExplicitFinish(); +// Note: No need to set media test prefs, since we're using XHR to fetch +// media data. +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_aspectratio_mp4.html b/dom/media/test/test_aspectratio_mp4.html new file mode 100644 index 0000000000..5e01875439 --- /dev/null +++ b/dom/media/test/test_aspectratio_mp4.html @@ -0,0 +1,46 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=975978 +--> + +<head> + <title>Media test: default video size</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=975978">Mozilla Bug 975978</a> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +// MP4 video with display size is difference to decode frame size. +// The display size is recorded in TrackHeaderBox 'tkhd' of this mp4 video. +var resource = + { name:"pixel_aspect_ratio.mp4", type:"video/mp4", width:525, height:288 }; + +var v = document.createElement("video"); +v.onloadedmetadata = function() { + is(v.videoWidth, resource.width, "Intrinsic width should match video width"); + is(v.videoHeight, resource.height, "Intrinsic height should match video height"); + SimpleTest.finish(); +} +v.addEventListener("error", function(ev) { + if (v.readyState < v.HAVE_METADATA) { + info("Video element returns with readyState " + v.readyState + " error.code " + v.error.code); + todo(false, "This platform doesn't support to retrieve MP4 metadata."); + SimpleTest.finish(); + } +}); + +v.src = resource.name; +v.preload = "auto"; + +document.body.appendChild(v); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_audio1.html b/dom/media/test/test_audio1.html new file mode 100644 index 0000000000..a5d9f12ab9 --- /dev/null +++ b/dom/media/test/test_audio1.html @@ -0,0 +1,33 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: Audio Constructor Test 1</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var a1 = new Audio(); + if (!a1.canPlayType(test.type)) + return; + manager.started(token); + + is(a1.getAttribute("preload"), "auto", "preload:auto automatically set"); + is(a1.src, "", "Src set?"); + a1 = null; + manager.finished(token); +} + +manager.runTests(gAudioTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_audio2.html b/dom/media/test/test_audio2.html new file mode 100644 index 0000000000..fde9adc791 --- /dev/null +++ b/dom/media/test/test_audio2.html @@ -0,0 +1,33 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: Audio Constructor Test 2</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var tmpAudio = new Audio(); + +var manager = new MediaTestManager; + +function startTest(test, token) { + if (!tmpAudio.canPlayType(test.type)) + return; + manager.started(token); + var a1 = new Audio(test.name); + is(a1.getAttribute("preload"), "auto", "Preload automatically set to auto"); + ok(a1.src.endsWith("/" + test.name), "src OK"); + manager.finished(token); +} + +manager.runTests(gAudioTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_audioDocumentTitle.html b/dom/media/test/test_audioDocumentTitle.html new file mode 100644 index 0000000000..2913debb39 --- /dev/null +++ b/dom/media/test/test_audioDocumentTitle.html @@ -0,0 +1,56 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=463830 +--> +<head> + <title>Test for Bug 463830</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463830">Mozilla Bug 463830</a> +<p id="display"></p> +<iframe id="i"></iframe> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 463830 **/ + +var gTests = [ + { file: "r11025_s16_c1.wav", title: "r11025_s16_c1.wav" } +]; + +var gTestNum = 0; + +addLoadEvent(runTest); + +var title; +var i = document.getElementById("i"); + +function runTest() { + if (gTestNum == gTests.length) { + SimpleTest.finish(); + return; + } + if (gTestNum == 0) { + i.addEventListener("load", function() { + is(i.contentDocument.title, title, "Doc title incorrect"); + setTimeout(runTest, 0); + }); + } + + title = gTests[gTestNum].title; + i.src = gTests[gTestNum].file; + gTestNum++; +} + +SimpleTest.waitForExplicitFinish(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_background_video_cancel_suspend_taint.html b/dom/media/test/test_background_video_cancel_suspend_taint.html new file mode 100644 index 0000000000..26691f3f43 --- /dev/null +++ b/dom/media/test/test_background_video_cancel_suspend_taint.html @@ -0,0 +1,70 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Suspend Cancels (Element Taint)</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +/** + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode resumes. + */ +function testSuspendTimerCanceledWhenTainted(video) { + function ended() { + video.removeEventListener("mozcancelvideosuspendtimer", canceled); + ok(false, `${video.token} ended before suspend cancels`); + this.ended_resolve(); + } + + function canceled() { + video.removeEventListener("ended", ended); + ok(true, `${video.token} suspend cancels`); + this.canceled_resolve(); + } + + let p = Promise.race([ + new Promise((resolve) => { + video.ended_resolve = resolve; + video.addEventListener('ended', ended, { 'once': true }); + }), + new Promise((resolve) => { + video.canceled_resolve = resolve; + video.addEventListener('mozcancelvideosuspendtimer', canceled, { 'once': true }); + }) + ]); + + Log(video.token, "Mark tainted"); + + let c = document.createElement('canvas'); + let g = c.getContext('2d'); + g.drawImage(video, 0, 0, c.width, c.height); + ok(video.hasSuspendTaint(), 'video used with drawImage is tainted.'); + + return p; +} + +startTest({ + desc: 'Test Background Video Suspend Cancels (Element Taint)', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 10000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilVisible(v) + .then(() => waitUntilPlaying(v)) + .then(() => testSuspendTimerStartedWhenHidden(v)) + .then(() => testSuspendTimerCanceledWhenTainted(v)) + .then(() => { manager.finished(token); }); + } +}); +</script> diff --git a/dom/media/test/test_background_video_cancel_suspend_visible.html b/dom/media/test/test_background_video_cancel_suspend_visible.html new file mode 100644 index 0000000000..e947b27734 --- /dev/null +++ b/dom/media/test/test_background_video_cancel_suspend_visible.html @@ -0,0 +1,69 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Suspend Cancels (Visibility)</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +/** + * Check that making the element visible before suspend timer delay causes the + * the suspend timer to be canceled. + * @param {HTMLMediaElement} video Video element under test. + * @returns {Promise} Promise that is resolved when video decode resumes. + */ +function testSuspendTimerCanceledWhenShown(video) { + function ended() { + video.removeEventListener("mozcancelvideosuspendtimer", canceled); + ok(false, `${video.token} ended before suspend cancels`); + this.ended_resolve(); + } + + function canceled() { + video.removeEventListener("ended", ended); + ok(true, `${video.token} suspend cancels`); + this.canceled_resolve(); + } + + let p = Promise.race([ + new Promise((resolve) => { + video.ended_resolve = resolve; + video.addEventListener('ended', ended, { 'once': true }); + }), + new Promise((resolve) => { + video.canceled_resolve = resolve; + video.addEventListener('mozcancelvideosuspendtimer', canceled, { 'once': true }); + }) + ]); + + Log(video.token, "Set visible"); + video.setVisible(true); + + return p; +} + +startTest({ + desc: 'Test Background Video Suspend Cancels (Visibility)', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 10000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilPlaying(v) + .then(() => testSuspendTimerStartedWhenHidden(v)) + .then(() => testSuspendTimerCanceledWhenShown(v)) + .then(() => { + ok(true, `${v.token} finished`); + manager.finished(token); + }); + } +}); diff --git a/dom/media/test/test_background_video_drawimage_with_suspended_video.html b/dom/media/test/test_background_video_drawimage_with_suspended_video.html new file mode 100644 index 0000000000..32bda51dbf --- /dev/null +++ b/dom/media/test/test_background_video_drawimage_with_suspended_video.html @@ -0,0 +1,76 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Displays Video Frame via drawImage When Suspended</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<style> +video, canvas { + border: 1px solid black; +} +</style> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +function drawVideoToCanvas(v) { + console.log('drawVideoToCanvas'); + let c = document.createElement('canvas'); + c.width = 4; + c.height = 4; + c.style.width = 64; + c.style.height = 64; + document.body.appendChild(c); + + let gfx = c.getContext('2d'); + if (!gfx) { + throw Error("Unable to obtain context '2d' from canvas"); + } + + gfx.drawImage(v, 0, 0, 4, 4); + let imageData = gfx.getImageData(0, 0, 4, 4); + let pixels = imageData.data; + + // Check that pixels aren't all the same colour. + // Implements by checking against rgb of the first pixel. + let rr = pixels[0], + gg = pixels[1], + bb = pixels[2], + allSame = true; + + for (let i = 0; i < 4*4; i++) { + let r = pixels[4*i+0]; + let g = pixels[4*i+1]; + let b = pixels[4*i+2]; + if (r != rr || g != gg || b != bb) { + allSame = false; + break; + } + } + + ok(!allSame, "Pixels aren't all the same color"); +} + +startTest({ + desc: 'Test Background Video Displays Video Frame via drawImage When Suspended', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 500 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilPlaying(v) + .then(() => testVideoSuspendsWhenHidden(v)) + .then(() => { + drawVideoToCanvas(v); + manager.finished(token); + }); + } +}); +</script> diff --git a/dom/media/test/test_background_video_ended_event.html b/dom/media/test/test_background_video_ended_event.html new file mode 100644 index 0000000000..99e2dd2f18 --- /dev/null +++ b/dom/media/test/test_background_video_ended_event.html @@ -0,0 +1,48 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Suspends</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +startTest({ + desc: "Test background video doesn't fire the 'ended' event twice.", + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 100 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + let count = 0; + v.addEventListener("ended", function() { + is(++count, 1, `${token} should get only one 'ended' event.`); + }); + + /* + * This test checks that, after a video element had finished its playback, + * resuming video decoder doesn't dispatch 2nd ended event. + * This issue was found in bug 1349097. + */ + waitUntilPlaying(v) + .then(() => testVideoSuspendsWhenHidden(v)) + .then(() => waitUntilEnded(v)) + .then(() => testVideoResumesWhenShown(v)) + .then(() => { + // Wait for a while and finish the test. We should get only one 'ended' event. + setTimeout(function() { + removeNodeAndSource(v); + manager.finished(token); + }, 3000); + }); + } +}); +</script> diff --git a/dom/media/test/test_background_video_no_suspend_disabled.html b/dom/media/test/test_background_video_no_suspend_disabled.html new file mode 100644 index 0000000000..f93977f2df --- /dev/null +++ b/dom/media/test/test_background_video_no_suspend_disabled.html @@ -0,0 +1,36 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Doesn't Suspend When Feature Disabled</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script> +"use strict"; + +var manager = new MediaTestManager; + +startTest({ + desc: "Test Background Video Doesn't Suspend When Feature Disabled.", + prefs: [ + [ 'media.test.video-suspend', true ], + [ 'media.suspend-bkgnd-video.enabled', false ], + [ 'media.suspend-bkgnd-video.delay-ms', 0 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + /* This test checks that suspend doesn't occur when the feature is disabled */ + waitUntilPlaying(v) + .then(() => checkVideoDoesntSuspend(v)) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); }) + .catch((e) => { + ok(false, 'Test Failed: ' + e.toString()); + manager.finished(token); }); + } +}); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_no_suspend_not_in_tree.html b/dom/media/test/test_background_video_no_suspend_not_in_tree.html new file mode 100644 index 0000000000..f65d363bd6 --- /dev/null +++ b/dom/media/test/test_background_video_no_suspend_not_in_tree.html @@ -0,0 +1,56 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Doesn't Suspend When Timeout Is Longer Than Video</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script> +"use strict"; + +var manager = new MediaTestManager; + +var MIN_DELAY = 100; + +/** + * @param {string} url video src. + * @returns {HTMLMediaElement} The created video element. + */ +function createVideoNotAppendToDoc(url, token, width, height) { + // Default size of (160, 120) is used by other media tests. + if (width === undefined) { width = 160; } + if (height === undefined) { height = 3*width/4; } + + let v = document.createElement('video'); + v.token = token; + v.width = width; + v.height = height; + v.src = url; + return v; +} + +startTest({ + desc: "Test Background Video Doesn't Suspend When If The Video Is Not In Tree.", + prefs: [ + [ 'media.test.video-suspend', true ], + [ 'media.suspend-bkgnd-video.enabled', true ], + [ 'media.suspend-bkgnd-video.delay-ms', MIN_DELAY ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = createVideoNotAppendToDoc(test.name, token); + manager.started(token); + + /* This test checks that suspend doesn't occur if a video element is not + append to tree. */ + waitUntilPlaying(v) + .then(() => checkVideoDoesntSuspend(v)) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); }) + .catch((e) => { + ok(false, 'Test Failed: ' + e.toString()); + manager.finished(token); }); + } +}); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_no_suspend_short_vid.html b/dom/media/test/test_background_video_no_suspend_short_vid.html new file mode 100644 index 0000000000..35597f6b96 --- /dev/null +++ b/dom/media/test/test_background_video_no_suspend_short_vid.html @@ -0,0 +1,38 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Doesn't Suspend When Timeout Is Longer Than Video</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script> +"use strict"; + +var manager = new MediaTestManager; + +startTest({ + desc: "Test Background Video Doesn't Suspend When Timeout Is Longer Than Video.", + prefs: [ + [ 'media.test.video-suspend', true ], + [ 'media.suspend-bkgnd-video.enabled', true ], + // Gizmo.mp4 is about 5.6s + [ 'media.suspend-bkgnd-video.delay-ms', 10000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + /* This test checks that suspend doesn't occur when the delay is longer + than the duration of the video that's playing */ + waitUntilPlaying(v) + .then(() => checkVideoDoesntSuspend(v)) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); }) + .catch((e) => { + ok(false, 'Test Failed: ' + e.toString()); + manager.finished(token); }); + } +}); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_resume_after_end_show_last_frame.html b/dom/media/test/test_background_video_resume_after_end_show_last_frame.html new file mode 100644 index 0000000000..767567e116 --- /dev/null +++ b/dom/media/test/test_background_video_resume_after_end_show_last_frame.html @@ -0,0 +1,131 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Suspends</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +function testSameContent(video1, video2) { + if (video1.videoWidth != video2.videoWidth || + video1.videoHeight != video2.videoHeight) { + ok(false, `${video1.token} video1 and video2 have different dimensions.`); + return; + } + + let w = video1.videoWidth; + let h = video1.videoHeight; + let c1 = document.createElement('canvas'); + let c2 = document.createElement('canvas'); + c1.width = w; + c1.height = h; + c2.width = w; + c2.height = h; + + let gfx1 = c1.getContext('2d'); + let gfx2 = c2.getContext('2d'); + if (!gfx1 || !gfx2) { + ok(false, "Unable to obtain context '2d' from canvas"); + return; + } + + gfx1.drawImage(video1, 0, 0, w, h); + gfx2.drawImage(video2, 0, 0, w, h); + + // Get content out. + let contentWidth = 4; + let contentHeight = 4; + let imageData1 = gfx1.getImageData(0, 0, contentWidth, contentHeight); + let imageData2 = gfx2.getImageData(0, 0, contentWidth, contentHeight); + let pixels1 = imageData1.data; + let pixels2 = imageData2.data; + + // Check that the content of two video are identical. + for (let i = 0; i < contentWidth*contentHeight; i++) { + let pixelCount = 4 * i; + if (pixels1[pixelCount+0] != pixels2[pixelCount+0] || + pixels1[pixelCount+1] != pixels2[pixelCount+1] || + pixels1[pixelCount+2] != pixels2[pixelCount+2] || + pixels1[pixelCount+3] != pixels2[pixelCount+3]) { + ok(false, `${video1.token} video1 and video2 have different content.`); + return; + } + } + + ok(true, `${video1.token} video1 and video2 have identical content.`); +} + +function waitUntilSeekToLastFrame(video) { + Log(video.token, "Waiting for seeking to the last frame"); + function callSeekToNextFrame() { + video.seekToNextFrame().then( + () => { + if (!video.seenEnded) { + callSeekToNextFrame(); + } + }, + () => { + // When seek reaches the end, the promise is resolved before 'ended' + // is fired. The resolver calls callSeekToNextFrame() to schedule + // another seek and then the 'ended' handler calls finish() to shut + // down the MediaDecoder which will reject the seek promise. So we don't + // raise an error in this case. + ok(video.seenEnded, "seekToNextFrame() failed."); + } + ); + } + + return new Promise(function(resolve, reject) { + video.seenEnded = false; + video.addEventListener("ended", () => { + video.seenEnded = true; + resolve(); + }, true); + callSeekToNextFrame(video); + }); +} + +startTest({ + desc: "Test resume an ended video shows the last frame.", + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 100 ], + [ "media.dormant-on-pause-timeout-ms", -1], + [ "media.decoder.skip-to-next-key-frame.enabled", false] + ], + tests: gDecodeSuspendTests, + + runTest: (test, token) => { + let v = appendVideoToDocWithoutLoad(token); + let vReference = appendVideoToDocWithoutLoad(token+"-ref"); + manager.started(token); + + /* + * This test checks that, after a video element had finished its playback, + * resuming video decoder should seek to the last frame. + * This issue was found in bug 1358057. + */ + waitUntilVisible(v) + .then(() => { + return Promise.all([loadAndWaitUntilLoadedmetadata(v, test.name), loadAndWaitUntilLoadedmetadata(vReference, test.name, "auto")]); + }) + .then(() => waitUntilPlaying(v)) + .then(() => testVideoSuspendsWhenHidden(v)) + .then(() => { + return Promise.all([waitUntilEnded(v), waitUntilSeekToLastFrame(vReference)]); + }) + .then(() => testVideoOnlySeekCompletedWhenShown(v)) + .then(() => { + testSameContent(v, vReference); + removeNodeAndSource(v); + removeNodeAndSource(vReference); + manager.finished(token); + }); + } +}); +</script> diff --git a/dom/media/test/test_background_video_resume_looping_video_without_audio.html b/dom/media/test/test_background_video_resume_looping_video_without_audio.html new file mode 100644 index 0000000000..53b380ce15 --- /dev/null +++ b/dom/media/test/test_background_video_resume_looping_video_without_audio.html @@ -0,0 +1,81 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Resume suspended looping video which doesn't contain audio track</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="manifest.js"></script> + <script src="background_video.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<script class="testbody" type="text/javascript"> +/** + * This test is used to ensure that the looping video (without audio track) which + * has been suspended can continute to playback correctly after we resume video + * decoding. + */ +async function startTest() { + const video = await createVisibleVideo(); + await startVideo(video); + await suspendVideoDecoding(video); + await resumeVideoDecoding(video); + await checkIfVideoIsStillPlaying(video); + endTestAndClearVideo(video); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({ 'set': [ + ["media.test.video-suspend", true], + ["media.suspend-bkgnd-video.enabled", true], + ["media.suspend-bkgnd-video.delay-ms", 0], + ]}, () => { + startTest(); +}); + +/** + * The following are test helper functions. + */ +async function createVisibleVideo() { + let video = document.createElement("video"); + video.src = "gizmo-noaudio.webm"; + video.controls = true; + video.loop = true; + document.body.appendChild(video); + info(`ensure video becomes visible`); + await waitUntilVisible(video); + return video; +} + +async function startVideo(video) { + info(`start playing video`); + const played = video && await video.play().then(() => true, () => false); + ok(played, "video has started playing"); +} + +async function suspendVideoDecoding(video) { + info(`suspend video decoding`); + video.setVisible(false); + await nextVideoSuspends(video); + info(`suspended video decoding`); +} + +async function resumeVideoDecoding(video) { + info(`resume video decoding.`); + video.setVisible(true); + await nextVideoResumes(video); + info(`resumed video decoding`); +} + +async function checkIfVideoIsStillPlaying(video) { + await once(video, "timeupdate"); + ok(!video.paused, "video is still playing after resuming video decoding.") +} + +function endTestAndClearVideo(video) { + removeNodeAndSource(video); + SimpleTest.finish(); +} + +</script> +</body> +</html> diff --git a/dom/media/test/test_background_video_suspend.html b/dom/media/test/test_background_video_suspend.html new file mode 100644 index 0000000000..a6a720ad13 --- /dev/null +++ b/dom/media/test/test_background_video_suspend.html @@ -0,0 +1,81 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Suspends</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css" /> +<script type="text/javascript"> + "use strict"; + + var manager = new MediaTestManager; + + var MIN_DELAY = 100; + + function testDelay(v, start, min) { + let end = performance.now(); + let delay = end - start; + ok(delay > min, `${v.token} suspended with a delay of ${delay} ms`); + } + + async function runTest(test, token) { + let video = appendVideoToDocWithoutLoad(token); + manager.started(token); + + let visible = waitUntilVisible(video); + let ended = nextVideoEnded(video); + let playing = nextVideoPlaying(video); + let resumes = nextVideoResumes(video); + let suspends = nextVideoSuspends(video); + + Log(token, "Waiting until video becomes visible"); + await visible; + + Log(token, "Waiting for metadata loaded"); + await loadAndWaitUntilLoadedmetadata(video, test.name); + + Log(token, "Start playing"); + video.play(); + + Log(token, "Waiting for video playing"); + await playing; + + let start = performance.now(); + + Log(token, "Set hidden"); + video.setVisible(false); + + Log(token, "Waiting for video suspend"); + await suspends; + + testDelay(video, start, MIN_DELAY); + + Log(token, "Set visible"); + video.setVisible(true); + + Log(token, "Waiting for video resume"); + await resumes; + + Log(token, "Waiting for ended"); + await ended; + + ok(video.currentTime >= video.duration, 'current time approximates duration.'); + + removeNodeAndSource(video); + manager.finished(token); + } + + startTest({ + desc: 'Test Background Video Suspends', + prefs: [ + ["media.test.video-suspend", true], + ["media.suspend-bkgnd-video.enabled", true], + // Use a short delay to ensure video decode suspend happens before end + // of video. + ["media.suspend-bkgnd-video.delay-ms", MIN_DELAY], + ["privacy.reduceTimerPrecision", false] + ], + tests: gDecodeSuspendTests, + runTest + }); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_suspend_ends.html b/dom/media/test/test_background_video_suspend_ends.html new file mode 100644 index 0000000000..c16ffff290 --- /dev/null +++ b/dom/media/test/test_background_video_suspend_ends.html @@ -0,0 +1,55 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Suspended Video Fires 'ended' Event</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css" /> +<script type="text/javascript"> + "use strict"; + + PARALLEL_TESTS = 1; + + var manager = new MediaTestManager; + + async function runTest(test, token) { + let video = appendVideoToDoc(test.name, token); + manager.started(token); + + // This test checks that 'ended' event is received for videos with + // suspended video decoding. This is important for looping video logic + // handling in HTMLMediaElement. + + let ended = nextVideoEnded(video); + let suspends = nextVideoSuspends(video); + + Log(token, "Start playing"); + video.play(); + + Log(token, "Set hidden"); + video.setVisible(false); + + Log(token, "Waiting for video suspend"); + await suspends; + + Log(token, "Waiting for ended"); + await ended; + + ok(video.currentTime >= video.duration, 'current time approximates duration.'); + + manager.finished(token); + } + + startTest({ + desc: "Test Background Suspended Video Fires 'ended' Event", + prefs: [ + ["media.test.video-suspend", true], + ["media.suspend-bkgnd-video.enabled", true], + // User a short delay to ensure video decode suspend happens before end + // of video. + ["media.suspend-bkgnd-video.delay-ms", 1000] + ], + tests: gDecodeSuspendTests, + runTest + }); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_tainted_by_capturestream.html b/dom/media/test/test_background_video_tainted_by_capturestream.html new file mode 100644 index 0000000000..5cbb40f3f7 --- /dev/null +++ b/dom/media/test/test_background_video_tainted_by_capturestream.html @@ -0,0 +1,46 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Is Tainted By captureStream</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +function captureVideoAsStream(v) { + v.mozCaptureStream(); +} + +startTest({ + desc: 'Test Background Video Is Tainted By captureStream', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 1000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + ok(true, `${test.name}`); + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilPlaying(v) + .then(() => { + captureVideoAsStream(v); + ok(v.hasSuspendTaint(), "Video is tainted after captured"); + return checkVideoDoesntSuspend(v); + }) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); + }) + .catch((e) => { + ok(false, 'Test failed: ' + e.toString()); + manager.finished(token); + }); + } +}); +</script> diff --git a/dom/media/test/test_background_video_tainted_by_createimagebitmap.html b/dom/media/test/test_background_video_tainted_by_createimagebitmap.html new file mode 100644 index 0000000000..cd884f9144 --- /dev/null +++ b/dom/media/test/test_background_video_tainted_by_createimagebitmap.html @@ -0,0 +1,42 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Is Tainted By createImageBitmap</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +startTest({ + desc: 'Test Background Video Is Tainted By createImageBitmap', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 1000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + ok(true, `${test.name}`); + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilPlaying(v) + .then(() => createImageBitmap(v)) + .then(() => { + ok(v.hasSuspendTaint(), "Video is tainted after drawing to canvas"); + return checkVideoDoesntSuspend(v); + }) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); + }) + .catch((e) => { + ok(false, 'Test failed: ' + e.toString()); + manager.finished(token); + }); + } +}); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_background_video_tainted_by_drawimage.html b/dom/media/test/test_background_video_tainted_by_drawimage.html new file mode 100644 index 0000000000..d5b8c75f3d --- /dev/null +++ b/dom/media/test/test_background_video_tainted_by_drawimage.html @@ -0,0 +1,58 @@ +<!DOCTYPE html> +<meta charset="utf-8"> +<title>Test Background Video Is Tainted By drawImage</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<script src="manifest.js"></script> +<script src="background_video.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +<script type="text/javascript"> +"use strict"; + +var manager = new MediaTestManager; + +function drawVideoToCanvas(v) { + let w = v.width, + h = v.height, + c = document.createElement('canvas'); + c.width = w; + c.height = h; + document.body.appendChild(c); + + let gfx = c.getContext('2d'); + if (!gfx) { + throw Error("Unable to obtain context '2d' from canvas"); + } + + gfx.drawImage(v, 0, 0, w, h); +} + +startTest({ + desc: 'Test Background Video Is Tainted By drawImage', + prefs: [ + [ "media.test.video-suspend", true ], + [ "media.suspend-bkgnd-video.enabled", true ], + [ "media.suspend-bkgnd-video.delay-ms", 1000 ] + ], + tests: gDecodeSuspendTests, + runTest: (test, token) => { + ok(true, `${test.name}`); + let v = appendVideoToDoc(test.name, token); + manager.started(token); + + waitUntilPlaying(v) + .then(() => { + drawVideoToCanvas(v); + ok(v.hasSuspendTaint(), "Video is tainted after drawing to canvas"); + return checkVideoDoesntSuspend(v); + }) + .then(() => { + ok(true, 'Video ended before decode was suspended'); + manager.finished(token); + }) + .catch((e) => { + ok(false, 'Test failed: ' + e.toString()); + manager.finished(token); + }); + } +}); +</script>
\ No newline at end of file diff --git a/dom/media/test/test_buffered.html b/dom/media/test/test_buffered.html new file mode 100644 index 0000000000..86d8eec28a --- /dev/null +++ b/dom/media/test/test_buffered.html @@ -0,0 +1,117 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=462957 +--> +<head> + <title>Test for Bug 462957</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=462957">Mozilla Bug 462957</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Test for Bug 462957; HTMLMediaElement.buffered. + +var manager = new MediaTestManager; + +function testBuffered(e) { + var v = e.target; + + // The whole media should be buffered... + var b = v.buffered; + is(b.length, 1, v._name + ": Should be buffered in one range"); + is(b.start(0), 0, v._name + ": First range start should be media start"); + ok(Math.abs(b.end(0) - v.duration) < 0.1, v._name + ": First range end should be media end"); + + // Ensure INDEX_SIZE_ERR is thrown when we access outside the range + var caught = false; + try { + b.start(-1); + } catch (ex) { + caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR; + } + is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on under start bounds range"); + + caught = false; + try { + b.end(-1); + } catch (ex) { + caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR; + } + is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on under end bounds range"); + + caught = false; + try { + b.start(b.length); + } catch (ex) { + caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR; + } + is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on over start bounds range"); + + caught = false; + try { + b.end(b.length); + } catch (ex) { + caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR; + } + is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on over end bounds range"); + + removeNodeAndSource(v); + manager.finished(v._token); +} + +function fetch(url, fetched_callback) { + var xhr = new XMLHttpRequest(); + xhr.open("GET", url, true); + xhr.responseType = "blob"; + + var loaded = function (event) { + if (xhr.status == 200 || xhr.status == 206) { + ok(true, `${url}: Fetch succeeded, status=${xhr.status}`); + // Request fulfilled. Note sometimes we get 206... Presumably because either + // httpd.js or Necko cached the result. + fetched_callback(window.URL.createObjectURL(xhr.response)); + } else { + ok(false, `${url}: Fetch failed, headers=${xhr.getAllResponseHeaders()}`); + } + }; + + xhr.addEventListener("load", loaded); + xhr.send(); +} + +function startTest(test, token) { + // Fetch the media resource using XHR so we can be sure the entire + // resource is loaded before we test buffered ranges. This ensures + // we have deterministic behaviour. + var onfetched = function(uri) { + var v = document.createElement('video'); + v._token = token; + v.src = uri; + v._name = test.name; + v._test = test; + v.addEventListener("loadeddata", testBuffered, {once: true}); + document.body.appendChild(v); + }; + + manager.started(token); + fetch(test.name, onfetched); +} + +// Note: No need to set media test prefs, since we're using XHR to fetch +// media data. +manager.runTests(gSeekTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug1113600.html b/dom/media/test/test_bug1113600.html new file mode 100644 index 0000000000..37a9cb0adb --- /dev/null +++ b/dom/media/test/test_bug1113600.html @@ -0,0 +1,50 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that a video element captured to a stream mid-playback can be played to the end</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +PARALLEL_TESTS = 1; +SimpleTest.requestCompleteLog(); +var manager = new MediaTestManager; + +function startTest(test, token) { + var v = document.createElement('video'); + v.style = "background-color:#aca;"; + v.width = 160; + v.height = 120; + + manager.started(token); + + v.src = test.name; + + v.ontimeupdate = function() { + if (v.currentTime < test.duration / 4) { + // Allow some time to pass before starting the capture. + return; + } + v.ontimeupdate = null; + v.mozCaptureStreamUntilEnded(); + info(test.name + " capture started at " + v.currentTime + ". Duration=" + test.duration); + }; + + v.onended = function() { + ok(true, test.name + " ended"); + removeNodeAndSource(v); + manager.finished(token); + }; + + document.body.appendChild(v); + v.play(); +} + +manager.runTests(gSmallTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug1242338.html b/dom/media/test/test_bug1242338.html new file mode 100644 index 0000000000..7b72153a6e --- /dev/null +++ b/dom/media/test/test_bug1242338.html @@ -0,0 +1,66 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Bug 1242338</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.preload = "metadata"; + video.token = token; + + var handler = { + "ontimeout": function() { + Log(token, "timed out"); + } + }; + manager.started(token, handler); + + video.src = test.name; + video.name = test.name; + + function finish() { + video.finished = true; + video.removeEventListener("loadedmetadata", onLoadedmetadata); + video.removeEventListener("ended", onEnded); + removeNodeAndSource(video); + manager.finished(video.token); + } + + function onLoadedmetadata() { + // seek to the media's duration + var duration = video.duration; + console.log("onloadedmetadata(), duration = " + duration); + video.currentTime = duration; + } + + function onEnded() { + ok(video.ended, test.name + " checking playback has ended"); + ok(!video.finished, test.name + " shouldn't be finished"); + ok(!video.seenEnded, test.name + " shouldn't be ended"); + video.seenEnded = true; + + ok(true, "Seeking to the duration triggers ended event"); + finish(); + } + + video.addEventListener("loadedmetadata", onLoadedmetadata); + video.addEventListener("ended", onEnded); + + document.body.appendChild(video); +} + +manager.runTests(gSeekTests, startTest); + +</script> +</pre> +</body> +</html>
\ No newline at end of file diff --git a/dom/media/test/test_bug1248229.html b/dom/media/test/test_bug1248229.html new file mode 100644 index 0000000000..3165795622 --- /dev/null +++ b/dom/media/test/test_bug1248229.html @@ -0,0 +1,35 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test garbage collection of captured stream (bug 1248229)</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="doTest()"> +<video id="v" src="black100x100-aspect3to2.ogv"></video> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +function doTest() { + /* global v */ + window.oak = v.mozCaptureStreamUntilEnded(); + v.mozCaptureStreamUntilEnded(); + v.play(); + + v.onended = function() { + info("Got ended."); + v.onended = null; + SpecialPowers.exactGC(function() { + info("GC completed."); + v.play(); + SimpleTest.finish(); + }); + } +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug1431810_opus_downmix_to_mono.html b/dom/media/test/test_bug1431810_opus_downmix_to_mono.html new file mode 100644 index 0000000000..647ddf0489 --- /dev/null +++ b/dom/media/test/test_bug1431810_opus_downmix_to_mono.html @@ -0,0 +1,139 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: disable phase inversion in opus decoder</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<audio preload=none id="a" controls></audio> +<audio preload=none id="b" controls></audio> +<script class="testbody" type="text/javascript"> +/* + This test makes use of an (stereo) opus file with phase inversion of 180 degrees (right = -left => right + left = 0). + Firstly, the phase inversion is verified on a normal stereo playback. + Secondly, mono playback is forced which results in the phase inversion being disabled (Bug 1431810). +*/ +SimpleTest.waitForExplicitFinish(); + +/* global a, b */ + +function areChannelsInverted(b1, b2) { + for (var i = 0; i < b1.length; i++) { + if (Math.abs(b1[i] + b2[i]) > 9e-2) { + return false; + } + } + return true; +} + +function areChannelsEqual(b1, b2) { + for (var i = 0; i < b1.length; i++) { + if (Math.abs(b1[i] - b2[i]) > 9e-3) { + return false; + } + } + return true; +} + +function isSilent(b) { + for (var i = 0; i < b.length; i++) { + if (b[i] != 0.0) { + return false; + } + } + return true; +} + +function mediaElementWithPhaseInversion(audioContext, mediaElement, success) { + let audio_source = audioContext.createMediaElementSource(mediaElement); + let script_processor = audioContext.createScriptProcessor(); + audio_source.connect(script_processor); + + mediaElement.onplay = () => { + script_processor.onaudioprocess = (e) => { + let right = e.inputBuffer.getChannelData(0); + let left = e.inputBuffer.getChannelData(1); + + // This is leading or trailing silence + // produced by ScriptProcessor. + if (isSilent(right) && isSilent(left)) { + return; + } + + ok(areChannelsInverted(right, left), "Channels must be inverted"); + } + } + + mediaElement.onended = () => { + ok(true, "End of file."); + mediaElement.onended = null; + script_processor.onaudioprocess = null; + success(); + } + + mediaElement.src = "test-stereo-phase-inversion-180.opus"; + // Normal playback channels will by inverted + mediaElement.play(); +} + +function mediaElementWithPhaseInversionDisabled(audioContext, mediaElement, success) { + let audio_source = audioContext.createMediaElementSource(mediaElement); + let script_processor = audioContext.createScriptProcessor(); + audio_source.connect(script_processor); + + mediaElement.onplay = () => { + script_processor.onaudioprocess = (e) => { + let right = e.inputBuffer.getChannelData(0); + let left = e.inputBuffer.getChannelData(1); + + // This is leading or trailing silence + // produced by ScriptProcessor. + if (isSilent(right) && isSilent(left)) { + return; + } + + ok(!areChannelsInverted(right, left), "Channels must not be inverted"); + ok(areChannelsEqual(right, left), "Channels must be equal"); + } + } + + mediaElement.onended = () => { + ok(true, "End of file."); + mediaElement.onended = null; + script_processor.onaudioprocess = null; + success(); + } + + mediaElement.src = "test-stereo-phase-inversion-180.opus"; + + // Downmix to mono will force to disable opus phase inversion + SpecialPowers.pushPrefEnv({"set": [["accessibility.monoaudio.enable", true]]}) + .then(() => { + mediaElement.play(); + }); +} + +let ac = new AudioContext(); + +function testPhaseInversion(mediaElement) { + return new Promise((accept, reject) => { + mediaElementWithPhaseInversion(ac, a, accept); + }); +} + +function testPhaseInversionDisabled(mediaElement) { + return new Promise((accept, reject) => { + mediaElementWithPhaseInversionDisabled(ac, b, accept); + }); +} + +// Start testing +testPhaseInversion(a) +.then( () => testPhaseInversionDisabled(b) ) +.then( () => SimpleTest.finish() ) + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug1512958.html b/dom/media/test/test_bug1512958.html new file mode 100644 index 0000000000..2513515542 --- /dev/null +++ b/dom/media/test/test_bug1512958.html @@ -0,0 +1,74 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that pausing and resuming a captured media element with audio doesn't stall</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<audio id="a"></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +function dumpEvent({target, type}) { + info(`${target.name} GOT EVENT ${type} currentTime=${target.currentTime} ` + + `paused=${target.paused} ended=${target.ended} ` + + `readyState=${target.readyState}`); +} + +function wait(ms) { + return new Promise(resolve => setTimeout(resolve, ms)); +} + +const a = document.getElementById('a'); + +const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"]; +for (let ev of events) { + a.addEventListener(ev, dumpEvent); +} + +(async () => { + try { + SimpleTest.waitForExplicitFinish(); + SimpleTest.requestFlakyTimeout("Timeouts for shortcutting test-timeout"); + + const test = getPlayableAudio(gTrackTests.filter(t => t.duration > 2)); + if (!test) { + todo(false, "No playable audio"); + return; + } + + // Start playing and capture + a.src = test.name; + a.name = test.name; + const ac = new AudioContext(); + ac.createMediaElementSource(a); + a.play(); + do { + await new Promise(r => a.ontimeupdate = r); + } while(a.currentTime == 0) + + // Pause to trigger recreating tracks in DecodedStream + a.pause(); + await new Promise(r => a.onpause = r); + + // Resuming should now work. Bug 1512958 would cause a stall because the + // original track wasn't ended and we'd block on it. + // See https://bugzilla.mozilla.org/show_bug.cgi?id=1512958#c5 + a.play(); + await new Promise(r => a.onplaying = r); + a.currentTime = test.duration - 1; + await Promise.race([ + new Promise(res => a.onended = res), + wait(30000).then(() => Promise.reject(new Error("Timeout"))), + ]); + } catch(e) { + ok(false, e); + } finally { + SimpleTest.finish(); + } +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug1553262.html b/dom/media/test/test_bug1553262.html new file mode 100644 index 0000000000..19b937f7fe --- /dev/null +++ b/dom/media/test/test_bug1553262.html @@ -0,0 +1,31 @@ +<!DOCTYPE HTML> +<html> +<head> +<meta charset="utf-8"> +<title>Bug 1553262</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<link rel="stylesheet" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<script> + const canvas = document.createElement('canvas') + const context = canvas.getContext('2d') + + const video = document.createElement('video') + const source = new AudioContext().createMediaElementSource(video) + const stream = canvas.captureStream() + + const xhr = new XMLHttpRequest() + + context.rect(256, -32768, 16, 16) + video.srcObject = stream + for (let i = 0; i < 16; i++) { + xhr.open('P', '', false) + xhr.send() + } + + video.srcObject = stream + ok(true, "success") +</script> +</body> +</html> diff --git a/dom/media/test/test_bug448534.html b/dom/media/test/test_bug448534.html new file mode 100644 index 0000000000..ed1135d85a --- /dev/null +++ b/dom/media/test/test_bug448534.html @@ -0,0 +1,71 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=448534 +--> + +<head> + <title>Test for Bug 448534</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=448535">Mozilla Bug 448534</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function loaded(event) { + var v = event.target; + info(v.token + ": event=" + event.type); + if (v._finished) + return; + v.play(); +} + +function started(event) { + var v = event.target; + info(v.token + ": event=" + event.type); + // For a short file, it could reach the end before 'play' received. We will + // skip the test for 'paused' would be true when ended. + if (v._finished || v.ended) + return; + ok(!v.paused, v.token + ": Video should not be paused while playing"); + v.remove(); + v._played = true; +} + +function stopped(event) { + var v = event.target; + info(v.token + ": event=" + event.type); + if (v._finished) + return; + v._finished = true; + ok(v.paused, v.token + ": Video should be paused after removing from the Document"); + removeNodeAndSource(v); + manager.finished(v.token); +} + + +function startTest(test, token) { + var v = document.createElement('video'); + v.preload = "metadata"; + v.token = token; + manager.started(token); + v.src = test.name; + v._played = false; + v._finished = false; + v.addEventListener("loadedmetadata", loaded); + v.addEventListener("play", started); + v.addEventListener("pause", stopped); + document.body.appendChild(v); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug463162.xhtml b/dom/media/test/test_bug463162.xhtml new file mode 100644 index 0000000000..a84b3488d1 --- /dev/null +++ b/dom/media/test/test_bug463162.xhtml @@ -0,0 +1,78 @@ +<html xmlns="http://www.w3.org/1999/xhtml" + xmlns:html="http://www.w3.org/1999/xhtml" + xmlns:svg="http://www.w3.org/2000/svg"> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=463162 +--> +<head> + <title>Test for Bug 463162</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/SimpleTest/EventUtils.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463162">Mozilla Bug 463162</a> + +<script class="testbody" type="text/javascript"> +<![CDATA[ + +var gExpectedResult = { + 'a1' : 'error', + 'a2' : 'loaded', + 'a3' : 'loaded', + 'a4' : 'error', +}; + +var gResultCount = 0; + +function onError(event, id) { + is('error', gExpectedResult[id], 'unexpected error loading ' + id); + gResultCount++; + dump('error('+id+') expected ' + gExpectedResult[id] + ' gResultCount=' + gResultCount + '\n'); + if (gResultCount == 4) + SimpleTest.finish(); +} + +function onMetaData(id) { + is('loaded', gExpectedResult[id], 'unexpected loadedmetadata loading ' + id); + gResultCount++; + dump('onMetaData('+id+') expected ' + gExpectedResult[id] + ' gResultCount=' + gResultCount + '\n'); + if (gResultCount == 4) + SimpleTest.finish(); +} + +]]> +</script> + +<video id="a1" preload="metadata" onloadedmetadata="onMetaData('a1');"><sauce/><source type="bad" src="404" onerror="onError(event, 'a1');"/></video> +<video id="a2" preload="metadata" onloadedmetadata="onMetaData('a2');"><source onerror="onError(event, 'a2');"/></video> +<video id="a3" preload="metadata" onloadedmetadata="onMetaData('a3');"><html:source onerror="onError(event, 'a3');"/></video> +<video id="a4" preload="metadata" onloadedmetadata="onMetaData('a4');"><svg:source/><source onerror="onError(event, 'a4');" type="bad" src="404"/></video> + +<script class="testbody" type="text/javascript"> +<![CDATA[ + +function setSource(id, res) { + var v = document.getElementById(id); + v.firstChild.src = res.name; + v.firstChild.type = res.type; +} + +var t = getPlayableVideo(gSmallTests); + +setSource('a1', t); +setSource('a2', t); +setSource('a3', t); +setSource('a4', t); + +SimpleTest.waitForExplicitFinish(); + +]]> +</script> + +<pre id="test"> + +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug465498.html b/dom/media/test/test_bug465498.html new file mode 100644 index 0000000000..157a972e77 --- /dev/null +++ b/dom/media/test/test_bug465498.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: Bug 465498 - Seeking after playback ended</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=465498">Mozilla Bug 465498</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(e) { + var v = e.target; + info(v._name + " loadedmetadata"); + e.target.play(); +} + +function playbackEnded(e) { + var v = e.target; + info(v._name + " ended"); + if (v._finished) + return; + ok(v.currentTime >= v.duration - 0.1 && v.currentTime <= v.duration + 0.1, + "Checking currentTime at end: " + v.currentTime + " for " + v._name); + ok(v.ended, "Checking playback has ended for " + v._name); + v.pause(); + v.currentTime = 0; + ok(!v.ended, "Checking ended is no longer true for " + v._name); + v._seeked = true; +} + +function seekEnded(e) { + var v = e.target; + info(v._name + " seeked"); + if (v._finished) + return; + ok(v.currentTime == 0, "Checking currentTime after seek: " + + v.currentTime + " for " + v._name); + ok(!v.ended, "Checking ended is false for " + v._name); + v._finished = true; + removeNodeAndSource(v); + manager.finished(v.token); +} + +function seeking(e) { + var v = e.target; + info(v._name + " seeking"); +} + +function initTest(test, token) { + var type = getMajorMimeType(test.type); + var v = document.createElement(type); + if (!v.canPlayType(test.type)) + return; + v.preload = "metadata"; + v.token = token; + manager.started(token); + v._name = test.name; + + var s = document.createElement("source"); + s.type = test.type; + s.src = test.name; + v.appendChild(s); + + v._seeked = false; + v._finished = false; + v.addEventListener("loadedmetadata", startTest); + v.addEventListener("ended", playbackEnded); + v.addEventListener("seeked", seekEnded); + v.addEventListener("seeking", seeking); + document.body.appendChild(v); +} + +manager.runTests(gSmallTests, initTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug495145.html b/dom/media/test/test_bug495145.html new file mode 100644 index 0000000000..3686e10270 --- /dev/null +++ b/dom/media/test/test_bug495145.html @@ -0,0 +1,95 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=495145 +--> + +<head> + <title>Bug 495145 - pausing while ended shouldn't cause problems</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495145">Mozilla Bug 495145</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function start(e) { + e.target.play(); +} + +function ended1(e) { + var v = e.target; + if (v._finished) + return; + + ++v._endCount; + if (v._endCount == 2) { + ok(true, "Playing after pause while ended works for " + v._name); + v._finished = true; + v.removeEventListener("loadedmetadata", start); + v.removeEventListener("ended", ended1); + removeNodeAndSource(v); + manager.finished(v.token); + return; + } + + v.pause(); + v.play(); +} + +function ended2(e) { + var v = e.target; + if (v._finished) + return; + + v.pause(); + v.currentTime = 0; +} + +function seeked2(e) { + var v = e.target; + if (v._finished) + return; + + ok(v.paused, "Paused after seek after pause while ended for " + v._name); + v._finished = true; + v.removeEventListener("loadedmetadata", start); + v.removeEventListener("ended", ended2); + v.removeEventListener("seeked", seeked2); + removeNodeAndSource(v); + manager.finished(v.token); +} + +function createVideo(test, x, token) { + var v = document.createElement('video'); + v.preload = "metadata"; + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name + "#" + x; + v._endCount = 0; + v._finished = false; + v.addEventListener("loadedmetadata", start); + v.addEventListener("ended", x == 1 ? ended1 : ended2); + if (x == 2) + v.addEventListener("seeked", seeked2); + document.body.appendChild(v); +} + +function startTest(test, token) { + createVideo(test, 1, token + "a"); + createVideo(test, 2, token + "b"); +} + +manager.runTests(gSmallTests, startTest); + + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug495300.html b/dom/media/test/test_bug495300.html new file mode 100644 index 0000000000..bf3b921402 --- /dev/null +++ b/dom/media/test/test_bug495300.html @@ -0,0 +1,63 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=495300 +--> + +<head> + <title>Bug 495300 - seeking to the end should behave as "ended"</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495300">Mozilla Bug 495300</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function filename(uri) { + return uri.substr(uri.lastIndexOf("/")+1); +} + +function mediaEnded(event) { + ok(true, "Got expected 'ended' event: " + filename(event.target.currentSrc)); + + if (event.target._expectedDuration) + ok(Math.abs(event.target.currentTime - event.target._expectedDuration) < 0.1, + "currentTime equals duration: " + filename(event.target.currentSrc)); + + event.target.removeEventListener("ended", mediaEnded); + manager.finished(event.target.token); + removeNodeAndSource(event.target); +} + +function mediaLoadedmetadata(event) { + event.target.currentTime = event.target.duration; + event.target.removeEventListener("loadedmetadata", mediaLoadedmetadata); +} + +function startTest(test, token) { + var elemType = /^audio/.test(test.type) ? "audio" : "video"; + var v1 = document.createElement(elemType); + v1.preload = "auto"; + + v1.src = test.name; + if (test.duration) { + v1._expectedDuration = test.duration; + } + v1.addEventListener("loadedmetadata", mediaLoadedmetadata); + v1.addEventListener("ended", mediaEnded); + v1.load(); + + v1.token = token; + manager.started(token); +} + +manager.runTests(gSeekTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug654550.html b/dom/media/test/test_bug654550.html new file mode 100644 index 0000000000..f57afae2b4 --- /dev/null +++ b/dom/media/test/test_bug654550.html @@ -0,0 +1,84 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=654550 +--> + +<head> + <title>Test for Bug 654550</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=654550">Mozilla Bug 654550</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + + /* Test for Bug 654550 */ + + // Parallel test must be disabled for media.video_stats.enabled is a global setting + // to prevent the setting from changing unexpectedly in the middle of the test. + PARALLEL_TESTS = 1; + SimpleTest.waitForExplicitFinish(); + var manager = new MediaTestManager; + + function checkStats(v, aShouldBeEnabled) { + if (aShouldBeEnabled) { + ok(v.mozParsedFrames != 0, + "At least one value should be different from 0 if stats are enabled"); + } else { + ok(!v.mozParsedFrames, + "mozParsedFrames should be 0 if stats are disabled"); + ok(!v.mozDecodedFrames, + "mozDecodedFrames should be 0 if stats are disabled"); + ok(!v.mozPresentedFrames, + "mozPresentedFrames should be 0 if stats are disabled"); + ok(!v.mozPaintedFrames, + "mozPaintedFrames should be 0 if stats are disabled"); + } + + } + + function ontimeupdate_statsEnabled(event) { + var v = event.target; + v.removeEventListener('timeupdate', ontimeupdate_statsEnabled); + checkStats(v, true); + SpecialPowers.popPrefEnv( + function() { + v.addEventListener("timeupdate", ontimeupdate_statsDisabled); + }); + } + + function ontimeupdate_statsDisabled(event) { + var v = event.target; + v.removeEventListener('timeupdate', ontimeupdate_statsDisabled); + checkStats(v, false); + removeNodeAndSource(v); + manager.finished(v.token); + } + + function startTest(test, token) { + var v = document.createElement('video'); + v.token = token; + v.src = test.name; + // playback may reach the end before pref is changed for the duration is short + // set 'loop' to true to keep playing so that we won't miss 'timeupdate' events + v.loop = true; + manager.started(token); + SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", true]]}, + function() { + v.play(); + v.addEventListener("timeupdate", ontimeupdate_statsEnabled); + }); + } + + SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", false]]}, + function() { + manager.runTests(gVideoTests, startTest); + }); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug686942.html b/dom/media/test/test_bug686942.html new file mode 100644 index 0000000000..66b48d69f0 --- /dev/null +++ b/dom/media/test/test_bug686942.html @@ -0,0 +1,68 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=686942 +--> + +<head> + <title>Test for Bug 448534</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=686942">Mozilla Bug 686942</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function onloaded(event) { + var v = event.target; + v.removeEventListener("loadedmetadata", onloaded); + v.currentTime = v.duration; + +} + +function checkNotPlaying(v) { + ok(v.currentTime == 0, "Should not be playing after seek to end and back to beginning"); + v._finished = true; + manager.finished(v.token); + removeNodeAndSource(v); +} + +function onseeked(event) { + var v = event.target; + v.removeEventListener("seeked", onseeked); + setTimeout(function() { checkNotPlaying(v); }, 500); +} + +function onended(event) { + var v = event.target; + v.removeEventListener("ended", onended); + if (v._finished) + return; + v.addEventListener("seeked", onseeked); + v.currentTime = 0; +} + +function startTest(test, token) { + var v = document.createElement('video'); + v.preload = "auto"; + v.token = token; + manager.started(token); + v.src = test.name; + v._played = false; + v._finished = false; + v.addEventListener("loadedmetadata", onloaded); + v.addEventListener("ended", onended); + + document.body.appendChild(v); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug726904.html b/dom/media/test/test_bug726904.html new file mode 100644 index 0000000000..bf8e2536ca --- /dev/null +++ b/dom/media/test/test_bug726904.html @@ -0,0 +1,56 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=726904 +--> + +<head> + <title>Media test: default video size</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="bodyLoaded();"> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=726904">Mozilla Bug 726904</a> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +var v1 = document.createElement("video"), + v2 = document.createElement("video"), + poster = "", + resource = getPlayableVideo(gSmallTests); + +function bodyLoaded(){ + // Note: For DASH, width and height would vary once the video started playing, so + // the values would not correlate with those in manifest.js. Since this test has + // no playing, this should not affect the result. + is(v1.videoWidth, resource.width, "Intrinsic width should match video width"); + is(v1.videoHeight, resource.height, "Intrinsic height should match video height"); + is(v2.clientWidth, 400, "clientWidth should be 400"); + is(v2.clientHeight, 400, "clientHeight should be 400"); + SimpleTest.finish(); +} + +if (resource) { + v1.poster = v2.poster = poster; + + v1.src = v2.src = "http://mochi.test:8888/tests/dom/media/test/" + resource.name; + + v1.preload = "auto"; + v2.preload = "none"; + + v1.muted = v2.muted = true; + + document.body.appendChild(v1); + document.body.appendChild(v2); +} else { + todo(false, "No types supported"); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug874897.html b/dom/media/test/test_bug874897.html new file mode 100644 index 0000000000..7801487fe3 --- /dev/null +++ b/dom/media/test/test_bug874897.html @@ -0,0 +1,68 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=874897 +--> + +<head> + <title>Test for Bug 874897</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function loadeddata(e) { + var v = e.target; + ok(v.readyState >= v.HAVE_CURRENT_DATA, + "readyState must be >= HAVE_CURRENT_DATA for " + v._name); + + var canvas = document.createElement("canvas"); + canvas.width = 210; + canvas.height = 120; + document.body.appendChild(canvas); + var ctx = canvas.getContext("2d"); + try { + ctx.drawImage(v, 0, 0, v.videoWidth, v.videoHeight, 0, 0, canvas.width, canvas.height); + ok(true, "Shouldn't throw exception while drawing to canvas from video for " + v._name); + } catch (ex) { + ok(false, "Shouldn't throw exception while drawing to canvas from video for " + v._name); + } + + v._finished = true; + v.remove(); + manager.finished(v.token); +} + +function startTest(test, token) { + var type = getMajorMimeType(test.type); + if (type != "video") + return; + + var v = document.createElement('video'); + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name; + v._finished = false; + v.autoplay = true; + v.style.display = "none"; + v.addEventListener("loadeddata", loadeddata); + document.body.appendChild(v); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest); +function beginTest() { + manager.runTests(gAspectRatioTests, startTest); +} + +</script> +</pre> + +</body> +</html> diff --git a/dom/media/test/test_bug879717.html b/dom/media/test/test_bug879717.html new file mode 100644 index 0000000000..c669773d3f --- /dev/null +++ b/dom/media/test/test_bug879717.html @@ -0,0 +1,130 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for bug 879717, check that a video element can be drawn into a canvas at various states of playback</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +var canvas = document.createElement('canvas'); +document.body.appendChild(canvas); + +var checkDrawImage = function(eventName, videoElement) { + var exception = null; + var exceptionName = "nothing"; + try { + var ctx = canvas.getContext('2d'); + ctx.drawImage(videoElement, 0, 0, canvas.width, canvas.height); + } catch (e) { + exception = e; + exceptionName = e.name; + } + ok(exception === null, + "drawImage shouldn't throw an exception on " + eventName + + " of " + videoElement.testName + ", got " + exceptionName); +}; + +var checkDrawImageEventHandler = function(ev) { + checkDrawImage(ev.type, ev.target); +}; +var startTest = function(media, token) { + manager.started(token); + + // File playback + var v1 = document.createElement("video"); + v1.autoplay = true; + + // Captured file playback + var v2 = document.createElement("video"); + + // Stream playback + var v3 = document.createElement("video"); + v3.autoplay = true; + + v1.gotLoadeddata = false; + v2.gotLoadeddata = false; + v3.gotLoadeddata = false; + + v1.testName = "v1 (" + media.name + ")"; + v2.testName = "v2 (Captured " + media.name + ")"; + v3.testName = "v3 (Stream of " + media.name + ")"; + + checkDrawImage("beforeplay", v1); + checkDrawImage("beforeplay", v2); + checkDrawImage("beforeplay", v3); + + v1.onloadedmetadata = checkDrawImageEventHandler; + v2.onloadedmetadata = checkDrawImageEventHandler; + v3.onloadedmetadata = checkDrawImageEventHandler; + + v1.onplay = checkDrawImageEventHandler; + v2.onplay = checkDrawImageEventHandler; + v3.onplay = checkDrawImageEventHandler; + + function onplaying(ev) { + if (!ev.target.gotPlaying) { + ev.target.gotPlaying = true; + checkDrawImageEventHandler(ev); + } + } + v1.onplaying = onplaying; + v2.onplaying = onplaying; + v3.onplaying = onplaying; + + var onloadeddata = function(ev) { + ev.target.gotLoadeddata = true; + checkDrawImageEventHandler(ev); + }; + + v1.onloadeddata = onloadeddata; + v2.onloadeddata = onloadeddata; + v3.onloadeddata = onloadeddata; + + var checkFinished = function() { + if (!v1.testFinished || !v2.testFinished || !v3.testFinished) { + return; + } + + ok(v1.gotLoadeddata, v1.testName + " should have gotten the 'loadeddata' event callback"); + ok(v2.gotLoadeddata, v2.testName + " should have gotten the 'loadeddata' event callback"); + ok(v3.gotLoadeddata, v3.testName + " should have gotten the 'loadeddata' event callback"); + + manager.finished(token); + }; + + var onended = function(ev) { + checkDrawImageEventHandler(ev); + removeNodeAndSource(ev.target); + ev.target.testFinished = true; + checkFinished(); + }; + + v1.onended = onended; + v2.onended = onended; + v3.onended = onended; + + document.body.appendChild(v1); + document.body.appendChild(v2); + document.body.appendChild(v3); + + v1.src = media.name; + v2.src = media.name; + v2.preload = 'metadata'; + + v2.addEventListener('loadedmetadata', function () { + v3.srcObject = v2.mozCaptureStreamUntilEnded(); + v2.play(); + }); +} + +manager.runTests(getPlayableVideos(gSmallTests), startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug895305.html b/dom/media/test/test_bug895305.html new file mode 100644 index 0000000000..56ed3c775c --- /dev/null +++ b/dom/media/test/test_bug895305.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=895305 +https://bugzilla.mozilla.org/show_bug.cgi?id=905320 +--> +<head> + <meta charset='utf-8'> + <title>Regression test for bug 895305 and 905320 - TextTrack* leaks</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<p id="display"></p> +<div id="content" style="display: none"> +</div> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +var audio = document.createElement("audio"); + +// Check leaking on TextTrackList objects. +/* global ttl, ttc */ +window.ttl = audio.textTracks; +ttl.addEventListener("click", function(){}); + +// Check leaking on VTTCue objects. +window.ttc = new VTTCue(3, 4, "Test."); +ttc.addEventListener("click", function() {}); + +// Check leaking on TextTrack objects. +audio.addTextTrack("subtitles", "label", "en-CA"); +ttl[0].addEventListener("click", function() {}); + +ok(true); // Need to have at least one assertion for Mochitest to be happy. +SimpleTest.finish(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_bug919265.html b/dom/media/test/test_bug919265.html new file mode 100644 index 0000000000..40eff135e4 --- /dev/null +++ b/dom/media/test/test_bug919265.html @@ -0,0 +1,30 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=919265 +--> +<head> + <meta charset='utf-8'> + <title>Regression test for bug 919265 - Leak on VTTCue::GetCueAsHTML()</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<p id="display"></p> +<div id="content" style="display: none"> +</div> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +// We shouldn't leak upon shutdown. +(new VTTCue(0, 0, "")).getCueAsHTML(); + +// We need to assert something for Mochitest to be happy. +ok(true); +SimpleTest.finish(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type.html b/dom/media/test/test_can_play_type.html new file mode 100644 index 0000000000..a28d137783 --- /dev/null +++ b/dom/media/test/test_can_play_type.html @@ -0,0 +1,40 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=469247 +--> +<head> + <title>Test for Bug 469247</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill +a Bug 469247</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script type="application/javascript"> + +var v = document.getElementById('v'); + +function check(type, expected) { + is(v.canPlayType(type), expected, type); +} + +// Invalid types +check("foo/bar", ""); +check("", ""); +check("!!!", ""); + +mediaTestCleanup(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type_mpeg.html b/dom/media/test/test_can_play_type_mpeg.html new file mode 100644 index 0000000000..a4b87272c0 --- /dev/null +++ b/dom/media/test/test_can_play_type_mpeg.html @@ -0,0 +1,168 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=799315 +--> +<head> + <title>Test for MP4 and MP3 support</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script> + +function check_mp4(v, enabled) { + function check(type, expected) { + var ex = enabled ? expected : ""; + is(v.canPlayType(type), ex, type + "='" + ex + "'"); + } + + check("video/mp4", "maybe"); + check("video/x-m4v", "maybe"); + check("audio/mp4", "maybe"); + check("audio/x-m4a", "maybe"); + + // Not the MIME type that other browsers respond to, so we won't either. + check("audio/m4a", ""); + check("video/m4v", ""); + + check("audio/aac", "maybe"); + check("audio/aacp", "maybe"); + + // H.264 Constrained Baseline Profile Level 3.0, AAC-LC + check("video/mp4; codecs=\"avc1.42E01E, mp4a.40.2\"", "probably"); + + // H.264 Constrained Baseline Profile Level 3.0, mp3 + check("video/mp4; codecs=\"avc1.42E01E, mp3\"", "probably"); + + check("video/mp4; codecs=\"avc1.42001E, mp4a.40.2\"", "probably"); + check("video/mp4; codecs=\"avc1.58A01E, mp4a.40.2\"", "probably"); + + // H.264 Main Profile Level 3.0, AAC-LC + check("video/mp4; codecs=\"avc1.4D401E, mp4a.40.2\"", "probably"); + // H.264 Main Profile Level 3.1, AAC-LC + check("video/mp4; codecs=\"avc1.4D401F, mp4a.40.2\"", "probably"); + // H.264 Main Profile Level 4.0, AAC-LC + check("video/mp4; codecs=\"avc1.4D4028, mp4a.40.2\"", "probably"); + // H.264 High Profile Level 3.0, AAC-LC + check("video/mp4; codecs=\"avc1.64001E, mp4a.40.2\"", "probably"); + // H.264 High Profile Level 3.1, AAC-LC + check("video/mp4; codecs=\"avc1.64001F, mp4a.40.2\"", "probably"); + + check("video/mp4; codecs=\"avc1.42E01E\"", "probably"); + check("video/mp4; codecs=\"avc1.42001E\"", "probably"); + check("video/mp4; codecs=\"avc1.58A01E\"", "probably"); + check("video/mp4; codecs=\"avc1.4D401E\"", "probably"); + check("video/mp4; codecs=\"avc1.64001F\"", "probably"); + + // AAC-LC + check("audio/mp4; codecs=\"mp4a.40.2\"", "probably"); + check("audio/mp4; codecs=mp4a.40.2", "probably"); + check("audio/x-m4a; codecs=\"mp4a.40.2\"", "probably"); + check("audio/x-m4a; codecs=mp4a.40.2", "probably"); + + check("audio/mp4; codecs=\"mp4a.40.2,\"", ""); // Invalid codecs string + + // HE-AAC v1 + check("audio/mp4; codecs=\"mp4a.40.5\"", "probably"); + check("audio/mp4; codecs=mp4a.40.5", "probably"); + check("audio/x-m4a; codecs=\"mp4a.40.5\"", "probably"); + check("audio/x-m4a; codecs=mp4a.40.5", "probably"); + // HE-AAC v2 + check("audio/mp4; codecs=\"mp4a.40.29\"", "probably"); + + // Opus + check("audio/mp4; codecs=\"opus\"", "probably"); + check("audio/mp4; codecs=opus", "probably"); + + // Flac. + var haveFlac = getPref("media.flac.enabled"); + check("audio/mp4; codecs=\"flac\"", haveFlac ? "probably" : ""); + check("audio/mp4; codecs=flac", haveFlac ? "probably" : ""); + + // VP9. + [ "video/mp4; codecs=vp9", + "video/mp4; codecs=\"vp9\"", + "video/mp4; codecs=\"vp9.0\"" + ].forEach((codec) => { + // canPlayType should support VP9 in MP4... + check(codec, "probably"); + if (!IsSupportedAndroid()) { + // VP9 codec is disabled on Android devices with no HW decoder. So skip it + // on Android for now. + ok(MediaSource.isTypeSupported(codec), "VP9 in MP4 should be supported in MSE"); + } + }); + + var haveAV1 = getPref("media.av1.enabled"); + check("video/mp4; codecs=\"av1\"", haveAV1 ? "probably" : ""); +} + +function check_mp3(v, enabled) { + function check(type, expected) { + var ex = enabled ? expected : ""; + is(v.canPlayType(type), ex, type + "='" + ex + "'"); + } + + check("audio/mpeg", "maybe"); + check("audio/mp3", "maybe"); + + check("audio/mpeg; codecs=\"mp3\"", "probably"); + check("audio/mpeg; codecs=mp3", "probably"); + + check("audio/mp3; codecs=\"mp3\"", "probably"); + check("audio/mp3; codecs=mp3", "probably"); +} + +function IsMacOS() { + return navigator.userAgent.includes("Mac OS"); +} + +function IsLinux() { + return navigator.userAgent.includes("Linux"); +} + +function getPref(name) { + return SpecialPowers.getBoolPref(name, false); +} + +function IsSupportedAndroid() { + return getAndroidVersion() >= 14; +} + +function IsJellyBeanOrLater() { + return getAndroidVersion() >= 16; +} + +var haveMp4 = + getPref("media.wmf.enabled") || + IsMacOS() || + (IsSupportedAndroid() && + (IsJellyBeanOrLater() || getPref("media.plugins.enabled"))) || + (IsLinux() && getPref("media.ffmpeg.enabled")); + +check_mp4(document.getElementById('v'), haveMp4); + +var haveMp3 = + getPref("media.wmf.enabled") || + (IsLinux() && getPref("media.ffmpeg.enabled")) || + (IsSupportedAndroid() && + ((IsJellyBeanOrLater() && getPref("media.android-media-codec.enabled")) || + getPref("media.plugins.enabled"))) || + IsMacOS(); + +check_mp3(document.getElementById('v'), haveMp3); + +mediaTestCleanup(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type_no_ogg.html b/dom/media/test/test_can_play_type_no_ogg.html new file mode 100644 index 0000000000..2e63f191d2 --- /dev/null +++ b/dom/media/test/test_can_play_type_no_ogg.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=469247 +--> +<head> + <title>Test for Bug 469247: Ogg backend disabled</title> + <script type="application/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill +a Bug 469247</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script src="can_play_type_ogg.js"></script> +<script> + +SimpleTest.waitForExplicitFinish(); + +function finish() { + mediaTestCleanup(); + SimpleTest.finish(); +} + +SpecialPowers.pushPrefEnv({"set": [["media.ogg.enabled", false]]}, + function() { + check_ogg(document.getElementById('v'), false, finish); + } +); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type_ogg.html b/dom/media/test/test_can_play_type_ogg.html new file mode 100644 index 0000000000..3e44f8ba8c --- /dev/null +++ b/dom/media/test/test_can_play_type_ogg.html @@ -0,0 +1,37 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=469247 +--> +<head> + <title>Test for Bug 469247: Ogg backend</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill +a Bug 469247</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script src="can_play_type_ogg.js"></script> +<script> + +SimpleTest.waitForExplicitFinish(); + +function finish() { + mediaTestCleanup(); + SimpleTest.finish(); +} + +check_ogg(document.getElementById('v'), true, finish); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type_wave.html b/dom/media/test/test_can_play_type_wave.html new file mode 100644 index 0000000000..e6d4e29d86 --- /dev/null +++ b/dom/media/test/test_can_play_type_wave.html @@ -0,0 +1,30 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=469247 +--> +<head> + <title>Test for Bug 469247: WAVE backend</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill +a Bug 469247</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script src="can_play_type_wave.js"></script> +<script> +check_wave(document.getElementById('v'), true); + +mediaTestCleanup(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_can_play_type_webm.html b/dom/media/test/test_can_play_type_webm.html new file mode 100644 index 0000000000..6b0e2adfad --- /dev/null +++ b/dom/media/test/test_can_play_type_webm.html @@ -0,0 +1,39 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=566245 +--> +<head> + <title>Test for Bug 566245: WebM backend</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=566245">Mozill +a Bug 566245</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> + +<video id="v"></video> + +<pre id="test"> +<script src="can_play_type_webm.js"></script> +<script> + async function runTest() { + try { + await check_webm(document.getElementById('v'), true); + mediaTestCleanup(); + } catch (e) { + info("Exception " + e.message); + ok(false, "Threw exception " + e.message); + } + SimpleTest.finish(); + } + SimpleTest.waitForExplicitFinish(); + addLoadEvent(runTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_chaining.html b/dom/media/test/test_chaining.html new file mode 100644 index 0000000000..6ebf94c88d --- /dev/null +++ b/dom/media/test/test_chaining.html @@ -0,0 +1,92 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: chained ogg files.</title> + <meta charset='utf-8'> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function finish_test(element) { + removeNodeAndSource(element); + manager.finished(element.token); +} + +function onended(e) { + var t = e.target; + is(t._metadataCount, t._links, "We should have received "+ t._links + + " metadataloaded event. " + t.src); + + // If we encounter a file that has links with a different numbers of channel, + // we stop the decoding at the end of the first link. Hence, we report normal + // |seekable| and |buffered| values. + if (t._links != 1) { + is(t.seekable.length, 0, "No seekable ranges should be reported." + t.src); + is(t.buffered.length, 0, "No buffered region should be reported." + t.src); + } + + is(t.played.length, 1, "The played region should be continuous." + t.src); + + if (t._links != 1) { + var oldCurrentTime = t.currentTime; + t.currentTime = 0.0; + is(t.currentTime, oldCurrentTime, + "Seeking should be disabled when playing chained media." + t.src); + } + + finish_test(t); +} + +function onmetadataloaded(e) { + var t = e.target; + if (! t._metadataCount) { + t._metadataCount = 0; + } + + if (t._metadataCount > 1 && t._links === 1) { + ok(false, "We should receive only one \"loadedmetadata\" event for a chained file we don't support.") + } + + // We should be able to assert equality here, but somehow it fails (rarely but + // still) on try. Instead, we give it a little slack and assert that the index + // increases monotonically. + ok(t.mozGetMetadata().index >= t._metadataCount || t._links === 1, + "The metadata index value should increase." + t.src); + + ok(t.currentTime >= t._prevCurrentTime, + "The currenttime should be increased correctly in new chained part."); + t._prevCurrentTime = t.currentTime; + + // The files have all a user comment name 'index' that increments at each link + // in the chained media. + t._metadataCount++; + if (!t.playing && !t.ended) { + t.play(); + } +} + +function startTest(test, token) { + var elemType = /^audio/.test(test.type) ? "audio" : "video"; + var element = document.createElement(elemType); + document.body.appendChild(element); + manager.started(token); + element._links= test.links; + element.src = test.name; + element.token = token; + element.controls = true; + element.addEventListener("loadedmetadata", onmetadataloaded); + element.addEventListener("ended", onended); + element.preload = "metadata"; + element._prevCurrentTime = 0; +} + +manager.runTests(gChainingTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_ended_video.html b/dom/media/test/test_cloneElementVisually_ended_video.html new file mode 100644 index 0000000000..331dda2db9 --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_ended_video.html @@ -0,0 +1,48 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Test that when we start cloning a video that has already ended, the + * clone displays the last frame from the video. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + let ended = waitForEventOnce(originalVideo, "ended"); + await originalVideo.play(); + await ended; + + await withNewClone(originalVideo, async clone => { + await SpecialPowers.wrap(originalVideo).cloneElementVisually(clone); + ok(await assertVideosMatch(originalVideo, clone), + "Visual clone should display final frame."); + }); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_mediastream.html b/dom/media/test/test_cloneElementVisually_mediastream.html new file mode 100644 index 0000000000..5bfcf7673f --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_mediastream.html @@ -0,0 +1,70 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>MediaStream</h1> + <video id="streamTarget"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Test that we can clone a video that is playing a MediaStream. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + let stream = originalVideo.mozCaptureStream(); + let streamTarget = document.getElementById("streamTarget"); + originalVideo.setAttribute("loop", true); + let playingPromise = waitForEventOnce(originalVideo, "playing"); + await originalVideo.play(); + await playingPromise; + + streamTarget.srcObject = stream; + playingPromise = waitForEventOnce(streamTarget, "playing"); + await streamTarget.play(); + await playingPromise + + await withNewClone(originalVideo, async clone => { + await SpecialPowers.wrap(streamTarget).cloneElementVisually(clone); + + originalVideo.loop = false; + originalVideo.currentTime = originalVideo.duration - 0.1; + await waitForEventOnce(streamTarget, "ended"); + + ok(await assertVideosMatch(originalVideo, streamTarget), + "Should match Original video"); + ok(await assertVideosMatch(streamTarget, clone), + "Should match MediaStream"); + }); + + // Capturing a stream from a video "taints" it which prevents testing + // shutdown decoder behaviour. To avoid interfering with future tests, + // we replace the video. + let newVideo = originalVideo.cloneNode(); + originalVideo.parentNode.replaceChild(newVideo, originalVideo); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html b/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html new file mode 100644 index 0000000000..04d1ed484c --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html @@ -0,0 +1,88 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Originals</h1> + <div id="originalContainer"></div> + <canvas id="canvas"></canvas> + <h1>MediaStream</h1> + <div id="streamTargetContainer"></div> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Test that a clone of a video that is playing a MediaStream properly tracks + * the selected video track. + */ +add_task(async () => { + await SpecialPowers.pushPrefEnv({ + set: [ + ["media.track.enabled", true], + ], + }); + + let originalVideo = document.createElement("video"); + originalVideo.id = "original"; + document.getElementById("originalContainer").appendChild(originalVideo); + + let streamTarget = document.createElement("video"); + document.getElementById("streamTargetContainer").appendChild(streamTarget); + + await setup(); + + let [track1] = originalVideo.mozCaptureStream().getTracks(); + let playingPromise = waitForEventOnce(originalVideo, "playing"); + await originalVideo.play(); + await playingPromise; + + let canvas = document.getElementById("canvas"); + canvas.width = originalVideo.videoWidth / 2; + canvas.height = originalVideo.videoHeight / 2; + let ctx = canvas.getContext("2d"); + let [track2] = canvas.captureStream().getTracks(); + ctx.fillStyle = "green"; + ctx.fillRect(0, 0, canvas.width, canvas.height); + + streamTarget.srcObject = new MediaStream([track1, track2]); + playingPromise = waitForEventOnce(streamTarget, "playing"); + await streamTarget.play(); + await playingPromise; + + await withNewClone(originalVideo, async clone => { + SpecialPowers.wrap(streamTarget).cloneElementVisually(clone); + + let selectedTrackIdx = streamTarget.videoTracks.selectedIndex; + streamTarget.videoTracks[++selectedTrackIdx % 2].selected = true; + await waitForEventOnce(streamTarget, "resize"); + + ok(await assertVideosMatch(streamTarget, clone), + "Should match MediaStream"); + }); + + // Capturing a stream from a video "taints" it which prevents testing + // shutdown decoder behaviour. To avoid interfering with future tests, + // we replace the video. + let newVideo = originalVideo.cloneNode(); + originalVideo.parentNode.replaceChild(newVideo, originalVideo); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_no_suspend.html b/dom/media/test/test_cloneElementVisually_no_suspend.html new file mode 100644 index 0000000000..d576bf8ab3 --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_no_suspend.html @@ -0,0 +1,90 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>MediaStream</h1> + <video id="streamTarget"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Tests that cloning a video prevents the decoder from being suspended + * if the original video stops being visible. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + await setVideoSrc(originalVideo, LONG_VIDEO); + + await originalVideo.play(); + + // Ensure that hiding and pausing this video will cause us to + // try suspending it. + await ensureVideoSuspendable(originalVideo); + + await withNewClone(originalVideo, async clone => { + SpecialPowers.wrap(originalVideo).cloneElementVisually(clone); + + // Go back to the beginning of the video to give us enough time to + // fail to suspend the video when it's being cloned before the + // video ends. + originalVideo.removeAttribute("loop"); + originalVideo.currentTime = 0; + await waitForEventOnce(originalVideo, "seeked"); + + let suspendTimerFired = false; + + let listener = () => { + suspendTimerFired = true; + } + originalVideo.addEventListener("mozstartvideosuspendtimer", listener); + + // Have to do this to access normally-preffed off binding methods for some + // reason. + // See bug 1544257. + SpecialPowers.wrap(originalVideo).setVisible(false); + + await waitForEventOnce(originalVideo, "ended"); + + originalVideo.removeEventListener("mozstartvideosuspendtimer", listener); + + ok(!suspendTimerFired, + "mozstartvideosuspendtimer should not have fired."); + + // Have to do this to access normally-preffed off binding methods for some + // reason. + // See bug 1544257. + SpecialPowers.wrap(originalVideo).setVisible(true); + }); + + await originalVideo.play(); + + // With the clone gone, the original video should be able to suspend now. + await ensureVideoSuspendable(originalVideo); + + await setVideoSrc(originalVideo, TEST_VIDEO_1); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_paused.html b/dom/media/test/test_cloneElementVisually_paused.html new file mode 100644 index 0000000000..1812becdd8 --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_paused.html @@ -0,0 +1,45 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Test that when we start cloning a paused video, the clone displays + * the first paused frame. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + await withNewClone(originalVideo, async clone => { + await SpecialPowers.wrap(originalVideo).cloneElementVisually(clone); + + ok(await assertVideosMatch(originalVideo, clone), + "Initial paused frame should match."); + }); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_poster.html b/dom/media/test/test_cloneElementVisually_poster.html new file mode 100644 index 0000000000..e7ba00edb0 --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_poster.html @@ -0,0 +1,52 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Test that when we start cloning a paused video, the clone displays + * the first paused frame. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + const POSTER_URL = "https://example.com:443/tests/dom/media/test/poster-test.jpg"; + originalVideo.setAttribute("poster", POSTER_URL); + + await withNewClone(originalVideo, async clone => { + SpecialPowers.wrap(originalVideo).cloneElementVisually(clone); + await originalVideo.play(); + await waitForEventOnce(originalVideo, "timeupdate"); + originalVideo.pause(); + await waitForEventOnce(originalVideo, "pause"); + + ok(await assertVideosMatch(originalVideo, clone), + "Video with a poster should clone properly."); + }); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_cloneElementVisually_resource_change.html b/dom/media/test/test_cloneElementVisually_resource_change.html new file mode 100644 index 0000000000..3a66906ea2 --- /dev/null +++ b/dom/media/test/test_cloneElementVisually_resource_change.html @@ -0,0 +1,67 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test cloneElementVisually</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"> +</head> +<body> +<div id="content"> + <h1>Original</h1> + <video id="original"></video> + <h1>MediaStream</h1> + <video id="streamTarget"></video> + <h1>Clone</h1> +</div> +<div id="results"> + <h1>Results</h1> + <canvas id="left"></canvas> + <canvas id="right"></canvas> +</div> + +<script type="application/javascript"> + +/* import-globals-from cloneElementVisually_helpers.js */ + +/** + * Tests that cloning survives changes to the underlying video resource. + */ +add_task(async () => { + await setup(); + + let originalVideo = document.getElementById("original"); + originalVideo.setAttribute("loop", true); + await originalVideo.play(); + + await withNewClone(originalVideo, async clone => { + SpecialPowers.wrap(originalVideo).cloneElementVisually(clone); + + await waitForEventOnce(originalVideo, "timeupdate"); + + originalVideo.pause(); + await waitForEventOnce(originalVideo, "pause"); + + ok(await assertVideosMatch(originalVideo, clone), + "Initial video should match."); + + await setVideoSrc(originalVideo, TEST_VIDEO_2); + + await originalVideo.play(); + await waitForEventOnce(originalVideo, "timeupdate"); + + originalVideo.pause(); + await waitForEventOnce(originalVideo, "pause"); + + ok(await assertVideosMatch(originalVideo, clone), + "New video should match."); + }); + + await setVideoSrc(originalVideo, TEST_VIDEO_1); +}); + +</script> + +</body> +</html> diff --git a/dom/media/test/test_clone_media_element.html b/dom/media/test/test_clone_media_element.html new file mode 100644 index 0000000000..35e5cd69d0 --- /dev/null +++ b/dom/media/test/test_clone_media_element.html @@ -0,0 +1,54 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test: cloned media element should continue to play to the end even after the source of the original element is cleared</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// tests must run in sequence otherwise concurrent running test will also +// update media cache which will hide the fact media cache not updated +// after changes in media cache streams. +PARALLEL_TESTS = 1; + +function startTest(test, token) { + manager.started(token); + info("Trying to load " + token); + var v = document.createElement('video'); + v.preload = "metadata"; + v.token = token; + v.src = test.name; + + v.onloadedmetadata = function(evt) { + info(evt.target.token + " metadata loaded."); + evt.target.onloadedmetadata = null; + var clone = evt.target.cloneNode(false); + clone.token = evt.target.token; + clone.play(); + + clone.onloadstart = function(event) { + info("cloned " + event.target.token + " start loading."); + event.target.onloadstart = null; + removeNodeAndSource(v); + } + + clone.onended = function(event) { + ok(true, "cloned " + event.target.token + " ended."); + event.target.onended = null; + removeNodeAndSource(event.target); + manager.finished(event.target.token); + } + } +} + +var manager = new MediaTestManager; +manager.runTests(gSmallTests.concat(gPlayedTests), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_closing_connections.html b/dom/media/test/test_closing_connections.html new file mode 100644 index 0000000000..c5eb565447 --- /dev/null +++ b/dom/media/test/test_closing_connections.html @@ -0,0 +1,58 @@ +hg diff<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=479863 +--> +<head> + <title>Test for Bug 479863 --- loading many connections</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="application/javascript" src="use_large_cache.js"></script> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479863">Mozilla Bug 479863</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> + +<script type="application/javascript"> +window.onload = function() { + ok(true, "loaded metadata for all videos"); + mediaTestCleanup(); + SimpleTest.finish(); +} + +/* With normal per-domain connection limits and a naive implementation, we + won't ever be able to load all these videos because the first 15 (or whatever) + will each take up one HTTP connection (which will be suspended) and then + the others will be blocked by the per-domain HTTP connection limit. We + pass this test by closing the connection for non-buffered videos after + we've got the first frame. +*/ + +var resource = getPlayableVideo(gClosingConnectionsTest); + +SimpleTest.waitForExplicitFinish(); +function beginTest() { + if (resource != null) { + for (var i=0; i<20; ++i) { + var v = document.createElement("video"); + v.src = resource.name; + v.preload = "metadata"; + document.body.appendChild(v); + } + } else { + todo(false, "No types supported"); + } +} +beginTest(); +</script> + +<pre id="test"> +<script type="application/javascript"> +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_constants.html b/dom/media/test/test_constants.html new file mode 100644 index 0000000000..1d4a8da250 --- /dev/null +++ b/dom/media/test/test_constants.html @@ -0,0 +1,228 @@ +<!DOCTYPE HTML> +<html> +<!-- + Adapted from: + http://simon.html5.org/test/html/dom/interfaces/htmlelement/htmlmediaelement/const-unsigned-short/001.htm +--> +<head> + <title>Media test: constants</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<video><source></video><audio><source></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +is(HTMLElement.NETWORK_EMPTY, undefined); +is(HTMLElement.NETWORK_IDLE, undefined); +is(HTMLElement.NETWORK_LOADING, undefined); +is(HTMLElement.NETWORK_NO_SOURCE, undefined); +is(HTMLElement.HAVE_NOTHING, undefined); +is(HTMLElement.HAVE_METADATA, undefined); +is(HTMLElement.HAVE_CURRENT_DATA, undefined); +is(HTMLElement.HAVE_FUTURE_DATA, undefined); +is(HTMLElement.HAVE_ENOUGH_DATA, undefined); +is(HTMLElement.MEDIA_ERR_ABORTED, undefined); +is(HTMLElement.MEDIA_ERR_NETWORK, undefined); +is(HTMLElement.MEDIA_ERR_DECODE, undefined); +is(HTMLElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLMediaElement.NETWORK_EMPTY, 0); +is(HTMLMediaElement.NETWORK_IDLE, 1); +is(HTMLMediaElement.NETWORK_LOADING, 2); +is(HTMLMediaElement.NETWORK_NO_SOURCE, 3); +is(HTMLMediaElement.HAVE_NOTHING, 0); +is(HTMLMediaElement.HAVE_METADATA, 1); +is(HTMLMediaElement.HAVE_CURRENT_DATA, 2); +is(HTMLMediaElement.HAVE_FUTURE_DATA, 3); +is(HTMLMediaElement.HAVE_ENOUGH_DATA, 4); +is(HTMLMediaElement.MEDIA_ERR_ABORTED, undefined); +is(HTMLMediaElement.MEDIA_ERR_NETWORK, undefined); +is(HTMLMediaElement.MEDIA_ERR_DECODE, undefined); +is(HTMLMediaElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLVideoElement.NETWORK_EMPTY, 0); +is(HTMLVideoElement.NETWORK_IDLE, 1); +is(HTMLVideoElement.NETWORK_LOADING, 2); +is(HTMLVideoElement.NETWORK_NO_SOURCE, 3); +is(HTMLVideoElement.HAVE_NOTHING, 0); +is(HTMLVideoElement.HAVE_METADATA, 1); +is(HTMLVideoElement.HAVE_CURRENT_DATA, 2); +is(HTMLVideoElement.HAVE_FUTURE_DATA, 3); +is(HTMLVideoElement.HAVE_ENOUGH_DATA, 4); +is(HTMLVideoElement.MEDIA_ERR_ABORTED, undefined); +is(HTMLVideoElement.MEDIA_ERR_NETWORK, undefined); +is(HTMLVideoElement.MEDIA_ERR_DECODE, undefined); +is(HTMLVideoElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLAudioElement.NETWORK_EMPTY, 0); +is(HTMLAudioElement.NETWORK_IDLE, 1); +is(HTMLAudioElement.NETWORK_LOADING, 2); +is(HTMLAudioElement.NETWORK_NO_SOURCE, 3); +is(HTMLAudioElement.HAVE_NOTHING, 0); +is(HTMLAudioElement.HAVE_METADATA, 1); +is(HTMLAudioElement.HAVE_CURRENT_DATA, 2); +is(HTMLAudioElement.HAVE_FUTURE_DATA, 3); +is(HTMLAudioElement.HAVE_ENOUGH_DATA, 4); +is(HTMLAudioElement.MEDIA_ERR_ABORTED, undefined); +is(HTMLAudioElement.MEDIA_ERR_NETWORK, undefined); +is(HTMLAudioElement.MEDIA_ERR_DECODE, undefined); +is(HTMLAudioElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLSourceElement.NETWORK_EMPTY, undefined); +is(HTMLSourceElement.NETWORK_IDLE, undefined); +is(HTMLSourceElement.NETWORK_LOADING, undefined); +is(HTMLSourceElement.NETWORK_NO_SOURCE, undefined); +is(HTMLSourceElement.HAVE_NOTHING, undefined); +is(HTMLSourceElement.HAVE_METADATA, undefined); +is(HTMLSourceElement.HAVE_CURRENT_DATA, undefined); +is(HTMLSourceElement.HAVE_FUTURE_DATA, undefined); +is(HTMLSourceElement.HAVE_ENOUGH_DATA, undefined); +is(HTMLSourceElement.MEDIA_ERR_ABORTED, undefined); +is(HTMLSourceElement.MEDIA_ERR_NETWORK, undefined); +is(HTMLSourceElement.MEDIA_ERR_DECODE, undefined); +is(HTMLSourceElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(MediaError.NETWORK_EMPTY, undefined); +is(MediaError.NETWORK_IDLE, undefined); +is(MediaError.NETWORK_LOADING, undefined); +is(MediaError.NETWORK_NO_SOURCE, undefined); +is(MediaError.HAVE_NOTHING, undefined); +is(MediaError.HAVE_METADATA, undefined); +is(MediaError.HAVE_CURRENT_DATA, undefined); +is(MediaError.HAVE_FUTURE_DATA, undefined); +is(MediaError.HAVE_ENOUGH_DATA, undefined); +is(MediaError.MEDIA_ERR_ABORTED, 1); +is(MediaError.MEDIA_ERR_NETWORK, 2); +is(MediaError.MEDIA_ERR_DECODE, 3); +is(MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, 4); +is(document.body.NETWORK_EMPTY, undefined); +is(document.body.NETWORK_IDLE, undefined); +is(document.body.NETWORK_LOADING, undefined); +is(document.body.NETWORK_NO_SOURCE, undefined); +is(document.body.HAVE_NOTHING, undefined); +is(document.body.HAVE_METADATA, undefined); +is(document.body.HAVE_CURRENT_DATA, undefined); +is(document.body.HAVE_FUTURE_DATA, undefined); +is(document.body.HAVE_ENOUGH_DATA, undefined); +is(document.body.MEDIA_ERR_ABORTED, undefined); +is(document.body.MEDIA_ERR_NETWORK, undefined); +is(document.body.MEDIA_ERR_DECODE, undefined); +is(document.body.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(document.getElementsByTagName("video")[0].NETWORK_EMPTY, 0); +is(document.getElementsByTagName("video")[0].NETWORK_IDLE, 1); +is(document.getElementsByTagName("video")[0].NETWORK_LOADING, 2); +is(document.getElementsByTagName("video")[0].NETWORK_NO_SOURCE, 3); +is(document.getElementsByTagName("video")[0].HAVE_NOTHING, 0); +is(document.getElementsByTagName("video")[0].HAVE_METADATA, 1); +is(document.getElementsByTagName("video")[0].HAVE_CURRENT_DATA, 2); +is(document.getElementsByTagName("video")[0].HAVE_FUTURE_DATA, 3); +is(document.getElementsByTagName("video")[0].HAVE_ENOUGH_DATA, 4); +is(document.getElementsByTagName("video")[0].MEDIA_ERR_ABORTED, undefined); +is(document.getElementsByTagName("video")[0].MEDIA_ERR_NETWORK, undefined); +is(document.getElementsByTagName("video")[0].MEDIA_ERR_DECODE, undefined); +is(document.getElementsByTagName("video")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(document.getElementsByTagName("audio")[0].NETWORK_EMPTY, 0); +is(document.getElementsByTagName("audio")[0].NETWORK_IDLE, 1); +is(document.getElementsByTagName("audio")[0].NETWORK_LOADING, 2); +is(document.getElementsByTagName("audio")[0].NETWORK_NO_SOURCE, 3); +is(document.getElementsByTagName("audio")[0].HAVE_NOTHING, 0); +is(document.getElementsByTagName("audio")[0].HAVE_METADATA, 1); +is(document.getElementsByTagName("audio")[0].HAVE_CURRENT_DATA, 2); +is(document.getElementsByTagName("audio")[0].HAVE_FUTURE_DATA, 3); +is(document.getElementsByTagName("audio")[0].HAVE_ENOUGH_DATA, 4); +is(document.getElementsByTagName("audio")[0].MEDIA_ERR_ABORTED, undefined); +is(document.getElementsByTagName("audio")[0].MEDIA_ERR_NETWORK, undefined); +is(document.getElementsByTagName("audio")[0].MEDIA_ERR_DECODE, undefined); +is(document.getElementsByTagName("audio")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(document.getElementsByTagName("source")[0].NETWORK_EMPTY, undefined); +is(document.getElementsByTagName("source")[0].NETWORK_IDLE, undefined); +is(document.getElementsByTagName("source")[0].NETWORK_LOADING, undefined); +is(document.getElementsByTagName("source")[0].NETWORK_NO_SOURCE, undefined); +is(document.getElementsByTagName("source")[0].HAVE_NOTHING, undefined); +is(document.getElementsByTagName("source")[0].HAVE_METADATA, undefined); +is(document.getElementsByTagName("source")[0].HAVE_CURRENT_DATA, undefined); +is(document.getElementsByTagName("source")[0].HAVE_FUTURE_DATA, undefined); +is(document.getElementsByTagName("source")[0].HAVE_ENOUGH_DATA, undefined); +is(document.getElementsByTagName("source")[0].MEDIA_ERR_ABORTED, undefined); +is(document.getElementsByTagName("source")[0].MEDIA_ERR_NETWORK, undefined); +is(document.getElementsByTagName("source")[0].MEDIA_ERR_DECODE, undefined); +is(document.getElementsByTagName("source")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLElement.prototype.NETWORK_EMPTY, undefined); +is(HTMLElement.prototype.NETWORK_IDLE, undefined); +is(HTMLElement.prototype.NETWORK_LOADING, undefined); +is(HTMLElement.prototype.NETWORK_NO_SOURCE, undefined); +is(HTMLElement.prototype.HAVE_NOTHING, undefined); +is(HTMLElement.prototype.HAVE_METADATA, undefined); +is(HTMLElement.prototype.HAVE_CURRENT_DATA, undefined); +is(HTMLElement.prototype.HAVE_FUTURE_DATA, undefined); +is(HTMLElement.prototype.HAVE_ENOUGH_DATA, undefined); +is(HTMLElement.prototype.MEDIA_ERR_ABORTED, undefined); +is(HTMLElement.prototype.MEDIA_ERR_NETWORK, undefined); +is(HTMLElement.prototype.MEDIA_ERR_DECODE, undefined); +is(HTMLElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLMediaElement.prototype.NETWORK_EMPTY, 0, "HTMLMediaElement.prototype.NETWORK_EMPTY"); +is(HTMLMediaElement.prototype.NETWORK_IDLE, 1, "HTMLMediaElement.prototype.NETWORK_IDLE"); +is(HTMLMediaElement.prototype.NETWORK_LOADING, 2, "HTMLMediaElement.prototype.NETWORK_LOADING"); +is(HTMLMediaElement.prototype.NETWORK_NO_SOURCE, 3, "HTMLMediaElement.prototype.NETWORK_NO_SOURCE"); +is(HTMLMediaElement.prototype.HAVE_NOTHING, 0, "HTMLMediaElement.prototype.HAVE_NOTHING"); +is(HTMLMediaElement.prototype.HAVE_METADATA, 1, "HTMLMediaElement.prototype.HAVE_METADATA"); +is(HTMLMediaElement.prototype.HAVE_CURRENT_DATA, 2, "HTMLMediaElement.prototype.HAVE_CURRENT_DATA"); +is(HTMLMediaElement.prototype.HAVE_FUTURE_DATA, 3, "HTMLMediaElement.prototype.HAVE_FUTURE_DATA"); +is(HTMLMediaElement.prototype.HAVE_ENOUGH_DATA, 4, "HTMLMediaElement.prototype.HAVE_ENOUGH_DATA"); +is(HTMLMediaElement.prototype.MEDIA_ERR_ABORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_ABORTED"); +is(HTMLMediaElement.prototype.MEDIA_ERR_NETWORK, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_NETWORK"); +is(HTMLMediaElement.prototype.MEDIA_ERR_DECODE, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_DECODE"); +is(HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED"); +is(HTMLVideoElement.prototype.NETWORK_EMPTY, 0); +is(HTMLVideoElement.prototype.NETWORK_IDLE, 1); +is(HTMLVideoElement.prototype.NETWORK_LOADING, 2); +is(HTMLVideoElement.prototype.NETWORK_NO_SOURCE, 3); +is(HTMLVideoElement.prototype.HAVE_NOTHING, 0); +is(HTMLVideoElement.prototype.HAVE_METADATA, 1); +is(HTMLVideoElement.prototype.HAVE_CURRENT_DATA, 2); +is(HTMLVideoElement.prototype.HAVE_FUTURE_DATA, 3); +is(HTMLVideoElement.prototype.HAVE_ENOUGH_DATA, 4); +is(HTMLVideoElement.prototype.MEDIA_ERR_ABORTED, undefined); +is(HTMLVideoElement.prototype.MEDIA_ERR_NETWORK, undefined); +is(HTMLVideoElement.prototype.MEDIA_ERR_DECODE, undefined); +is(HTMLVideoElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLAudioElement.prototype.NETWORK_EMPTY, 0); +is(HTMLAudioElement.prototype.NETWORK_IDLE, 1); +is(HTMLAudioElement.prototype.NETWORK_LOADING, 2); +is(HTMLAudioElement.prototype.NETWORK_NO_SOURCE, 3); +is(HTMLAudioElement.prototype.HAVE_NOTHING, 0); +is(HTMLAudioElement.prototype.HAVE_METADATA, 1); +is(HTMLAudioElement.prototype.HAVE_CURRENT_DATA, 2); +is(HTMLAudioElement.prototype.HAVE_FUTURE_DATA, 3); +is(HTMLAudioElement.prototype.HAVE_ENOUGH_DATA, 4); +is(HTMLAudioElement.prototype.MEDIA_ERR_ABORTED, undefined); +is(HTMLAudioElement.prototype.MEDIA_ERR_NETWORK, undefined); +is(HTMLAudioElement.prototype.MEDIA_ERR_DECODE, undefined); +is(HTMLAudioElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(HTMLSourceElement.prototype.NETWORK_EMPTY, undefined); +is(HTMLSourceElement.prototype.NETWORK_IDLE, undefined); +is(HTMLSourceElement.prototype.NETWORK_LOADING, undefined); +is(HTMLSourceElement.prototype.NETWORK_NO_SOURCE, undefined); +is(HTMLSourceElement.prototype.HAVE_NOTHING, undefined); +is(HTMLSourceElement.prototype.HAVE_METADATA, undefined); +is(HTMLSourceElement.prototype.HAVE_CURRENT_DATA, undefined); +is(HTMLSourceElement.prototype.HAVE_FUTURE_DATA, undefined); +is(HTMLSourceElement.prototype.HAVE_ENOUGH_DATA, undefined); +is(HTMLSourceElement.prototype.MEDIA_ERR_ABORTED, undefined); +is(HTMLSourceElement.prototype.MEDIA_ERR_NETWORK, undefined); +is(HTMLSourceElement.prototype.MEDIA_ERR_DECODE, undefined); +is(HTMLSourceElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined); +is(MediaError.prototype.NETWORK_EMPTY, undefined); +is(MediaError.prototype.NETWORK_IDLE, undefined); +is(MediaError.prototype.NETWORK_LOADING, undefined); +is(MediaError.prototype.NETWORK_NO_SOURCE, undefined); +is(MediaError.prototype.HAVE_NOTHING, undefined); +is(MediaError.prototype.HAVE_METADATA, undefined); +is(MediaError.prototype.HAVE_CURRENT_DATA, undefined); +is(MediaError.prototype.HAVE_FUTURE_DATA, undefined); +is(MediaError.prototype.HAVE_ENOUGH_DATA, undefined); +is(MediaError.prototype.MEDIA_ERR_ABORTED, 1); +is(MediaError.prototype.MEDIA_ERR_NETWORK, 2); +is(MediaError.prototype.MEDIA_ERR_DECODE, 3); +is(MediaError.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, 4); +ok(document.getElementsByTagName("video")[0].buffered instanceof TimeRanges, "video.buffered must be TimeRanges object"); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_controls.html b/dom/media/test/test_controls.html new file mode 100644 index 0000000000..5c9015fdf2 --- /dev/null +++ b/dom/media/test/test_controls.html @@ -0,0 +1,33 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: controls</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<video id='v1'></video><audio id='a1'></audio> +<video id='v2' controls></video><audio id='a2' controls></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var v2 = document.getElementById('v2'); +var a2 = document.getElementById('a2'); +ok(!v1.controls, "v1.controls should be false by default"); +ok(!a1.controls, "v1.controls should be false by default"); +ok(v2.controls, "v2.controls should be true"); +ok(a2.controls, "v2.controls should be true"); +v2.controls=false; +a2.controls=false; +ok(!v2.controls, "v2.controls should be false"); +ok(!a2.controls, "a2.controls should be false"); +v2.controls=true; +a2.controls=true; +ok(v2.controls, "v2.controls should be true"); +ok(a2.controls, "a2.controls should be true"); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_cueless_webm_seek-1.html b/dom/media/test/test_cueless_webm_seek-1.html new file mode 100644 index 0000000000..db58a89665 --- /dev/null +++ b/dom/media/test/test_cueless_webm_seek-1.html @@ -0,0 +1,136 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=657791 +--> +<head> + <title>Test for Bug 657791</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Subset of seek tests for cueless WebMs. When random seeking (rather than just +// in buffered ranges) is implemented for WebM, these tests can be removed and +// the cueless WebM(s) references can be moved to the general test_seek test +// array. +// Test array is defined in manifest.js + +var manager = new MediaTestManager; + +// Exercise functionality as in test_seek-1 +function testWebM1(e) { + var v = e.target; + v.removeEventListener('loadeddata', testWebM1); + + var startPassed = false; + var endPassed = false; + var seekFlagStart = false; + var seekFlagEnd = false; + var readonly = true; + var completed = false; + + ok(v.buffered.length >= 1, "Should have a buffered range"); + var halfBuffered = v.buffered.end(0) / 2; + + function start() { + is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start"); + is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end"); + ok(!completed, "Should not be completed yet"); + ok(!v.seeking, "seeking should default to false"); + try { + v.seeking = true; + readonly = v.seeking === false; + } + catch(ex) { + readonly = "threw exception: " + ex; + } + is(readonly, true, "seeking should be readonly"); + + v.currentTime = halfBuffered; + seekFlagStart = v.seeking; + } + + function seekStarted() { + ok(!completed, "should not be completed yet"); + startPassed = true; + } + + function seekEnded() { + ok(!completed, "should not be completed yet"); + ok(Math.abs(v.currentTime - halfBuffered) < 0.1, + "Video currentTime should be around " + halfBuffered + ": " + v.currentTime + " (seeked)"); + endPassed = true; + seekFlagEnd = v.seeking; + v.play(); + } + + function playbackEnded() { + ok(!completed, "should not be completed yet"); + + completed = true; + ok(startPassed, "seeking event"); + ok(endPassed, "seeked event"); + ok(seekFlagStart, "seeking flag on start should be true"); + ok(!seekFlagEnd, "seeking flag on end should be false"); + removeNodeAndSource(v); + manager.finished(v._token); + } + + once(v, "ended", playbackEnded); + once(v, "seeking", seekStarted); + once(v, "seeked", seekEnded); + + start(); +} + +// Fetch the media resource using XHR so we can be sure the entire +// resource is loaded before we test buffered ranges. This ensures +// we have deterministic behaviour. +function fetch(url, fetched_callback) { + var xhr = new XMLHttpRequest(); + xhr.open("GET", url, true); + xhr.responseType = "blob"; + + var loaded = function (event) { + if (xhr.status == 200 || xhr.status == 206) { + // Request fulfilled. Note sometimes we get 206... Presumably because either + // httpd.js or Necko cached the result. + fetched_callback(window.URL.createObjectURL(xhr.response)); + } else { + ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders()); + } + }; + + xhr.addEventListener("load", loaded); + xhr.send(); +} + +function startTest(test, token) { + var onfetched = function(uri) { + var v = document.createElement('video'); + v._token = token; + v.src = uri; + v.addEventListener("loadeddata", testWebM1); + document.body.appendChild(v); + } + manager.started(token); + fetch(test.name, onfetched); +} + +SimpleTest.waitForExplicitFinish(); +manager.runTests(gCuelessWebMTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_cueless_webm_seek-2.html b/dom/media/test/test_cueless_webm_seek-2.html new file mode 100644 index 0000000000..720cc18399 --- /dev/null +++ b/dom/media/test/test_cueless_webm_seek-2.html @@ -0,0 +1,126 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=657791 +--> +<head> + <title>Test for Bug 657791</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Subset of seek tests for cueless WebMs. When random seeking (rather than just +// in buffered ranges) is implemented for WebM, these tests can be removed and +// the cueless WebM(s) references can be moved to the general test_seek test +// array. +// Test array is defined in manifest.js + +var manager = new MediaTestManager; + +// Exercise functionality as in test_seek-2 +function testWebM2(e) { + var v = e.target; + v.removeEventListener('loadeddata', testWebM2); + + var startPassed = false; + var endPassed = false; + var completed = false; + + ok(v.buffered.length >= 1, "Should have a buffered range"); + var halfBuffered = v.buffered.end(0) / 2; + + function start() { + if (completed) + return; + + is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start"); + is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end"); + v.currentTime=halfBuffered; + v.play(); + } + + function seekStarted() { + if (completed) + return; + + startPassed = true; + } + + function seekEnded() { + if (completed) + return; + + endPassed = true; + } + + function playbackEnded() { + if (completed) + return; + + completed = true; + ok(startPassed, "send seeking event"); + ok(endPassed, "send seeked event"); + ok(v.ended, "Checking playback has ended"); + ok(Math.abs(v.currentTime - v.duration) <= 0.1, "Checking currentTime at end: " + v.currentTime); + removeNodeAndSource(v); + manager.finished(v._token); + } + + v.addEventListener("ended", playbackEnded); + v.addEventListener("seeking", seekStarted); + v.addEventListener("seeked", seekEnded); + + start(); +} + +// Fetch the media resource using XHR so we can be sure the entire +// resource is loaded before we test buffered ranges. This ensures +// we have deterministic behaviour. +function fetch(url, fetched_callback) { + var xhr = new XMLHttpRequest(); + xhr.open("GET", url, true); + xhr.responseType = "blob"; + + var loaded = function (event) { + if (xhr.status == 200 || xhr.status == 206) { + // Request fulfilled. Note sometimes we get 206... Presumably because either + // httpd.js or Necko cached the result. + fetched_callback(window.URL.createObjectURL(xhr.response)); + } else { + ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders()); + } + }; + + xhr.addEventListener("load", loaded); + xhr.send(); +} + +function startTest(test, token) { + var onfetched = function(uri) { + var v = document.createElement('video'); + v._token = token; + v.src = uri; + v.addEventListener("loadeddata", testWebM2); + document.body.appendChild(v); + } + manager.started(token); + fetch(test.name, onfetched); +} + +SimpleTest.waitForExplicitFinish(); +manager.runTests(gCuelessWebMTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_cueless_webm_seek-3.html b/dom/media/test/test_cueless_webm_seek-3.html new file mode 100644 index 0000000000..d6e3e50d7d --- /dev/null +++ b/dom/media/test/test_cueless_webm_seek-3.html @@ -0,0 +1,120 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=657791 +--> +<head> + <title>Test for Bug 657791</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Subset of seek tests for cueless WebMs. When random seeking (rather than just +// in buffered ranges) is implemented for WebM, these tests can be removed and +// the cueless WebM(s) references can be moved to the general test_seek test +// array. +// Test array is defined in manifest.js + +var manager = new MediaTestManager; + +// Exercise functionality as in test_seek-3 +function testWebM3(e) { + var v = e.target; + v.removeEventListener('loadeddata', testWebM3); + + var completed = false; + var gotTimeupdate = false; + + ok(v.buffered.length >= 1, "Should have a buffered range"); + var halfBuffered = v.buffered.end(0) / 2; + + function start() { + if (completed) + return; + + is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start"); + is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end"); + v.currentTime=halfBuffered; + } + + function timeupdate() { + gotTimeupdate = true; + v.removeEventListener("timeupdate", timeupdate); + } + + function seekStarted() { + if (completed) + return; + + v.addEventListener("timeupdate", timeupdate); + } + + function seekEnded() { + if (completed) + return; + + var t = v.currentTime; + ok(Math.abs(t - halfBuffered) <= 0.1, "Video currentTime should be around " + halfBuffered + ": " + t); + ok(gotTimeupdate, "Should have got timeupdate between seeking and seekended"); + completed = true; + removeNodeAndSource(v); + manager.finished(v._token); + } + + v.addEventListener("seeking", seekStarted); + v.addEventListener("seeked", seekEnded); + + start() +} + +// Fetch the media resource using XHR so we can be sure the entire +// resource is loaded before we test buffered ranges. This ensures +// we have deterministic behaviour. +function fetch(url, fetched_callback) { + var xhr = new XMLHttpRequest(); + xhr.open("GET", url, true); + xhr.responseType = "blob"; + + var loaded = function (event) { + if (xhr.status == 200 || xhr.status == 206) { + // Request fulfilled. Note sometimes we get 206... Presumably because either + // httpd.js or Necko cached the result. + fetched_callback(window.URL.createObjectURL(xhr.response)); + } else { + ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders()); + } + }; + + xhr.addEventListener("load", loaded); + xhr.send(); +} + +function startTest(test, token) { + var onfetched = function(uri) { + var v = document.createElement('video'); + v._token = token; + v.src = uri; + v.addEventListener("loadeddata", testWebM3); + document.body.appendChild(v); + } + manager.started(token); + fetch(test.name, onfetched); +} + +SimpleTest.waitForExplicitFinish(); +manager.runTests(gCuelessWebMTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_currentTime.html b/dom/media/test/test_currentTime.html new file mode 100644 index 0000000000..b38c8c2c53 --- /dev/null +++ b/dom/media/test/test_currentTime.html @@ -0,0 +1,19 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: currentTime</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<video id='v1'></video><audio id='a1'></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +is(v1.currentTime, 0.0); +is(a1.currentTime, 0.0); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_decode_error.html b/dom/media/test/test_decode_error.html new file mode 100644 index 0000000000..d6d71102f1 --- /dev/null +++ b/dom/media/test/test_decode_error.html @@ -0,0 +1,66 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: unknown/invalid formats raise decode error</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) { + var ok = function (condition, name) { + SimpleTest.ok(condition, test.name + ": " + name); + } + var is = function (a, b, name) { + SimpleTest.is(a, b, test.name + ": " + name); + } + + var v = document.createElement("video"); + manager.started(token); + v.addEventListener("error", function (event) { + var el = event.currentTarget; + is(event.type, "error", "Expected event of type 'error'"); + ok(el.error, "Element 'error' attr expected to have a value"); + ok(el.error instanceof MediaError, "Element 'error' attr expected to be MediaError"); + if (v.readyState == v.HAVE_NOTHING) { + is(el.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, "Expected media not supported error"); + } else { + is(el.error.code, MediaError.MEDIA_ERR_DECODE, "Expected a decode error"); + } + ok(typeof el.error.message === 'string' || el.error.message instanceof String, "Element 'message' attr expected to be a string"); + ok(el.error.message.length > 0, "Element 'message' attr has content"); + el._sawError = true; + manager.finished(token); + }); + + v.addEventListener("loadeddata", function () { + ok(false, "Unexpected loadeddata event"); + manager.finished(token); + }); + + v.autoplay = true; + v.addEventListener("ended", function () { + ok(false, "Unexpected ended event"); + manager.finished(token); + }); + + v.src = test.name; // implicitly starts a load. +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({ + "set": [ + ["media.cache_size", 40000], + ] +}, beginTest); +function beginTest() { + manager.runTests(gDecodeErrorTests, startTest); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_decode_error_crossorigin.html b/dom/media/test/test_decode_error_crossorigin.html new file mode 100644 index 0000000000..24c1430a5b --- /dev/null +++ b/dom/media/test/test_decode_error_crossorigin.html @@ -0,0 +1,54 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Invalid formats raise decode errors with default messages for CORS cross-origin media</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +const manager = new MediaTestManager; +let gotErrSrcNotSupported = false; +let gotErrDecode = false; + +function startTest(test, token) { + const is = function(a, b, name) { + SimpleTest.is(a, b, `${test.name}: ${name}`); + }; + const v = document.createElement("video"); + manager.started(token); + v.addEventListener("error", event => { + if (v.readyState == v.HAVE_NOTHING) { + is(v.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, + "Expected code for a load error"); + is(v.error.message, "Failed to open media", + "Expected message for a load error"); + gotErrSrcNotSupported = true; + } else { + is(v.error.code, MediaError.MEDIA_ERR_DECODE, + "Expected code for a decode error"); + is(v.error.message, "Failed to decode media", + "Expected message for a decode error"); + gotErrDecode = true; + } + manager.finished(token); + }); + + v.autoplay = true; + + // CORS-cross-origin URL. + v.src = `http://example.com/tests/dom/media/test/${test.name}`; +} + +gTestPrefs.push(["media.cache_size", 40000]); +manager.onFinished = () => { + ok(gotErrSrcNotSupported, "At least one test led to src-not-supported"); + ok(gotErrDecode, "At least one test led to a decode error"); +}; +manager.runTests(gErrorTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_decoder_disable.html b/dom/media/test/test_decoder_disable.html new file mode 100644 index 0000000000..dd0d2cc51b --- /dev/null +++ b/dom/media/test/test_decoder_disable.html @@ -0,0 +1,78 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=448600 +--> +<head> + <title>Test for Bug 448600</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=448600">Mozilla Bug 448600</a> +<p id="display"></p> + + +<pre id="test"> +<script type="application/javascript"> + +function filename(uri) { + return uri.substr(uri.lastIndexOf("/")+1); +} + +function e(id) { + return document.getElementById(id); +} + +var gLoadError = {}; + +gLoadError.video1 = 0; +gLoadError.video2 = 0; +gLoadError.video3 = 0; + +var gErrorCount = 0; + +SimpleTest.waitForExplicitFinish(); + +function finishTest() { + is(e('video1').currentSrc, + "", + 'video1 currentSrc should be empty when there\'s no playable typed source children'); + is(filename(e('video2').currentSrc), + filename(e('video2').src), + 'video2 currentSrc should match src'); + is(filename(e('video3').currentSrc), + filename(e('video3').src), + 'video3 currentSrc should match src'); + + is(gLoadError.video1, 2, "Expect one error per invalid source child on video1"); + is(gLoadError.video2, 1, "Expect one error on video2"); + is(gLoadError.video3, 1, "Expect one error on video3"); + + SimpleTest.finish(); +} + +function videoError(event, id) { + gLoadError[id]++; + gErrorCount++; + if (gErrorCount >= 4) { + finishTest(); + } +} + +</script> +<!-- We make the resource URIs unique to ensure that they are (re)loaded with the new disable-decoder prefs. --> +<div id="content"> +</div> +<script> +function makeVideos() { + document.getElementById('content').innerHTML = '<video id="video1" preload="metadata"><source type="video/ogg" src="320x240.ogv?decoder_disabled=1" onerror="videoError(event, \'video1\');"/><source type="audio/wave" src="r11025_u8_c1.wav?decoder_disabled=1" id=\'s2\' onerror="videoError(event, \'video1\');"/></video><video id="video2" preload="metadata" src="320x240.ogv?decoder_disabled=2" onerror="videoError(event, \'video2\');"></video><video id="video3" preload="metadata" src="r11025_u8_c1.wav?decoder_disabled=2" onerror="videoError(event, \'video3\');"></video>'; +} + +SpecialPowers.pushPrefEnv({"set": [["media.ogg.enabled", false], ["media.wave.enabled", false]]}, makeVideos); +</script> + +</pre> + +</body> +</html> diff --git a/dom/media/test/test_defaultMuted.html b/dom/media/test/test_defaultMuted.html new file mode 100644 index 0000000000..77bfa3d29a --- /dev/null +++ b/dom/media/test/test_defaultMuted.html @@ -0,0 +1,54 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: defaultMuted</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="../../../dom/html/test/reflect.js"></script> +</head> +<body> + <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=706731">Mozilla Bug 706731</a> + <p id="display"></p> + <div id="content" style="display: none"></div> + <video id='v1'></video><audio id='a1'></audio> + <video id='v2' muted></video><audio id='a2' muted></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> + reflectBoolean({ + element: document.createElement("video"), + attribute: { content: "muted", idl: "defaultMuted" }, + }); + + reflectBoolean({ + element: document.createElement("audio"), + attribute: { content: "muted", idl: "defaultMuted" }, + }); + + var v1 = document.getElementById('v1'); + var a1 = document.getElementById('a1'); + var v2 = document.getElementById('v2'); + var a2 = document.getElementById('a2'); + + // Check that muted state correspond to the default value. + is(v1.muted, false, "v1.muted should be false by default"); + is(a1.muted, false, "a1.muted should be false by default"); + is(v2.muted, true, "v2.muted should be true by default"); + is(a2.muted, true, "a2.muted should be true by default"); + + // Changing defaultMuted value should not change current muted state. + v1.defaultMuted = true; + a1.defaultMuted = true; + v2.defaultMuted = false; + a2.defaultMuted = false; + + is(v1.muted, false, "v1.muted should not have changed"); + is(a1.muted, false, "a1.muted should not have changed"); + is(v2.muted, true, "v2.muted should not have changed"); + is(a2.muted, true, "a2.muted should not have changed"); + + mediaTestCleanup(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_delay_load.html b/dom/media/test/test_delay_load.html new file mode 100644 index 0000000000..d10812a7c1 --- /dev/null +++ b/dom/media/test/test_delay_load.html @@ -0,0 +1,108 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=479711 +--> +<head> + <title>Test for Bug 479711</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> + <script> + + var gRegisteredElements = []; + var testWindows = []; + + function register(v) { + gRegisteredElements.push(v); + } + + function loaded() { + info("onload fired!"); + + for (var i = 0; i < gRegisteredElements.length; ++i) { + var v = gRegisteredElements[i]; + ok(v.readyState >= v.HAVE_CURRENT_DATA, + v._name + ":" + v.id + " is not ready before onload fired (" + v.readyState + ")"); + } + + for (i=0; i<testWindows.length; ++i) { + testWindows[i].close(); + } + + mediaTestCleanup(); + + SimpleTest.finish(); + } + + addLoadEvent(loaded); + + </script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479711">Mozilla Bug 479711</a> +<p id="display"></p> +<div id="content" style="display: none"></div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 479711 **/ + +function createVideo(name, type, id) { + var v = document.createElement("video"); + v.preload = "metadata"; + // Make sure each video is a unique resource + v.src = name + "?" + id; + v._name = name; + v.id = id; + register(v); + return v; +} + +var test = getPlayableVideo(gSmallTests); + +// Straightforward add, causing a load. +var v = createVideo(test.name, test.type, "1"); +document.body.appendChild(v); + +// Load, add, then remove. +v = createVideo(test.name, test.type, "1"); +v.load(); +document.body.appendChild(v); +v.remove(); + +// Load and add. +v = createVideo(test.name, test.type, "2"); +v.load(); +document.body.appendChild(v); + +// Load outside of doc. +v = createVideo(test.name, test.type, "3"); +v.load(); + +// Open a new window for the following test. We open it here instead of in +// the event handler to ensure that our document load event doesn't fire while +// window.open is spinning the event loop. +var w = window.open("", "testWindow", "width=400,height=400"); +testWindows.push(w); + +v = createVideo(test.name, test.type, "4"); +v.onloadstart = function(e) { + // Using a new window to do this is a bit annoying, but if we use an iframe here, + // delaying of the iframe's load event might interfere with the firing of our load event + // in some confusing way. So it's simpler just to use another window. + w.document.body.appendChild(v); +}; +v.load(); // load started while in this document, this doc's load will block until + // the video's finished loading (in the other document). + +if (gRegisteredElements.length > 0) { + SimpleTest.waitForExplicitFinish(); +} else { + todo(false, "No types supported"); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_duration_after_error.html b/dom/media/test/test_duration_after_error.html new file mode 100644 index 0000000000..ad6bbe414f --- /dev/null +++ b/dom/media/test/test_duration_after_error.html @@ -0,0 +1,54 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of media files that should have errors</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function checkDuration(name, e, test) { + if (test.duration) { + ok(Math.abs(e.duration - test.duration) < 0.1, + name + " duration (" + e.duration + ") should be around " + test.duration); + } else { + ok(false, "Test doesn't include the duration field!") + } +} + +function startTest(test, token) { + manager.started(token); + + let v = document.createElement('video'); + v._loadedMetadata = false; + let name = test.name; + + v.onloadedmetadata = function() { + v.onloadedmetadata = null; + v._loadedMetadata = true; + ok(v._loadedMetadata , name + " has loaded metadata."); + } + + v.onerror = function() { + v.onerror = null; + ok(v._loadedMetadata , name + " should load metadata before getting error."); + checkDuration(name, v, test); + manager.finished(token); + } + + v.src = name; + document.body.appendChild(v); + v.play(); +} + +manager.runTests(gDurationTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_autoplay.html b/dom/media/test/test_eme_autoplay.html new file mode 100644 index 0000000000..010995c095 --- /dev/null +++ b/dom/media/test/test_eme_autoplay.html @@ -0,0 +1,119 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +var EMEmanifest = [ + { + name:"bipbop 10s", + tracks: [ + { + name:"video", + type:"video/mp4; codecs=\"avc1.4d4015\"", + fragments:[ "bipbop-cenc-video-10s.mp4", + ] + } + ], + keys: { + "7e571d037e571d037e571d037e571d11" : "7e5733337e5733337e5733337e573311", + }, + sessionType:"temporary", + sessionCount:1, + duration:10.01 + }, +]; + +function startTest(test, token) +{ + manager.started(token); + + let v = document.createElement("video"); + v.controls = true; + v.autoplay = true; + + document.body.appendChild(v); + + var eventCounts = { play: 0, playing: 0}; + function ForbiddenEvents(e) { + var video = e.target; + ok(video.readyState >= video.HAVE_FUTURE_DATA, "Must not have received event too early"); + is(eventCounts[e.type], 0, "event should have only be fired once"); + eventCounts[e.type]++; + } + // Log events for debugging. + var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause", "durationchange", "seeking", "seeked"]; + function logEvent(e) { + info("got " + e.type + " event"); + } + events.forEach(function(e) { + v.addEventListener(e, logEvent); + }); + v.addEventListener("play", ForbiddenEvents); + v.addEventListener("playing", ForbiddenEvents); + + var gotWaitingForKey = 0; + + let waitForKey = new EMEPromise; + v.addEventListener("waitingforkey", function() { + gotWaitingForKey += 1; + waitForKey.resolve(); + }); + + v.addEventListener("loadedmetadata", function() { + ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v), + TimeStamp(token) + " isEncrypted should be true"); + is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content"); + }); + + let finish = new EMEPromise; + v.addEventListener("playing", function() { + ok(true, TimeStamp(token) + " got playing event"); + // We expect only one waitingForKey as we delay until all sessions are ready. + // I.e. one waitingForKey should be fired, after which, we process all sessions, + // so it should not be possible to be blocked by a key after that point. + ok(gotWaitingForKey == 1, "Expected number 1 wait, got: " + gotWaitingForKey); + + finish.resolve(); + }); + + Promise.all([ + LoadInitData(v, test, token), + CreateAndSetMediaKeys(v, test, token), + LoadTest(test, v, token, false /* do not call endOfStream */), + waitForKey.promise]) + .then(values => { + let initData = values[0]; + return ProcessInitData(v, test, token, initData); + }) + .then(sessions => { + Log(token, "Updated all sessions, loading complete"); + finish.promise.then(() => CloseSessions(v, sessions)); + return finish.promise; + }) + .catch(reason => ok(false, reason)) + .then(() => manager.finished(token)); +} + +function beginTest() { + manager.runTests(EMEmanifest, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_canvas_blocked.html b/dom/media/test/test_eme_canvas_blocked.html new file mode 100644 index 0000000000..714fceafb7 --- /dev/null +++ b/dom/media/test/test_eme_canvas_blocked.html @@ -0,0 +1,60 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) +{ + manager.started(token); + + let v = document.createElement("video"); + v.preload = "auto"; // Required due to "canplay" not firing for MSE unless we do this. + + var p1 = new EMEPromise; + v.addEventListener("loadeddata", function(ev) { + var video = ev.target; + var canvas = document.createElement("canvas"); + canvas.width = video.videoWidth; + canvas.height = video.videoHeight; + document.body.appendChild(canvas); + var ctx = canvas.getContext("2d"); + var threwError = false; + try { + ctx.drawImage(video, 0, 0); + } catch (ex) { + threwError = true; + } + ok(threwError, TimeStamp(token) + " - Should throw an error when trying to draw EME video to canvas."); + p1.resolve(); + }); + + let p2 = SetupEME(v, test, token); + + Promise.all([p1.promise, p2]) + .catch(reason => ok(false, reason)) + .then(() => { + CleanUpMedia(v); + manager.finished(token); + }); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_createMediaKeys_iframes.html b/dom/media/test/test_eme_createMediaKeys_iframes.html new file mode 100644 index 0000000000..2309fd76f4 --- /dev/null +++ b/dom/media/test/test_eme_createMediaKeys_iframes.html @@ -0,0 +1,201 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test creation of MediaKeys in iframes</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody"> +// Helper functions. + +// We take navigator explicitly as an argument to avoid ambiguity in fetching +// it. This is to avoid issues with the following +// ``` +// iframe.contentWindow.createMediaKeys = createMediaKeys; +// await iframe.contentWindow.createMediaKeys(); +// ``` +// If we don't pass a navigator, and just use `navigator` in the function, this +// ends up being equivalent to +// ``` +// iframe.contentWindow.createMediaKeys = createMediaKeys; +// await iframe.contentWindow.createMediaKeys(window.navigator); +// ``` +// i.e. the function will use the navigator from the global window for the top +// browsing context, not the iframe's. This would result in the tests not +// correctly testing within the iframe. +async function createMediaKeys(aNavigator) { + const clearKeyOptions = [ + { + initDataTypes: ["webm"], + videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }], + }, + ]; + + let access = await aNavigator.requestMediaKeySystemAccess( + "org.w3.clearkey", + clearKeyOptions + ); + + return access.createMediaKeys(); +} +// End helper functions. + +// Setup for the tests. +add_task(async () => { + info("Setting up EME prefs"); + // Wrap EME pref setup in a promise so it plays nice with add_task. + return new Promise(r => { + SetupEMEPref(r); + }); +}); + +// These tests check that the following work using different iframe combinations +// - navigator.requestMediaKeySystem(...) successfully grants access. +// - the resulting MediaKeySystemAccess object's createMediaKeys() creates +// MediaKeys as expected. + +// Same origin iframe, using src attribute, wait for onload. +add_task(async () => { + info( + "Starting same origin iframe, using src attribute, wait for onload test" + ); + let iframe = document.createElement("iframe"); + let iframeLoadPromise = new Promise(r => { + iframe.onload = r; + }); + iframe.src = "file_eme_createMediaKeys.html"; + document.body.appendChild(iframe); + await iframeLoadPromise; + info("iframe loaded"); + + // Setup our handler for when the iframe messages to tell us if it + // created MediaKeys or not. + let iframeMessagePromise = new Promise(r => { + window.onmessage = message => { + is( + message.data, + "successCreatingMediaKeys", + "iframe should have posted us a message saying keys were successfully created" + ); + r(); + }; + }); + // Post a message to the iframe to ask it to try and create media keys. + iframe.contentWindow.postMessage("", "*"); + // Wait until we've got a message back from our iframe. + await iframeMessagePromise; +}); + +// Same origin iframe, call via JS, wait for onload. +add_task(async () => { + info("Starting same origin iframe, call via JS, wait for onload test"); + let iframe = document.createElement("iframe"); + let iframeLoadPromise = new Promise(r => { + iframe.onload = r; + }); + iframe.src = ""; // No src iframes are same origin. + document.body.appendChild(iframe); + await iframeLoadPromise; + info("iframe loaded"); + + try { + iframe.contentWindow.createMediaKeys = createMediaKeys; + let mediaKeys = await iframe.contentWindow.createMediaKeys( + iframe.contentWindow.navigator + ); + ok(mediaKeys, "Should get media keys"); + } catch (e) { + ok( + false, + `Should not get any errors while trying to get media keys, got ${e}` + ); + } +}); + +// Same origin iframe, call via JS, *do not* wait for onload. +// +// Note, sites shouldn't do this, because +// https://bugzilla.mozilla.org/show_bug.cgi?id=543435 +// means not waiting for onload results in weird behavior, however +// https://bugzilla.mozilla.org/show_bug.cgi?id=1675360 +// shows sites doing this in the wild because historically this worked in +// Firefox. +// +// Breaking this test case isn't necessarily against any specifications +// I'm (bryce) aware of, but it will probably break site compat, so be really +// sure you want to before doing so. +add_task(async () => { + info( + "Starting same origin iframe, call via JS, *do not* wait for onload test" + ); + let iframe = document.createElement("iframe"); + let iframeLoadPromise = new Promise(r => { + iframe.onload = r; + }); + iframe.src = ""; // No src iframes are same origin. + document.body.appendChild(iframe); + info("iframe appended (we're not waiting for load)"); + + try { + iframe.contentWindow.createMediaKeys = createMediaKeys; + let mediaKeys = await iframe.contentWindow.createMediaKeys( + iframe.contentWindow.navigator + ); + ok(mediaKeys, "Should get media keys"); + + // We await the load to see if they keys persist through the load. + // This could fail if gecko internally associates the keys with the + // about:blank page that is replaced by the load. + await iframeLoadPromise; + ok(mediaKeys, "Media keys should still exist after the load"); + } catch (e) { + ok( + false, + `Should not get any errors while trying to get media keys, got ${e}` + ); + } +}); + +// Different origin iframe, using src attribute, wait for onload +add_task(async () => { + info( + "Starting different origin iframe, using src attribute, wait for onload test" + ); + let iframe = document.createElement("iframe"); + let iframeLoadPromise = new Promise(r => { + iframe.onload = r; + }); + // Make our iframe cross origin (see build/pgo/server-locations.txt for more + // info the url used). + iframe.src = + "https://w3c-test.org:443/tests/dom/media/test/file_eme_createMediaKeys.html"; + iframe.allow = "encrypted-media"; + document.body.appendChild(iframe); + await iframeLoadPromise; + info("iframe loaded"); + + // Setup our handler for when the iframe messages to tell us if it + // created MediaKeys or not. + let iframeMessagePromise = new Promise(r => { + window.onmessage = message => { + is( + message.data, + "successCreatingMediaKeys", + "iframe should have posted us a message saying keys were successfully created" + ); + r(); + }; + }); + // Post a message to the iframe to ask it to try and create media keys. + iframe.contentWindow.postMessage("", "*"); + // Wait until we've got a message back from our iframe. + await iframeMessagePromise; +}); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_detach_media_keys.html b/dom/media/test/test_eme_detach_media_keys.html new file mode 100644 index 0000000000..aad035db14 --- /dev/null +++ b/dom/media/test/test_eme_detach_media_keys.html @@ -0,0 +1,63 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<video id="v" controls></video> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function createAndSet() { + return new Promise(function(resolve, reject) { + var m; + navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig) + .then(function (access) { + return access.createMediaKeys(); + }) + .then(function (mediaKeys) { + m = mediaKeys; + return document.getElementById("v").setMediaKeys(mediaKeys); + }) + .then(function() { + resolve(m); + }); + } +)} + +var m1,m2; + +// Test that if we create and set two MediaKeys on one video element, +// that if the first MediaKeys we set on the media elemnt is still usable +// after the second MediaKeys has been set on the media element. +SetupEMEPref(() => { + createAndSet().then((m) => { + m1 = m; // Stash MediaKeys. + return createAndSet(); + }) + .then((m) => { + m2 = m; + is(document.getElementById("v").mediaKeys, m2, "Should have set MediaKeys on media element"); + ok(document.getElementById("v").mediaKeys != m1, "First MediaKeys should no longer be set on media element"); + var s = m1.createSession("temporary"); + return s.generateRequest("webm", StringToArrayBuffer(atob('YAYeAX5Hfod+V9ANHtANHg=='))); + }) + .then(() => { + ok(true, "Was able to generateRequest using second CDM"); + SimpleTest.finish(); + }, () => { + ok(false, "Was *NOT* able to generateRequest using second CDM"); + SimpleTest.finish(); + }); +}); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html b/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html new file mode 100644 index 0000000000..0e379a0e5a --- /dev/null +++ b/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html @@ -0,0 +1,145 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script> +</head> +<body> +<pre id="test"> +<video id="v" controls></video> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +var EMEmanifest = [ + { + name:"bipbop 10s", + tracks: [ + { + name:"video", + type:"video/mp4; codecs=\"avc1.4d4015\"", + fragments:[ "bipbop-cenc-video-10s.mp4", + ] + } + ], + keys: { + "7e571d037e571d037e571d037e571d11" : "7e5733337e5733337e5733337e573311", + }, + sessionType:"temporary", + sessionCount:1, + duration:10.01 + }, +]; + +function sleep(time) { + return new Promise((resolve) => setTimeout(resolve, time)); +} + +// To check if playback can be blocked and resumed correctly after +// detaching original mediakeys and reattach it back. +function startTest(test, token) +{ + manager.started(token); + + var mk_ori; + let finish = new EMEPromise; + + let v = document.getElementById("v"); + let sessions = []; + function onSessionCreated(session) { + sessions.push(session); + } + + function closeSessions() { + let p = new EMEPromise; + Promise.all(sessions.map(s => s.close())) + .then(p.resolve, p.reject); + return p.promise; + } + + function setMediaKeysToElement(mk, solve, reject) { + v.setMediaKeys(mk).then(solve, reject); + } + + function ReattachOriMediaKeys() { + function onOriMediaKeysSetOK() { + ok(true, TimeStamp(token) + " (ENCRYPTED) Set original MediaKeys back OK!"); + } + function onOriMediaKeysSetFailed() { + ok(false, " Failed to set original mediakeys back."); + } + + function onCanPlayAgain(ev) { + Promise.all([closeSessions()]) + .then(() => { + ok(true, " (ENCRYPTED) Playback can be resumed."); + manager.finished(token); + }, () => { + ok(false, TimeStamp(token) + " Sessions are closed incorrectly."); + manager.finished(token); + }); + } + + once(v, "canplay", onCanPlayAgain); + setMediaKeysToElement(mk_ori, onOriMediaKeysSetOK, onOriMediaKeysSetFailed) + } + + function triggerSeek() { + v.currentTime = v.duration / 2; + } + + function onCanPlay(ev) { + function onSetMediaKeysToNullOK() { + ok(true, TimeStamp(token) + " Set MediaKeys to null. OK!"); + + triggerSeek(); + + SimpleTest.requestFlakyTimeout("To reattach mediakeys back again in 5s."); + sleep(5000).then(ReattachOriMediaKeys); + } + function onSetMediaKeysToNullFailed() { + ok(false, TimeStamp(token) + " Set MediaKeys to null. FAILED!"); + } + + SimpleTest.requestFlakyTimeout("To detach mediakeys after receiving 'canplay' event in 2s"); + sleep(2000).then(() => { + setMediaKeysToElement(null, onSetMediaKeysToNullOK, onSetMediaKeysToNullFailed); + }); + } + + once(v, "canplay", onCanPlay); + + var p1 = LoadInitData(v, test, token); + var p2 = CreateAndSetMediaKeys(v, test, token); + var p3 = LoadTest(test, v, token); + Promise.all([p1, p2, p3]) + .then(values => { + let initData = values[0]; + // stash the mediakeys + mk_ori = v.mediaKeys; + initData.map(ev => { + let session = v.mediaKeys.createSession(); + onSessionCreated(session); + MakeRequest(test, token, ev, session); + }); + }) + .then(() => { + return finish.promise; + }) + .catch(reason => ok(false, reason)) +} + +function beginTest() { + manager.runTests(EMEmanifest, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_getstatusforpolicy.html b/dom/media/test/test_eme_getstatusforpolicy.html new file mode 100644 index 0000000000..fd2c4b11be --- /dev/null +++ b/dom/media/test/test_eme_getstatusforpolicy.html @@ -0,0 +1,57 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<video id="v" controls></video> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function createMediaKeysAndSet() { + return navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig) + .then(function (access) { + return access.createMediaKeys(); + }) + .then(function (mediaKeys) { + document.getElementById("v").setMediaKeys(mediaKeys); + return mediaKeys; + }); +} + +function test() { + SetupEMEPref(() => { + createMediaKeysAndSet() + .then((m) => { + let video = document.getElementById("v"); + is(video.mediaKeys, m, "Should have set MediaKeys on media element"); + // getStatusForPolicy() is not suppored by ClearKey key system. + // The promise will always be rejected with NotSupportedError. + return video.mediaKeys.getStatusForPolicy({minHdcpVersion: "hdcp-2.0"}); + }) + .then((mediaKeyStatus) => { + ok(false, "Promise of getStatusForPolicy should not be resolved with clearkey key system"); + }) + // Promise rejected with NotSupportedError as expected. + .catch(reason => is("NotSupportedError", reason.name, + "Promise should be rejected with NotSupportedError.")) + .then(() => SimpleTest.finish()); + }); +} + +SpecialPowers.pushPrefEnv({"set": + [ + ["media.eme.hdcp-policy-check.enabled", true], + ] + }, test); + +</script> +</pre> +</body> +</html>
\ No newline at end of file diff --git a/dom/media/test/test_eme_initDataTypes.html b/dom/media/test/test_eme_initDataTypes.html new file mode 100644 index 0000000000..5dfb873138 --- /dev/null +++ b/dom/media/test/test_eme_initDataTypes.html @@ -0,0 +1,133 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var tests = [ + { + name: "One keyId", + initDataType: 'keyids', + initData: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A"]}', + expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A"],"type":"temporary"}', + sessionType: 'temporary', + expectPass: true, + }, + { + name: "Two keyIds", + initDataType: 'keyids', + initData: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}', + expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"temporary"}', + sessionType: 'temporary', + expectPass: true, + }, + { + name: "Two keyIds, temporary session", + initDataType: 'keyids', + initData: '{"type":"temporary", "kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}', + expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"temporary"}', + sessionType: 'temporary', + expectPass: true, + }, + { + name: "Two keyIds, persistent session, type before kids", + initDataType: 'keyids', + initData: '{"type":"persistent-license", "kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}', + expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"persistent-license"}', + sessionType: 'persistent-license', + expectPass: false, + }, + { + name: "Invalid keyId", + initDataType: 'keyids', + initData: '{"kids":["0"]}', + sessionType: 'temporary', + expectPass: false, + }, + { + name: "Empty keyId", + initDataType: 'keyids', + initData: '{"kids":[""]}', + sessionType: 'temporary', + expectPass: false, + }, + { + name: "Invalid initData", + initDataType: 'keyids', + initData: 'invalid initData', + sessionType: 'temporary', + expectPass: false, + }, + { + name: "'webm' initDataType", + initDataType: 'webm', + initData: 'YAYeAX5Hfod+V9ANHtANHg==', + expectedRequest: '{"kids":["YAYeAX5Hfod-V9ANHtANHg"],"type":"temporary"}', + sessionType: 'temporary', + expectPass: true, + }, + { + name: "'webm' initDataType with non 16 byte keyid", + initDataType: 'webm', + initData: 'YAYeAX5Hfod', + expectedRequest: '{\"kids\":[\"YAYeAX5Hfoc\"],\"type\":\"temporary\"}', + sessionType: 'temporary', + expectPass: true, + }, +]; + +function PrepareInitData(initDataType, initData) +{ + if (initDataType == "keyids") { + return new TextEncoder().encode(initData); + } else if (initDataType == "webm") { + return StringToArrayBuffer(atob(initData)); + } +} + +function Test(test) { + return new Promise(function(resolve, reject) { + var configs = [{ + initDataTypes: [test.initDataType], + videoCapabilities: [{contentType: 'video/mp4' }], + }]; + navigator.requestMediaKeySystemAccess('org.w3.clearkey', configs) + .then((access) => access.createMediaKeys()) + .then((mediaKeys) => { + var session = mediaKeys.createSession(test.sessionType); + session.addEventListener("message", function(event) { + is(event.messageType, "license-request", "'" + test.name + "' MediaKeyMessage type should be license-request."); + var text = new TextDecoder().decode(event.message); + is(text, test.expectedRequest, "'" + test.name + "' got expected response."); + is(text == test.expectedRequest, test.expectPass, + "'" + test.name + "' expected to " + (test.expectPass ? "pass" : "fail")); + resolve(); + }); + var initData = PrepareInitData(test.initDataType, test.initData); + return session.generateRequest(test.initDataType, initData); + } + ).catch((x) => { + ok(!test.expectPass, "'" + test.name + "' expected to fail."); + resolve(); + }); + }); +} + +function beginTest() { + Promise.all(tests.map(Test)).then(function() { SimpleTest.finish(); }); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_missing_pssh.html b/dom/media/test/test_eme_missing_pssh.html new file mode 100644 index 0000000000..d39a7c9821 --- /dev/null +++ b/dom/media/test/test_eme_missing_pssh.html @@ -0,0 +1,92 @@ +<!DOCTYPE HTML> +<html> + <head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> + </head> + <body> + <audio controls id="audio"></audio> + <pre id="test"> + <script class="testbody" type="text/javascript"> + + // Tests that a fragmented MP4 file without a PSSH, but with valid encrypted + // tracks with valid TENC boxes, is able to load with EME. + // We setup MSE before starting up EME, so that we exercise the "waiting for + // cdm" step in the MediaDecoderStateMachine. + + SimpleTest.waitForExplicitFinish(); + + var pssh = [ + 0x00, 0x00, 0x00, 0x00, + 0x70, 0x73, 0x73, 0x68, // BMFF box header (76 bytes, 'pssh') + 0x01, 0x00, 0x00, 0x00, // Full box header (version = 1, flags = 0) + 0x10, 0x77, 0xef, 0xec, 0xc0, 0xb2, 0x4d, 0x02, // SystemID + 0xac, 0xe3, 0x3c, 0x1e, 0x52, 0xe2, 0xfb, 0x4b, + 0x00, 0x00, 0x00, 0x01, // KID_count (1) + 0x2f, 0xef, 0x8a, 0xd8, 0x12, 0xdf, 0x42, 0x97, + 0x83, 0xe9, 0xbf, 0x6e, 0x5e, 0x49, 0x3e, 0x53, + 0x00, 0x00, 0x00, 0x00 // Size of Data (0) + ]; + + var audio = document.getElementById("audio"); + + function LoadEME() { + var options = [{ + initDataType: 'cenc', + audioType: 'audio/mp4; codecs="mp4a.40.2"', + }]; + navigator.requestMediaKeySystemAccess("org.w3.clearkey", options) + .then((keySystemAccess) => { + return keySystemAccess.createMediaKeys(); + }, bail("Failed to request key system access.")) + + .then((mediaKeys) => { + audio.setMediaKeys(mediaKeys); + var session = mediaKeys.createSession(); + once(session, "message", (message) => { + is(message.messageType, 'license-request', "Expected a license-request"); + var license = new TextEncoder().encode(JSON.stringify({ + 'keys': [{ + 'kty':'oct', + 'kid':'L--K2BLfQpeD6b9uXkk-Uw', + 'k':HexToBase64('7f412f0575f44f718259beef56ec7771') + }], + 'type': 'temporary' + })); + session.update(license); + }); + session.generateRequest('cenc', new Uint8Array(pssh)); + }); + } + + function DownloadMedia(url, type, mediaSource) { + return new Promise(function(resolve, reject) { + var sourceBuffer = mediaSource.addSourceBuffer(type); + fetchWithXHR(url, (response) => { + once(sourceBuffer, "updateend", resolve); + sourceBuffer.appendBuffer(new Uint8Array(response)); + }); + }); + } + + function LoadMSE() { + var ms = new MediaSource(); + audio.src = URL.createObjectURL(ms); + + once(ms, "sourceopen", ()=>{ + DownloadMedia('short-audio-fragmented-cenc-without-pssh.mp4', 'audio/mp4; codecs="mp4a.40.2"', ms) + .then(() => { ms.endOfStream(); LoadEME();}); + }); + + audio.addEventListener("loadeddata", SimpleTest.finish); + } + + SetupEMEPref(LoadMSE); + + </script> + </pre> + </body> +</html> diff --git a/dom/media/test/test_eme_non_mse_fails.html b/dom/media/test/test_eme_non_mse_fails.html new file mode 100644 index 0000000000..efd87b0a95 --- /dev/null +++ b/dom/media/test/test_eme_non_mse_fails.html @@ -0,0 +1,100 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Bug 1131392 - Test that EME does not work for non-MSE media</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +function DoSetMediaKeys(v, test, token) +{ + var options = [{ + initDataTypes: ["cenc"], + audioCapabilities: [{contentType: test.audioType}], + videoCapabilities: [{contentType: test.videoType}], + }]; + + return navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options) + + .then(function(keySystemAccess) { + return keySystemAccess.createMediaKeys(); + }) + + .catch(function() { + ok(false, token + " was not expecting failure (yet)"); + }) + + .then(function(mediaKeys) { + return v.setMediaKeys(mediaKeys); + }); +} + +function TestSetMediaKeys(test, token) +{ + manager.started(token); + + var v = document.createElement("video"); + + v.addEventListener("encrypted", function() { + ok(false, token + " should not fire encrypted event"); + }); + + var loadedMetadata = false; + v.addEventListener("loadedmetadata", function() { + loadedMetadata = true; + }); + + v.addEventListener("error", function() { + ok(true, token + " expected error event"); + ok(loadedMetadata, token + " expected loadedmetadata to have fired"); + manager.finished(token); + }); + + v.src = test.name; +} + +function TestSetSrc(test, token) +{ + manager.started(token); + + var v = document.createElement("video"); + v.addEventListener("error", function(err) { + ok(true, token + " got error setting src on video element, as expected"); + manager.finished(token); + }); + + DoSetMediaKeys(v, test, token) + + .then(function() { + v.src = test.name; + }) + + .catch(function() { + ok(false, token + " got error setting media keys"); + }); +} + +function startTest(test, token) +{ + TestSetMediaKeys(test, token + "_setMediaKeys"); + TestSetSrc(test, token + "_setSrc"); +} + +function beginTest() { + manager.runTests(gEMENonMSEFailTests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_playback.html b/dom/media/test/test_eme_playback.html new file mode 100644 index 0000000000..dc6092e486 --- /dev/null +++ b/dom/media/test/test_eme_playback.html @@ -0,0 +1,193 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +function ArrayBuffersEqual(a, b) { + if (a.byteLength != b.byteLength) { + return false; + } + var ua = new Uint8Array(a); + var ub = new Uint8Array(b); + for (var i = 0; i < ua.length; i++) { + if (ua[i] != ub[i]) { + return false; + } + } + return true; +} + +function KeysChangeFunc(session, keys, token) { + session.keyIdsReceived = []; + for (var keyid in keys) { + Log(token, "Set " + keyid + " to false in session[" + session.sessionId + "].keyIdsReceived"); + session.keyIdsReceived[keyid] = false; + } + return function(ev) { + var s = ev.target; + s.gotKeysChanged = true; + + var keyList = []; + var valueList = []; + var map = s.keyStatuses; + + // Test that accessing keys not known to the CDM has expected behaviour. + var absentKey = new Uint8Array([0xeb, 0xdd, 0x62, 0xf1, 0x68, 0x14, 0xd2, 0x7b, + 0x68, 0xef, 0x12, 0x2a, 0xfc, 0xe4, 0xae, 0x3c]); + is(map.has(absentKey), false, "Shouldn't have a key that's not in the media"); + is(map.get(absentKey), undefined, "Unknown keys should undefined status"); + + // Verify known keys have expected status. + for (let [key, val] of map.entries()) { + is(key.constructor, ArrayBuffer, "keyId should be ArrayBuffer"); + ok(map.has(key), "MediaKeyStatusMap.has() should work."); + is(map.get(key), val, "MediaKeyStatusMap.get() should work."); + keyList.push(key); + valueList.push(val); + is(val, "usable", token + ": key status should be usable"); + var kid = Base64ToHex(window.btoa(ArrayBufferToString(key))); + ok(kid in s.keyIdsReceived, TimeStamp(token) + " session[" + s.sessionId + "].keyIdsReceived contained " + kid + " as expected."); + s.keyIdsReceived[kid] = true; + } + + var index = 0; + for (var keyId of map.keys()) { + ok(ArrayBuffersEqual(keyId, keyList[index]), "MediaKeyStatusMap.keys() should correspond to entries"); + index++; + } + index = 0; + for (let val of map.values()) { + is(val, valueList[index], "MediaKeyStatusMap.values() should correspond to entries"); + index++; + } + } +} + +function startTest(test, token) +{ + manager.started(token); + + var sessions = []; + + function onSessionCreated(session) { + sessions.push(session); + session.addEventListener("keystatuseschange", KeysChangeFunc(session, test.keys, token)); + + session.numKeystatuseschangeEvents = 0; + session.numOnkeystatuseschangeEvents = 0; + + session.addEventListener("keystatuseschange", function() { + session.numKeystatuseschangeEvents += 1; + }); + session.onkeystatuseschange = function() { + session.numOnkeystatuseschangeEvents += 1; + }; + + session.numMessageEvents = 0; + session.numOnMessageEvents = 0; + session.addEventListener("message", function() { + session.numMessageEvents += 1; + }); + session.onmessage = function() { + session.numOnMessageEvents += 1; + }; + } + + let v = document.createElement("video"); + document.body.appendChild(v); + + var gotEncrypted = 0; + let finish = new EMEPromise; + + v.addEventListener("encrypted", function(ev) { + gotEncrypted += 1; + }); + + v.addEventListener("loadedmetadata", function() { + ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v), + TimeStamp(token) + " isEncrypted should be true"); + is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content"); + }); + + v.addEventListener("ended", function(ev) { + ok(true, TimeStamp(token) + " got ended event"); + + is(gotEncrypted, test.sessionCount, + TimeStamp(token) + " encrypted events expected: " + test.sessionCount + + ", actual: " + gotEncrypted); + + ok(Math.abs(test.duration - v.duration) < 0.1, + TimeStamp(token) + " Duration of video should be corrrect"); + ok(Math.abs(test.duration - v.currentTime) < 0.1, + TimeStamp(token) + " Current time should be same as duration"); + + // Verify all sessions had all keys went sent to the CDM usable, and thus + // that we received keystatuseschange event(s). + is(sessions.length, test.sessionCount, TimeStamp(token) + " should have " + + test.sessionCount + + " session" + (test.sessionCount === 1 ? "" : "s")); + var keyIdsReceived = []; + for (var keyid in test.keys) { keyIdsReceived[keyid] = false; } + for (var i = 0; i < sessions.length; i++) { + var session = sessions[i]; + ok(session.gotKeysChanged, + TimeStamp(token) + " session[" + session.sessionId + + "] should have received at least one keychange event"); + for (let kid in session.keyIdsReceived) { + Log(token, "session[" + session.sessionId + "] key " + kid + " = " + (session.keyIdsReceived[kid] ? "true" : "false")); + if (session.keyIdsReceived[kid]) { keyIdsReceived[kid] = true; } + } + ok(session.numKeystatuseschangeEvents > 0, TimeStamp(token) + " should get key status changes"); + is(session.numKeystatuseschangeEvents, session.numOnkeystatuseschangeEvents, + TimeStamp(token) + " should have as many keystatuseschange event listener calls as event handler calls."); + + ok(session.numMessageEvents > 0, TimeStamp(token) + " should get message events"); + is(session.numMessageEvents, session.numOnMessageEvents, + TimeStamp(token) + " should have as many message event listener calls as event handler calls."); + } + for (let kid in keyIdsReceived) { + ok(keyIdsReceived[kid], TimeStamp(token) + " key with id " + kid + " was usable as expected"); + } + + CloseSessions(v, sessions).then(finish.resolve, finish.reject); + }); + + Promise.all([ + LoadInitData(v, test, token), + CreateAndSetMediaKeys(v, test, token), + LoadTest(test, v, token)]) + .then(values => { + v.play(); + let initData = values[0]; + initData.map(ev => { + let session = v.mediaKeys.createSession(); + onSessionCreated(session); + MakeRequest(test, token, ev, session); + }); + return finish.promise; + }) + .catch(reason => ok(false, reason)) + .then(() => manager.finished(token)); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_pssh_in_moof.html b/dom/media/test/test_eme_pssh_in_moof.html new file mode 100644 index 0000000000..d1965be844 --- /dev/null +++ b/dom/media/test/test_eme_pssh_in_moof.html @@ -0,0 +1,141 @@ +<!DOCTYPE HTML>
+<html>
+
+ <head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+ </head>
+
+ <body>
+ <script class="testbody" type="text/javascript">
+ let manager = new MediaTestManager;
+
+ let psshTests = [
+ { name: "bear-640x360-cenc-key-rotation",
+ tracks : [
+ {
+ name: "video",
+ type: "video/mp4; codecs=\"avc1.64000d\"",
+ fragments: ["bear-640x360-v_frag-cenc-key_rotation.mp4"],
+ initDatas: 6
+ },
+ {
+ name: "audio",
+ type: "audio/mp4; codecs=\"mp4a.40.2\"",
+ fragments: ["bear-640x360-a_frag-cenc-key_rotation.mp4"],
+ initDatas: 7
+ }
+ ],
+ keys : {
+ "30313233343536373839303132333435" :
+ "ebdd62f16814d27b68ef122afce4ae3c"
+ }
+ },
+ { name: "bipbop-clearkey-keyrotation-clear-lead",
+ tracks : [
+ {
+ name: "video",
+ type: "video/mp4; codecs=\"avc1.4d4015\"",
+ fragments: ["bipbop-clearkey-keyrotation-clear-lead-video.mp4"],
+ initDatas: 3
+ },
+ {
+ name: "audio",
+ type: "audio/mp4; codecs=\"mp4a.40.2\"",
+ fragments: ["bipbop-clearkey-keyrotation-clear-lead-audio.mp4"],
+ initDatas: 3
+ }
+ ],
+ keys : {
+ "00112233445566778899aabbccddeeff" :
+ "00112233445566778899aabbccddeeff",
+ "112233445566778899aabbccddeeff00" :
+ "112233445566778899aabbccddeeff00"
+ }
+ }
+ ];
+
+ // Specialized create media keys function, since the one in eme.js relies
+ // on listening for encrypted events, and we want to manage those
+ // ourselves within this test.
+ async function createAndSetMediaKeys(video, test, token) {
+ function streamType(type) {
+ var x = test.tracks.find(o => o.name == type);
+ return x ? x.type : undefined;
+ }
+
+ var configuration = { initDataTypes: ["cenc"] };
+ if (streamType("video")) {
+ configuration.videoCapabilities = [{contentType: streamType("video")}];
+ }
+ if (streamType("audio")) {
+ configuration.audioCapabilities = [{contentType: streamType("audio")}];
+ }
+ let keySystemAccess = await navigator.requestMediaKeySystemAccess(
+ "org.w3.clearkey", [configuration]);
+ let mediaKeys = await keySystemAccess.createMediaKeys();
+ return video.setMediaKeys(mediaKeys);
+ }
+
+ function startTest(test, token) {
+ manager.started(token);
+ let video = document.createElement("video");
+ document.body.appendChild(video);
+
+ let encryptedEventCount = 0;
+
+ let initDatas = new Map();
+
+ function handleEncrypted(event) {
+ // We get one 'encrypted' event for every run of contiguous PSSH boxes
+ // in each stream. Note that some of the PSSH boxes contained in the
+ // "bear" streams aren't in the Common Open PSSH box format, so our
+ // ClearKey CDM will reject those license requests. Some of the init
+ // data is also repeated.
+ encryptedEventCount++;
+
+ let hexStr = StringToHex(new TextDecoder().decode(event.initData));
+ if (initDatas.has(hexStr)) {
+ // Already have a session for this.
+ return;
+ }
+
+ let initData = new Uint8Array(event.initData);
+ is(event.initDataType, "cenc", "'encrypted' event should have 'cenc' " +
+ "initDataType");
+ Log(token, "encrypted event => len=" + initData.length + " " + hexStr);
+ let session = event.target.mediaKeys.createSession();
+ initDatas.set(hexStr, session);
+ session.addEventListener("message", e => {
+ e.target.update(
+ GenerateClearKeyLicense(e.message, test.keys));
+ });
+
+ session.generateRequest(event.initDataType, event.initData);
+ }
+
+ video.addEventListener("encrypted", handleEncrypted);
+
+ video.addEventListener("ended", () => {
+ let expectedEncryptedEvents =
+ test.tracks.reduce((sum, track) => sum += track.initDatas, 0);
+ is(encryptedEventCount, expectedEncryptedEvents,
+ "Should get one 'encrypted' event per run of contiguous PSSH " +
+ "boxes in media.");
+ manager.finished(token);
+ });
+
+ video.autoplay = true;
+
+ createAndSetMediaKeys(video, test, token)
+ .then(() => LoadTest(test, video, token))
+ }
+
+ manager.runTests(psshTests, startTest);
+
+ </script>
+ </body>
+</html>
diff --git a/dom/media/test/test_eme_requestKeySystemAccess.html b/dom/media/test/test_eme_requestKeySystemAccess.html new file mode 100644 index 0000000000..59c7ef566d --- /dev/null +++ b/dom/media/test/test_eme_requestKeySystemAccess.html @@ -0,0 +1,480 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +const SUPPORTED_LABEL = "pass label"; + +function ValidateConfig(name, expected, observed) { + info("ValidateConfig " + name); + info("expected cfg=" + JSON.stringify(expected)); + info("observed cfg=" + JSON.stringify(observed)); + + is(observed.label, expected.label, name + " label should match"); + if (expected.initDataTypes) { + ok(expected.initDataTypes.every((element, index, array) => observed.initDataTypes.includes(element)), name + " initDataTypes should match."); + } + + if (expected.audioCapabilities) { + ok(expected.audioCapabilities.length == 1, "Test function can only handle one capability."); + ok(observed.audioCapabilities.length == 1, "Test function can only handle one capability."); + is(observed.audioCapabilities[0].contentType, expected.audioCapabilities[0].contentType, name + " audioCapabilities should match."); + } + if (typeof expected.videoCapabilities !== 'undefined') { + info("expected.videoCapabilities=" + expected.videoCapabilities); + dump("expected.videoCapabilities=" + expected.videoCapabilities + "\n"); + ok(expected.videoCapabilities.length == 1, "Test function can only handle one capability."); + ok(observed.videoCapabilities.length == 1, "Test function can only handle one capability."); + is(observed.videoCapabilities[0].contentType, expected.videoCapabilities[0].contentType, name + " videoCapabilities should match."); + } + if (expected.sessionTypes) { + is(expected.sessionTypes.length, observed.sessionTypes.length, "Should have expected number of sessionTypes"); + for (var i = 0; i < expected.sessionTypes.length; i++) { + is(expected[i], observed[i], "Session type " + i + " should match"); + } + } +} + +function Test(test) { + var name = "'" + test.name + "'"; + return new Promise(function(resolve, reject) { + var p; + if (test.options) { + var keySystem = (test.keySystem !== undefined) ? test.keySystem : CLEARKEY_KEYSYSTEM; + p = navigator.requestMediaKeySystemAccess(keySystem, test.options); + } else { + p = navigator.requestMediaKeySystemAccess(keySystem); + } + p.then( + function(keySystemAccess) { + ok(test.shouldPass, name + " passed and was expected to " + (test.shouldPass ? "pass" : "fail")); + is(keySystemAccess.keySystem, CLEARKEY_KEYSYSTEM, "CDM keySystem should be in MediaKeySystemAccess.keySystem"); + ValidateConfig(name, test.expectedConfig, keySystemAccess.getConfiguration()); + resolve(); + }, + function(ex) { + if (test.shouldPass) { + info(name + " failed: " + ex); + } + ok(!test.shouldPass, name + " failed and was expected to " + (test.shouldPass ? "pass" : "fail")); + resolve(); + }); + }); +} + +var tests = [ + { + name: 'Empty keySystem string', + keySystem: '', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + } + ], + shouldPass: false, + }, + { + name: 'Empty options specified', + options: [ ], + shouldPass: false, + }, + { + name: 'Undefined options', + shouldPass: false, + }, + { + name: 'Basic MP4 cenc', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + videoCapabilities: [{contentType: 'video/mp4'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + videoCapabilities: [{contentType: 'video/mp4'}], + }, + shouldPass: true, + }, + { + name: 'Invalid keysystem failure', + keySystem: 'bogusKeySystem', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + } + ], + shouldPass: false, + }, + { + name: 'Invalid initDataType', + options: [ + { + initDataTypes: ['bogus'], + audioCapabilities: [{contentType: 'audio/mp4'}], + } + ], + shouldPass: false, + }, + { + name: 'Valid initDataType after invalid', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['bogus', 'invalid', 'cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + }, + shouldPass: true, + }, + { + name: 'Invalid videoType', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/bogus'}], + } + ], + shouldPass: false, + }, + { + name: 'Invalid distinctiveIdentifier fails', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + distinctiveIdentifier: 'bogus', + persistentState: 'bogus', + } + ], + shouldPass: false, + }, + { + name: 'distinctiveIdentifier is prohibited for ClearKey', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + distinctiveIdentifier: 'required', + } + ], + shouldPass: false, + }, + { + name: 'Invalid persistentState fails', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + persistentState: 'bogus', + } + ], + shouldPass: false, + }, + { + name: 'Invalid robustness unsupported', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4', robustness: 'very much so'}], + } + ], + shouldPass: false, + }, + { + name: 'Unexpected config entry should be ignored', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + unexpectedEntry: 'this should be ignored', + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + }, + shouldPass: true, + }, + { + name: 'Invalid option followed by valid', + options: [ + { + label: "this config should not be supported", + initDataTypes: ['bogus'], + }, + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + }, + shouldPass: true, + }, + { + name: 'Persistent-license should not be supported by ClearKey', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + sessionTypes: ['persistent-license'], + persistentState: 'optional', + } + ], + shouldPass: false, + }, + { + name: 'Persistent-usage-record should not be supported by ClearKey', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4'}], + sessionTypes: ['persistent-usage-record'], + persistentState: 'optional', + } + ], + shouldPass: false, + }, + { + name: 'MP4 audio container', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4'}], + }, + shouldPass: true, + }, + { + name: 'MP4 audio container with AAC-LC', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}], + }, + shouldPass: true, + }, + { + name: 'MP4 audio container with invalid codecs', + options: [ + { + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4; codecs="bogus"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 audio container with mp3 is unsupported', + options: [ + { + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4; codecs="mp3"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 video container type with an mp3 codec is unsupported', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="mp3"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 audio container type with a video codec is unsupported', + options: [ + { + initDataTypes: ['cenc'], + audioCapabilities: [{contentType: 'audio/mp4; codecs="avc1.42E01E"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 video container with constrained baseline h.264', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}], + }, + shouldPass: true, + }, + { + name: 'MP4 video container with invalid codecs', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="bogus"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 video container with both audio and video codec type in videoType', + options: [ + { + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E,mp4a.40.2"'}], + } + ], + shouldPass: false, + }, + { + name: 'MP4 audio and video type both specified', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}], + audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['cenc'], + videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}], + audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}], + }, + shouldPass: true, + }, + { + name: 'Basic WebM video', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm'}], + }, + shouldPass: true, + }, + { + name: 'Basic WebM audio', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + audioCapabilities: [{contentType: 'audio/webm'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + audioCapabilities: [{contentType: 'audio/webm'}], + }, + shouldPass: true, + }, + { + name: 'Webm with Vorbis audio and VP8 video.', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm;codecs="vp8"'}], + audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm;codecs="vp8"'}], + audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}], + }, + shouldPass: true, + }, + { + name: 'Webm with Vorbis audio and VP9 video.', + options: [ + { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm;codecs="vp9"'}], + audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}], + } + ], + expectedConfig: { + label: SUPPORTED_LABEL, + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm;codecs="vp9"'}], + audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}], + }, + shouldPass: true, + }, + { + name: 'Webm with bogus video.', + options: [ + { + initDataTypes: ['webm'], + videoCapabilities: [{contentType: 'video/webm;codecs="bogus"'}], + } + ], + shouldPass: false, + }, +]; + +function beginTest() { + Promise.all(tests.map(Test)).then(function() { SimpleTest.finish(); }); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html b/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html new file mode 100644 index 0000000000..6bafb768c1 --- /dev/null +++ b/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html @@ -0,0 +1,209 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions access can be gated by application</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// These test cases should be used to make a request to +// requestMediaKeySystemAccess and have the following members: +// name: a name describing the test. +// askAppApproval: used to set prefs such so that Gecko will ask for app +// approval for EME if true, or not if false. +// appApproves: used to set prefs to simulate app approval of permission +// request, true if the app approves the request, false if not. +// expectedKeySystemAccess: true if we expect to be granted key system access, +// false if not. +const testCases = [ + { + name: "Don't check for app approval", + askAppApproval: false, + expectedKeySystemAccess: true, + }, + { + name: "Check for app approval and app denies request", + askAppApproval: true, + appApproves: false, + expectedKeySystemAccess: false, + }, + { + name: "Check for app approval and app allows request", + askAppApproval: true, + appApproves: true, + expectedKeySystemAccess: true, + }, +]; + +// Options for requestMediaKeySystemAccess that are expected to work. +const options = [{ + initDataTypes: ['webm'], + audioCapabilities: [ + { contentType: 'audio/webm; codecs="opus"' }, + ], + videoCapabilities: [ + { contentType: 'video/webm; codecs="vp8"' } + ] +}]; + +async function setTestPrefs({askAppApproval, appApproves}) { + if (!askAppApproval) { + // Test doesn't want app approval, set pref so we don't ask and unset prefs + // used to determine permission response as we don't need them. + await SpecialPowers.pushPrefEnv({ + set: [["media.eme.require-app-approval", false]], + clear: [ + ["media.eme.require-app-approval.prompt.testing"], + ["media.eme.require-app-approval.prompt.testing.allow"], + ] + }); + return; + } + + // Test wants app approval, and will approve deny requests per appApproces + // value, set prefs accordingly. + await SpecialPowers.pushPrefEnv({ + set: [ + ["media.eme.require-app-approval", true], + ["media.eme.require-app-approval.prompt.testing", true], + ["media.eme.require-app-approval.prompt.testing.allow", appApproves], + ], + }); +} + +// Run a test case that makes a single requestMediaKeySystemAccess call. The +// outcome of such a test run should depend on the test case's setting of +// preferences controlling the eme app approval. +async function testSingleRequest(testCase) { + await setTestPrefs(testCase); + + try { + await navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options); + ok(testCase.expectedKeySystemAccess, + `testSingleRequest ${testCase.name}: allowed media key system access.`); + } catch(e) { + is(e.name, + "NotSupportedError", + "Should get NotSupportedError when request is blocked."); + is(e.message, + "The application embedding this user agent has blocked MediaKeySystemAccess", + "Should get blocked error message."); + ok(!testCase.expectedKeySystemAccess, + `testSingleRequest ${testCase.name}: denied media key system access.`); + } +} + +// Run a test case that, but using invalid arguments for +// requestMediaKeySystemAccess. Because we expect the args to be checked +// before requesting app approval, this test ensures that we always fail when +// using bad args, regardless of the app approval prefs set. +async function testRequestWithInvalidArgs(testCase) { + const badOptions = [{ + initDataTypes: ['badType'], + audioCapabilities: [ + { contentType: 'audio/webm; codecs="notACodec"' }, + ], + videoCapabilities: [ + { contentType: 'video/webm; codecs="notACodec"' } + ] + }]; + + await setTestPrefs(testCase); + + // Check that calls with a bad key system fail. + try { + await navigator.requestMediaKeySystemAccess("BadKeySystem", options); + ok(false, + `testRequestWithInvalidArgs ${testCase.name}: should not get access when using bad key system.`); + } catch(e) { + is(e.name, + "NotSupportedError", + "Should get NotSupportedError using bad key system."); + is(e.message, + "Key system is unsupported", + "Should get not supported key system error message."); + } + + // Check that calls with the bad options fail. + try { + await navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, badOptions); + ok(false, + `testRequestWithInvalidArgs ${testCase.name}: should not get access when using bad options.`); + } catch(e) { + is(e.name, + "NotSupportedError", + "Should get NotSupportedError using bad options."); + is(e.message, + "Key system configuration is not supported", + "Should get not supported config error message."); + } +} + +// Run a test case and make multiple requests with the same case. Check that +// all requests are resolved with the expected outcome. +async function testMultipleRequests(testCase) { + // Number of requests to concurrently make. + const NUM_REQUESTS = 5; + + await setTestPrefs(testCase); + + let resolveHandler = () => { + ok(testCase.expectedKeySystemAccess, + `testMultipleRequests ${testCase.name}: allowed media key system access.`); + } + + let rejectHandler = e => { + is(e.name, + "NotSupportedError", + "Should get NotSupportedError when request is blocked."); + is(e.message, + "The application embedding this user agent has blocked MediaKeySystemAccess", + "Should get blocked error message."); + ok(!testCase.expectedKeySystemAccess, + `testMultipleRequests ${testCase.name}: denied media key system access.`); + } + + let accessPromises = []; + for(let i = 0; i < NUM_REQUESTS; i++) { + // Request access then chain to our resolve and reject handlers. The + // handlers assert test state then resolve the promise chain. Ensuring the + // chain is always resolved allows us to correctly await all outstanding + // requests -- otherwise rejects short circuit the Promise.all call below. + let accessPromise = navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options) + .then(resolveHandler, rejectHandler); + accessPromises.push(accessPromise); + } + // Wait for all promises to be resolved. If not, we'll time out. Because + // our reject handler chains back into a resolved promise, this should wait + // for all requests to be serviced, even when requestMediaKeySystemAccess's + // promise is rejected. + await Promise.all(accessPromises); +} + +async function beginTest() { + // The tests rely on prefs being set, so run them in sequence. If we run in + // parallel the tests break each other by overriding prefs. + for (const testCase of testCases) { + await testSingleRequest(testCase); + } + for (const testCase of testCases) { + await testRequestWithInvalidArgs(testCase); + } + for (const testCase of testCases) { + await testMultipleRequests(testCase); + } + SimpleTest.finish(); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_request_notifications.html b/dom/media/test/test_eme_request_notifications.html new file mode 100644 index 0000000000..b3c62c9cf6 --- /dev/null +++ b/dom/media/test/test_eme_request_notifications.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function SetPrefs(prefs) { + return SpecialPowers.pushPrefEnv({"set": prefs}); +} + +function observe() { + return new Promise(function(resolve, reject) { + var observer = function(subject, topic, data) { + SpecialPowers.Services.obs.removeObserver(observer, "mediakeys-request"); + resolve(JSON.parse(data).status); + }; + SpecialPowers.Services.obs.addObserver(observer, "mediakeys-request"); + }); +} + +function Test(test) { + var p = test.prefs ? SetPrefs(test.prefs) : Promise.resolve(); + var name = "'" + test.keySystem + "'"; + + var res = observe().then((status) => { + is(status, test.expectedStatus, name + " expected status"); + }); + + p.then(() => navigator.requestMediaKeySystemAccess(test.keySystem, gCencMediaKeySystemConfig)) + .then((keySystemAccess) => keySystemAccess.createMediaKeys()); + + return res; +} + +var tests = [ + { + keySystem: CLEARKEY_KEYSYSTEM, + expectedStatus: 'cdm-created', + prefs: [["media.eme.enabled", false]] + }, + { + keySystem: "com.widevine.alpha", + expectedStatus: 'api-disabled', + prefs: [["media.eme.enabled", false]] + }, + { + keySystem: "com.widevine.alpha", + expectedStatus: 'cdm-disabled', + prefs: [["media.eme.enabled", true], ["media.gmp-widevinecdm.enabled", false]] + }, + { + keySystem: "com.widevine.alpha", + expectedStatus: 'cdm-not-installed', + prefs: [["media.eme.enabled", true], ["media.gmp-widevinecdm.enabled", true]] + }, + { + keySystem: CLEARKEY_KEYSYSTEM, + expectedStatus: 'cdm-created', + prefs: [["media.eme.enabled", true]] + } +]; + +SetupEMEPref(function() { + tests.reduce(function(p,c,i,array) { + return p.then(function() { return Test(c); }); + }, Promise.resolve()).then(SimpleTest.finish); +}); + + +SimpleTest.waitForExplicitFinish(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_sample_groups_playback.html b/dom/media/test/test_eme_sample_groups_playback.html new file mode 100644 index 0000000000..be56bff21d --- /dev/null +++ b/dom/media/test/test_eme_sample_groups_playback.html @@ -0,0 +1,130 @@ +<!DOCTYPE HTML> +<html> + +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> + +<body> + <video controls id="video"></video> + <pre id="test"> + <script class="testbody" type="text/javascript"> + + // Tests that files with a default key and a seperate sample keyids in the + // sgpd box play correctly (if the keyid from the sgpd box is not parsed + // or assigned to the sample we will wait indefinitely for the default + // key). + + SimpleTest.waitForExplicitFinish(); + + // Test files for samples encrypted with different media keys. + var gEMESampleGoupTests = [ + { + name:"video with 4 keys in sgpd (sbgp in traf sgpd in stbl)", + track: { + name:"video", + type:"video/mp4; codecs=\"avc1.64000d\"", + fragments:[ "sample-encrypted-sgpdstbl-sbgptraf.mp4" + ] + }, + keys: { + // "keyid" : "key" + "279926496a7f5d25da69f2b3b2799a7f": "5544694d47473326622665665a396b36", + "597669572e55547e656b56586e2f6f68": "7959493a764556786527517849756635", + "205b2b293a342f3d3268293e6f6f4e29": "3a4f3674376d6c48675a273464447b40", + "32783e367c2e4d4d6b46467b3e6b5478": "3e213f6d45584f51713d534f4b417855", + }, + sessionType:"temporary", + sessionCount:1, + duration:2, + }, + ], + test = gEMESampleGoupTests[0]; + + var video = document.getElementById("video"); + video.addEventListener("encrypted", () => { + Log(test.name, "Recieved encrypted event"); + }); + + video.addEventListener("waitingforkey", () => { + Log(test.name, "waitingforkey"); + ok(false, test.name + " Video is waitingforkey, indicating that the samples are not being assigned the correct id from the sgpd box!"); + SimpleTest.finish(); + }); + + function LoadEME() { + var options = [{ + initDataType: "cenc", + videoType: test.track.type, + }]; + + return navigator.requestMediaKeySystemAccess("org.w3.clearkey", options) + .then((keySystemAccess) => { + return keySystemAccess.createMediaKeys(); + }, bail("Failed to request key system access.")) + + .then((mediaKeys) => { + video.setMediaKeys(mediaKeys); + + var session = mediaKeys.createSession(); + once(session, "message", (ev) => { + is(ev.messageType, "license-request", "Expected a license-request"); + session.update(GenerateClearKeyLicense(ev.message, test.keys)); + }); + + var json = JSON.stringify({ + "kids":Object.keys(test.keys).map(HexToBase64) + }); + var request = new TextEncoder().encode(json); + session.generateRequest("keyids", request) + .then(e => { + Log(test.name, "Request license success"); + }, reason => { + Log("Request license failed! " + reason); + }); + }); + } + + function DownloadMedia(url, type, mediaSource) { + return new Promise((resolve, reject) => { + var sourceBuffer = mediaSource.addSourceBuffer(type); + fetchWithXHR(url, (response) => { + once(sourceBuffer, "updateend", resolve); + sourceBuffer.appendBuffer(new Uint8Array(response)); + }); + }); + } + + function LoadMSE() { + // Only set the source of the video and download the tracks after we + // have set the license keys, so we don't hit the waitingforkey event + // unless samples are being incorrectly assigned the default key + // (and we can safely fail). + LoadEME() + .then(() => { + var ms = new MediaSource(); + video.src = URL.createObjectURL(ms); + + once(ms, "sourceopen", () => { + Promise.all(test.track.fragments.map(fragment => DownloadMedia(fragment, test.track.type, ms))) + .then(() => { + ms.endOfStream(); + video.play(); + }); + }); + + once(video, "ended", SimpleTest.finish); + }); + } + + SetupEMEPref(LoadMSE); + + </script> + </pre> +</body> + +</html>
\ No newline at end of file diff --git a/dom/media/test/test_eme_session_callable_value.html b/dom/media/test/test_eme_session_callable_value.html new file mode 100644 index 0000000000..24413a025d --- /dev/null +++ b/dom/media/test/test_eme_session_callable_value.html @@ -0,0 +1,35 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +function Test() { + navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig) + .then(access => access.createMediaKeys()) + .then(mediaKeys => { + var initData = (new TextEncoder()).encode( 'this is an invalid license, and that is ok'); + var s = mediaKeys.createSession("temporary"); + s.generateRequest("cenc", initData); // ignore result. + // "update()" call should fail, because MediaKeySession is "not callable" + // yet, since CDM won't have had a chance to set the sessionId on MediaKeySession. + return s.update(initData); + }) + .then(()=>{ok(false, "An exception should be thrown; MediaKeySession should be not callable."); SimpleTest.finish();}, + ()=>{ok(true, "We expect this to fail; MediaKeySession should be not callable."); SimpleTest.finish();}); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(Test); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html b/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html new file mode 100644 index 0000000000..bb36b23a2e --- /dev/null +++ b/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html @@ -0,0 +1,38 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +function beginTest() { + var video = document.createElement("video"); + + navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig) + .then(function(keySystemAccess) { + return keySystemAccess.createMediaKeys(); + }) + .then(mediaKeys => { + return video.setMediaKeys(mediaKeys); + }) + .then(() => { + var ms = new MediaSource(); + ms.addEventListener("sourceopen", ()=>{ok(true, "MediaSource should open"); SimpleTest.finish();}); + video.addEventListener("error", ()=>{ok(false, "Shouldn't error."); SimpleTest.finish();}); + video.src = URL.createObjectURL(ms); + }); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_stream_capture_blocked_case1.html b/dom/media/test/test_eme_stream_capture_blocked_case1.html new file mode 100644 index 0000000000..228baee13a --- /dev/null +++ b/dom/media/test/test_eme_stream_capture_blocked_case1.html @@ -0,0 +1,64 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) +{ + // Case 1. setting MediaKeys on an element captured by MediaElementSource should fail. + var case1token = token + "_case1"; + let v1 = document.createElement("video"); + + function setMediaKeys() { + let p = new EMEPromise; + CreateMediaKeys(v1, test, case1token) + .then(mediaKeys => { + v1.setMediaKeys(mediaKeys) + .then(() => { + p.reject(`${case1token} setMediaKeys shouldn't succeed.`); + }, () => { + ok(true, TimeStamp(case1token) + " setMediaKeys failed as expected."); + p.resolve(); + }) + }, p.reject); + return p.promise; + } + + var context = new AudioContext(); + context.createMediaElementSource(v1); + v1.addEventListener("loadeddata", function(ev) { + ok(false, TimeStamp(case1token) + " should never reach loadeddata, as setMediaKeys should fail"); + }); + + manager.started(case1token); + + Promise.all([ + LoadTest(test, v1, case1token), + setMediaKeys()]) + .catch(reason => ok(false, reason)) + .then(() => { + CleanUpMedia(v1); + manager.finished(case1token); + }); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_stream_capture_blocked_case2.html b/dom/media/test/test_eme_stream_capture_blocked_case2.html new file mode 100644 index 0000000000..8fb8f5431b --- /dev/null +++ b/dom/media/test/test_eme_stream_capture_blocked_case2.html @@ -0,0 +1,57 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) +{ + // Case 2. creating a MediaElementSource on a media element should always succeed + // (no matter whether it's restricted content or not), and + var p1 = new EMEPromise; + var case2token = token + "_case2"; + let v2 = document.createElement("video"); + + v2.addEventListener("loadeddata", function(ev) { + ok(true, case2token + " should reach loadeddata"); + var threw = false; + try { + var context = new AudioContext(); + context.createMediaElementSource(v2); + } catch (e) { + threw = true; + } + ok(!threw, "Should always work when creating a MediaElementSource."); + p1.resolve(); + }); + + manager.started(case2token); + let p2 = SetupEME(v2, test, case2token); + + Promise.all([p1.promise, p2]) + .catch(reason => ok(false, reason)) + .then(() => { + CleanUpMedia(v2); + manager.finished(case2token); + }); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_stream_capture_blocked_case3.html b/dom/media/test/test_eme_stream_capture_blocked_case3.html new file mode 100644 index 0000000000..5703328d40 --- /dev/null +++ b/dom/media/test/test_eme_stream_capture_blocked_case3.html @@ -0,0 +1,55 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) +{ + // Case 3. capturing a media element with mozCaptureStream that has a MediaKeys should fail. + var p1 = new EMEPromise; + var case3token = token + "_case3"; + let v3 = document.createElement("video"); + + v3.addEventListener("loadeddata", function(ev) { + ok(true, TimeStamp(case3token) + " should reach loadeddata"); + var threw = false; + try { + v3.mozCaptureStreamUntilEnded(); + } catch (e) { + threw = true; + } + ok(threw, TimeStamp(case3token) + " Should throw an error calling mozCaptureStreamUntilEnded an EME video."); + p1.resolve(); + }); + + manager.started(case3token); + let p2 = SetupEME(v3, test, case3token); + + Promise.all([p1.promise, p2]) + .catch(reason => ok(false, reason)) + .then(() => { + CleanUpMedia(v3); + manager.finished(case3token); + }); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_unsetMediaKeys_then_capture.html b/dom/media/test/test_eme_unsetMediaKeys_then_capture.html new file mode 100644 index 0000000000..c3691fe71c --- /dev/null +++ b/dom/media/test/test_eme_unsetMediaKeys_then_capture.html @@ -0,0 +1,113 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +// Test that if we can capture a video frame while playing clear content after +// removing the MediaKeys object which was used for a previous encrypted content +// playback on the same video element +function startTest(test, token) +{ + manager.started(token); + var sessions = []; + function onSessionCreated(session) { + sessions.push(session); + } + + function closeSessions() { + let p = new EMEPromise; + Promise.all(sessions.map(s => s.close())) + .then(p.resolve, p.reject); + return p.promise; + } + + let v = document.createElement("video"); + document.body.appendChild(v); + + let finish = new EMEPromise; + + function onVideoEnded(ev) { + ok(true, TimeStamp(token) + " (ENCRYPTED) content playback ended."); + + function playClearVideo() { + var p1 = once(v, 'loadeddata', (e) => { + ok(true, TimeStamp(token) + " Receiving event 'loadeddata' for (CLEAR) content."); + let canvasElem = document.createElement('canvas'); + document.body.appendChild(canvasElem); + let ctx2d = canvasElem.getContext('2d'); + + var gotTypeError = false; + try { + ctx2d.drawImage(v, 0, 0); + } catch (ex) { + if (ex instanceof TypeError) { + gotTypeError = true; + } + } + ok(!gotTypeError, TimeStamp(token) + " Canvas2D context drawImage succeed.") + }); + v.src = 'bipbop_225w_175kbps.mp4'; + v.play(); + Promise.all([p1]).then(() => { + manager.finished(token); + }, () => { + ok(false, TimeStamp(token) + " Something wrong."); + manager.finished(token); + }); + } + + closeSessions() + .then(() => { + v.setMediaKeys(null) + .then(() => { + ok(true, TimeStamp(token) + " Setting MediaKeys to null."); + playClearVideo(); + }, () => { + ok(false, TimeStamp(token) + " Setting MediaKeys to null."); + });; + }); + } + + once(v, "ended", onVideoEnded); + + // Create a MediaKeys object and set to HTMLMediaElement then start the playback. + Promise.all([ + LoadInitData(v, test, token), + CreateAndSetMediaKeys(v, test, token), + LoadTest(test, v, token)]) + .then(values => { + let initData = values[0]; + v.play(); + initData.map(ev => { + let session = v.mediaKeys.createSession(); + onSessionCreated(session); + MakeRequest(test, token, ev, session); + }); + }) + .then(() => { + return finish.promise; + }) + .catch(reason => ok(false, reason)) +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_eme_waitingforkey.html b/dom/media/test/test_eme_waitingforkey.html new file mode 100644 index 0000000000..2506c56fd1 --- /dev/null +++ b/dom/media/test/test_eme_waitingforkey.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test Encrypted Media Extensions</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +/* import-globals-from eme.js */ +var manager = new MediaTestManager; + +function startTest(test, token) +{ + manager.started(token); + + let v = document.createElement("video"); + document.body.appendChild(v); + + var gotWaitingForKey = 0; + var gotOnwaitingforkey = 0; + + let waitForKey = new EMEPromise; + v.addEventListener("waitingforkey", function() { + gotWaitingForKey += 1; + waitForKey.resolve(); + }); + + v.onwaitingforkey = function() { + gotOnwaitingforkey += 1; + }; + + v.addEventListener("loadedmetadata", function() { + ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v), + TimeStamp(token) + " isEncrypted should be true"); + is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content"); + }); + + let finish = new EMEPromise; + v.addEventListener("ended", function() { + ok(true, TimeStamp(token) + " got ended event"); + // We expect only one waitingForKey as we delay until all sessions are ready. + // I.e. one waitingForKey should be fired, after which, we process all sessions, + // so it should not be possible to be blocked by a key after that point. + ok(gotWaitingForKey == 1, "Expected number 1 wait, got: " + gotWaitingForKey); + ok(gotOnwaitingforkey == gotWaitingForKey, "Should have as many event listener calls as event handler calls, got: " + gotOnwaitingforkey); + + finish.resolve(); + }); + + Promise.all([ + LoadInitData(v, test, token), + CreateAndSetMediaKeys(v, test, token), + LoadTest(test, v, token), + waitForKey.promise]) + .then(values => { + let initData = values[0]; + return ProcessInitData(v, test, token, initData); + }) + .then(sessions => { + Log(token, "Updated all sessions, loading complete -> Play"); + v.play(); + finish.promise.then(() => CloseSessions(v, sessions)); + return finish.promise; + }) + .catch(reason => ok(false, reason)) + .then(() => manager.finished(token)); +} + +function beginTest() { + manager.runTests(gEMETests, startTest); +} + +SimpleTest.waitForExplicitFinish(); +SetupEMEPref(beginTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_empty_resource.html b/dom/media/test/test_empty_resource.html new file mode 100644 index 0000000000..8f65bb503d --- /dev/null +++ b/dom/media/test/test_empty_resource.html @@ -0,0 +1,58 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=1094549 +--> +<head> + <title>Test for Bug 1094549</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1094549">Mozilla Bug 1094549</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Shorter timeout for this test should finish soon. +SimpleTest.requestLongerTimeout(0.3); + +function finish(v) { + isnot(v.error, null, "should've got an error event"); + SimpleTest.finish(); +} + +function onload() { + info("iframe loaded"); + var v = SpecialPowers.wrap(document.body.getElementsByTagName("iframe")[0]) + .contentDocument.body.getElementsByTagName("video")[0]; + + // Got 'error' as expected, finish the test. + if (v.error) { + finish(v); + return; + } + + // Otherwise, wait for it. + v.onerror = function() { + finish(v); + } +} + +SimpleTest.waitForExplicitFinish(); +var f = document.createElement("iframe"); +// Assign a resource file with zero length and expect the error event from +// the video element since decoding metadata will fail. +f.src = "data:video/webm,"; +f.addEventListener("load", onload); +document.body.appendChild(f); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_error_in_video_document.html b/dom/media/test/test_error_in_video_document.html new file mode 100644 index 0000000000..e376ea95e3 --- /dev/null +++ b/dom/media/test/test_error_in_video_document.html @@ -0,0 +1,59 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=604067 +--> +<head> + <title>Test for Bug 604067</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=604067">Mozilla Bug 604067</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 604067 **/ + +function documentVideo() { + return document.body.getElementsByTagName("iframe")[0] + .contentDocument.body.getElementsByTagName("video")[0]; +} + +function check() { + var v = documentVideo(); + + // Debug info for Bug 608634 + ok(true, "iframe src=" + document.body.getElementsByTagName("iframe")[0].src); + is(v.readyState, v.HAVE_NOTHING, "Ready state"); + + isnot(v.error, null, "Error object"); + is(v.networkState, v.NETWORK_NO_SOURCE, "Network state"); + is(v.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, "Expected media not supported error"); + SimpleTest.finish(); +} + +// Find an error test that we'd think we should be able to play (if it +// wasn't already known to fail). +var t = getPlayableVideo(gErrorTests); +if (!t) { + todo(false, "No types supported"); +} else { + SimpleTest.waitForExplicitFinish(); + + var f = document.createElement("iframe"); + f.src = t.name; + f.addEventListener("load", check); + document.body.appendChild(f); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_error_on_404.html b/dom/media/test/test_error_on_404.html new file mode 100644 index 0000000000..94f7cc2ea7 --- /dev/null +++ b/dom/media/test/test_error_on_404.html @@ -0,0 +1,84 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=476731 +--> +<head> + <title>Test for Bug 476731</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=476731">Mozilla Bug </a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> +<pre id="test"> +<script type="application/javascript"> +/** Test for Bug 476731 **/ + +var videos = []; + +function FinishedLoads() { + if (videos.length == 0) + return false; + for (var i=0; i<videos.length; ++i) { + if (videos[i]._loadedData) { + // A video loadeddata, it should have 404'd. Signal the end of the test + // so the error will be reported. + return true; + } + if (!videos[i]._loadError) { + return false; + } + } + return true; +} + +function loadError(evt) { + var v = evt.target; + is(v._loadError, false, "Shouldn't receive multiple error events for " + v.src); + v._loadError = true; + is(v._loadStart, true, "Receive loadstart for " + v.src); + is(v._loadedData, false, "Shouldn't receive loadeddata for " + v.src); + if (FinishedLoads(videos)) + SimpleTest.finish(); +} + +function loadedData(evt) { + evt.target._loadedData = true; +} + +function loadStart(evt) { + evt.target._loadStart = true; +} + +// Create all video objects. +for (var i=0; i<g404Tests.length; ++i) { + var v = document.createElement("video"); + v.preload = "metadata"; + var test = g404Tests[i]; + if (!v.canPlayType(test.type)) { + continue; + } + v.src = test.name; + v._loadedData = false; + v._loadStart = false; + v._loadError = false; + v.addEventListener("error", loadError); + v.addEventListener("loadstart", loadStart); + v.addEventListener("loadeddata", loadedData); + document.body.appendChild(v); // Will start load. + videos.push(v); +} + +if (videos.length == 0) { + todo(false, "No types supported"); +} else { + SimpleTest.waitForExplicitFinish(); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_fastSeek-forwards.html b/dom/media/test/test_fastSeek-forwards.html new file mode 100644 index 0000000000..1e50021b13 --- /dev/null +++ b/dom/media/test/test_fastSeek-forwards.html @@ -0,0 +1,77 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=1022913 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug 1022913</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1022913">Mozilla Bug 1022913</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> + <script type="application/javascript"> + + // Test that if we're doing a fastSeek() forwards that we don't end up + // seeking backwards. This can happen if the keyframe before the seek + // target is before the current playback position. We'd prefer to seek to + // the keyframe after the seek target in this case, but we don't implement + // this yet (bug 1026330). + var manager = new MediaTestManager; + + var onSecondSeekComplete = function(event) { + var v = event.target; + v.removeEventListener("seeked", onSecondSeekComplete); + ok(v.currentTime >= v.firstSeekTarget, v.name + " seek never go backwards. time=" + v.currentTime + " firstSeekTarget=" + v.firstSeekTarget + " secondSeekTarget=" + v.secondSeekTarget); + manager.finished(v.token); + removeNodeAndSource(v); + }; + + var onFirstSeekComplete = function(event) { + var v = event.target; + v.removeEventListener("seeked", onFirstSeekComplete); + // Seek to 75% of the way between the start and the first keyframe + // using fastSeek. We then test that the currentTime doesn't drop back + // to the previous keyframe, currentTime should go forwards. + v.addEventListener("seeked", onSecondSeekComplete); + v.secondSeekTarget = v.keyframes[1] * 0.75; + v.fastSeek(v.secondSeekTarget); + } + + var onLoadedMetadata = function(event) { + // Seek to the mid-point between the start and the first keyframe. + var v = event.target; + v.removeEventListener("loadedmetadata", onLoadedMetadata); + v.addEventListener("seeked", onFirstSeekComplete); + v.firstSeekTarget = v.keyframes[1] * 0.5; + v.currentTime = v.firstSeekTarget; + } + + function startTest(test, token) { + manager.started(token); + let v = document.createElement("video"); + v.src = test.name; + v.name = test.name; + v.preload = "metadata"; + v.token = token; + v.target = 0; + v.keyframes = test.keyframes; + v.keyframeIndex = 0; + ok(v.keyframes.length >= 2, v.name + " - video should have at least two sync points"); + v.addEventListener("loadedmetadata", onLoadedMetadata); + document.body.appendChild(v); + } + + manager.runTests(gFastSeekTests, startTest); + + </script> +</body> +</html> diff --git a/dom/media/test/test_fastSeek.html b/dom/media/test/test_fastSeek.html new file mode 100644 index 0000000000..e3078c6c4e --- /dev/null +++ b/dom/media/test/test_fastSeek.html @@ -0,0 +1,88 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=778077 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug 778077</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=778077">Mozilla Bug 778077</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> + <script type="application/javascript"> + + /** Test for Bug 778077 - HTMLMediaElement.fastSeek() **/ + // Iterate through a list of keyframe timestamps, and seek to + // halfway between the keyframe and the keyframe after it. + var manager = new MediaTestManager; + + function doSeek(v) { + // fastSeek to half way between this keyframe and the next, or if this is the last + // keyframe seek to halfway between this keyframe and the end of media. + var nextKeyFrame = (v.keyframeIndex + 1) < v.keyframes.length ? v.keyframes[v.keyframeIndex + 1] : v.duration; + v.target = (v.keyframes[v.keyframeIndex] + nextKeyFrame) / 2; + v.fastSeek(v.target); + ok(Math.abs(v.currentTime - v.target) < 0.01, + v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime + ") should be close to seek target initially"); + } + + function onloadedmetadata(event) { + var v = event.target; + doSeek(v); + } + + function onseeked(event) { + var v = event.target; + var keyframe = v.keyframes[v.keyframeIndex]; + + // Check that the current time ended up roughly after the keyframe. + // We must be a bit fuzzy here, as the decoder backend may actually + // seek to the audio sample prior to the keyframe. + ok(v.currentTime >= keyframe - 0.05, + v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime + + ") should be end up roughly after keyframe (" + keyframe + ") after fastSeek"); + + ok(v.currentTime <= v.target, + v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime + + ") should be end up less than target after fastSeek"); + + v.keyframeIndex++ + if (v.keyframeIndex == v.keyframes.length) { + v.src = ""; + v.remove(); + manager.finished(v.token); + } else { + doSeek(v); + } + } + + function startTest(test, token) { + manager.started(token); + let v = document.createElement("video"); + v.src = test.name; + v.name = test.name; + v.preload = "metadata"; + v.token = token; + v.target = 0; + v.keyframes = test.keyframes; + v.keyframeIndex = 0; + ok(v.keyframes.length > 0, v.name + " - video should have at least one sync point"); + v.addEventListener("loadedmetadata", onloadedmetadata); + v.addEventListener("seeked", onseeked); + document.body.appendChild(v); + } + + manager.runTests(gFastSeekTests, startTest); + + </script> +</body> +</html> diff --git a/dom/media/test/test_fragment_noplay.html b/dom/media/test/test_fragment_noplay.html new file mode 100644 index 0000000000..6a1119f342 --- /dev/null +++ b/dom/media/test/test_fragment_noplay.html @@ -0,0 +1,127 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: fragment tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="fragment_noplay.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +// Fragment parameters to try +var gFragmentParams = [ + // W3C Media fragment tests + // http://www.w3.org/2008/WebVideo/Fragments/TC/ua-test-cases + { fragment: "#t=banana", start: null, end: null }, // TC0027-UA + { fragment: "#t=3,banana", start: null, end: null }, // TC0028-UA + { fragment: "#t=banana,7", start: null, end: null }, // TC0029-UA + { fragment: "#t='3'", start: null, end: null }, // TC0030-UA + { fragment: "#t=3-7", start: null, end: null }, // TC0031-UA + { fragment: "#t=3:7", start: null, end: null }, // TC0032-UA + { fragment: "#t=3,7,9", start: null, end: null }, // TC0033-UA + { fragment: "#t%3D3", start: null, end: null }, // TC0034-UA + { fragment: "#%74=3", start: 3, end: null }, // TC0035-UA + { fragment: "#t=%33", start: 3, end: null }, // TC0036-UA + { fragment: "#t=3%2C7", start: 3, end: 7 }, // TC0037-UA + { fragment: "#t=%6Ept:3", start: 3, end: null }, // TC0038-UA + { fragment: "#t=npt%3A3", start: 3, end: null }, // TC0039-UA + { fragment: "#t=-1,3", start: null, end: null }, // TC0044-UA + { fragment: "#t=3&", start: 3, end: null }, // TC0051-UA + { fragment: "#u=12&t=3", start: 3, end: null }, // TC0052-UA + { fragment: "#t=foo:7&t=npt:3", start: 3, end: null }, // TC0053-UA + { fragment: "#&&=&=tom&jerry=&t=3&t=meow:0#", start: 3, end: null }, // TC0054-UA + { fragment: "#t=7&t=3", start: 3, end: null }, // TC0055-UA + { fragment: "#T=3,7", start: null, end: null }, // TC0058-UA + { fragment: "#t=", start: null, end: null }, // TC0061-UA + { fragment: "#t=.", start: null, end: null }, // TC0062-UA + { fragment: "#t=.0", start: null, end: null }, // TC0063-UA + { fragment: "#t=0s", start: null, end: null }, // TC0064-UA + { fragment: "#t=,0s", start: null, end: null }, // TC0065-UA + { fragment: "#t=0s,0s", start: null, end: null }, // TC0066-UA + { fragment: "#t=00:00:00s", start: null, end: null }, // TC0067-UA + { fragment: "#t=s", start: null, end: null }, // TC0068-UA + { fragment: "#t=npt:", start: null, end: null }, // TC0069-UA + { fragment: "#t=1e-1:", start: null, end: null }, // TC0070-UA + { fragment: "#t=00:00:01.1e-1", start: null, end: null }, // TC0071-UA + { fragment: "#t=3.", start: 3, end: null }, // TC0072-UA + { fragment: "#t=0:0:0", start: null, end: null }, // TC0073-UA + { fragment: "#t=0:00:60", start: null, end: null }, // TC0074-UA + { fragment: "#t=0:01:60", start: null, end: null }, // TC0075-UA + { fragment: "#t=0:60:00", start: null, end: null }, // TC0076-UA + { fragment: "#t=0:000:000", start: null, end: null }, // TC0077-UA + { fragment: "#t=00:00:03,00:00:07", start: 3, end: 7 }, // TC0078-UA + { fragment: "#t=3,00:00:07", start: 3, end: 7 }, // TC0079-UA + { fragment: "#t=00:00.", start: null, end: null }, // TC0080-UA + { fragment: "#t=0:00:00.", start: null, end: null }, // TC0081-UA + { fragment: "#t=0:00:10e-1", start: null, end: null }, // TC0082-UA + { fragment: "#t=0:00:60.000", start: null, end: null }, // TC0083-UA + { fragment: "#t=0:60:00.000", start: null, end: null }, // TC0084-UA + { fragment: "#t=3,7&t=foo", start: 3, end: 7 }, // TC0085-UA + { fragment: "#foo&t=3,7", start: 3, end: 7 }, // TC0086-UA + { fragment: "#t=3,7&foo", start: 3, end: 7 }, // TC0087-UA + { fragment: "#t=3,7&&", start: 3, end: 7 }, // TC0088-UA + { fragment: "#&t=3,7", start: 3, end: 7 }, // TC0089-UA + { fragment: "#&&t=3,7", start: 3, end: 7 }, // TC0090-UA + { fragment: "#&t=3,7&", start: 3, end: 7 }, // TC0091-UA + { fragment: "#t%3d10", start: null, end: null }, // TC0092-UA + { fragment: "#t=10%26", start: null, end: null }, // TC0093-UA + { fragment: "#&t=3,7,", start: null, end: null } // TC0094-UA +]; + +function createTestArray() { + var tests = []; + var tmpVid = document.createElement("video"); + + for (var testNum=0; testNum<gFragmentTests.length; testNum++) { + var test = gFragmentTests[testNum]; + if (!tmpVid.canPlayType(test.type)) { + continue; + } + + for (var fragNum=0; fragNum<gFragmentParams.length; fragNum++) { + var p = gFragmentParams[fragNum]; + var t = {}; + t.name = test.name + p.fragment; + t.type = test.type; + t.duration = test.duration; + t.start = p.start; + t.end = p.end; + tests.push(t); + } + } + return tests; +} + +function startTest(test, token) { + var video = document.createElement('video'); + manager.started(token); + video.preload = "metadata"; + video.src = test.name; + video.token = token; + video.controls = true; + document.body.appendChild(video); + var name = test.name + " fragment test"; + var localIs = function(n) { return function(a, b, msg) { + is(a, b, n + ": " + msg); + }}(name); + var localOk = function(n) { return function(a, msg) { + ok(a, n + ": " + msg); + }}(name); + var localFinish = function(v, m) { return function() { + removeNodeAndSource(v); + m.finished(v.token); + }}(video, manager); + window.test_fragment_noplay(video, test.start, test.end, localIs, localOk, localFinish); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_fragment_play.html b/dom/media/test/test_fragment_play.html new file mode 100644 index 0000000000..eab422bab1 --- /dev/null +++ b/dom/media/test/test_fragment_play.html @@ -0,0 +1,91 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="fragment_play.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +PARALLEL_TESTS = 1; +var manager = new MediaTestManager; + +// Fragment parameters to try. These tests +// try playing the video. Tests for other fragment +// formats are in test_fragment_noplay.html. +var gFragmentParams = [ + { fragment: "", start: null, end: null }, + { fragment: "#t=,", start: null, end: null }, + { fragment: "#t=3,3", start: null, end: null }, + { fragment: "#t=7,3", start: null, end: null }, + { fragment: "#t=7,15", start: 7, end: null }, + { fragment: "#t=15,20", start: 9.287982, end: null }, + { fragment: "#t=5", start: 5, end: null }, + { fragment: "#t=5.5", start: 5.5, end: null }, + { fragment: "#t=5,", start: null, end: null }, + { fragment: "#t=,5", start: 0, end: 5, todo: "See bugs 682141 and 720248" }, + { fragment: "#t=2.5,5.5", start: 2.5, end: 5.5, todo: "See bugs 682141 and 720248" }, + { fragment: "#t=1,2.5", start: 1, end: 2.5, todo: "See bugs 682141 and 720248" }, + { fragment: "#t=,15", start: 0, end: null } +]; + +function createTestArray() { + var tests = []; + var tmpVid = document.createElement("video"); + + for (var testNum=0; testNum<gFragmentTests.length; testNum++) { + var test = gFragmentTests[testNum]; + if (!tmpVid.canPlayType(test.type)) { + continue; + } + + for (var fragNum=0; fragNum<gFragmentParams.length; fragNum++) { + var p = gFragmentParams[fragNum]; + var t = {}; + t.name = test.name + p.fragment; + t.type = test.type; + t.duration = test.duration; + t.start = p.start; + t.end = p.end; + t.todo = p.todo; + tests.push(t); + } + } + return tests; +} + +function startTest(test, token) { + if (test.todo) { + todo(false, test.todo); + return; + } + var video = document.createElement('video'); + manager.started(token); + video.preload = "metadata"; + video.src = test.name; + video.token = token; + video.controls = true; + document.body.appendChild(video); + var name = test.name + " fragment test"; + var localIs = function(n) { return function(a, b, msg) { + is(a, b, n + ": " + msg); + }}(name); + var localOk = function(n) { return function(a, msg) { + ok(a, n + ": " + msg); + }}(name); + var localFinish = function(v, m) { return function() { + removeNodeAndSource(v); + m.finished(v.token); + }}(video, manager); + window.test_fragment_play(video, test.start, test.end, localIs, localOk, localFinish); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_hls_player_independency.html b/dom/media/test/test_hls_player_independency.html new file mode 100644 index 0000000000..cea5c140ed --- /dev/null +++ b/dom/media/test/test_hls_player_independency.html @@ -0,0 +1,53 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of 2 HLS video at the same page </title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + <div id='player1'> + <video id='player4x3' controls autoplay> + </video> + </div> + <p> 4x3 basic stream<span> + <span> + <div height = 10> + <span> + <div id='player2'> + <video id='player16x9' controls autoplay> + </video> + </div> + <p> 16x9 basic stream<span> + +<script class="testbody" type="text/javascript"> + +function startTest() { + var v4x3 = document.getElementById('player4x3'); + var v16x9 = document.getElementById('player16x9'); + + var p1 = once(v4x3, 'ended', function onended(e) { + is(v4x3.videoWidth, 400, "4x3 content, the width should be 400."); + is(v4x3.videoHeight, 300, "4x3 content, the height should be 300."); + }); + + var p2 = once(v16x9, 'ended', function onended(e) { + is(v16x9.videoWidth, 416, "16x9 content, the width should be 416."); + is(v16x9.videoHeight, 234, "16x9 content, the height should be 234."); + }); + + v4x3.src = serverUrl + "/bipbop_4x3_single.m3u8"; + v16x9.src = serverUrl + "/bipbop_16x9_single.m3u8"; + Promise.all([p1, p2]).then(() => { + SimpleTest.finish(); + }); +} + +startTest(); +SimpleTest.waitForExplicitFinish(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_imagecapture.html b/dom/media/test/test_imagecapture.html new file mode 100644 index 0000000000..d61dd358bb --- /dev/null +++ b/dom/media/test/test_imagecapture.html @@ -0,0 +1,128 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=1041393 +--> +<head> + <meta charset="utf-8"> + <title>ImageCapture tests</title> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1041393">ImageCapture tests</a> +<script type="application/javascript"> + +// Check if the callback returns even no JS reference on it. +async function gcTest(track) { + const repeat = 100; + const promises = []; + for (let i = 0; i < repeat; ++i) { + const imageCapture = new ImageCapture(track); + promises.push(new Promise((resolve, reject) => { + imageCapture.onphoto = resolve; + imageCapture.onerror = () => + reject(new Error(`takePhoto gcTest failure for capture ${i}`)); + })); + imageCapture.takePhoto(); + } + info("Call gc "); + SpecialPowers.gc(); + await Promise.all(promises); +} + +// Continue calling takePhoto() in rapid succession. +async function rapidTest(track) { + const repeat = 100; + const imageCapture = new ImageCapture(track); + // "error" can fire synchronously in `takePhoto` + const errorPromise = new Promise(r => imageCapture.onerror = r); + for (let i = 0; i < repeat; ++i) { + imageCapture.takePhoto(); + } + for (let i = 0; i < repeat; ++i) { + await Promise.race([ + new Promise(r => imageCapture.onphoto = r), + errorPromise.then(() => Promise.reject(new Error("Capture failed"))), + ]); + } +} + +// Check if the blob is decodable. +async function blobTest(track) { + const imageCapture = new ImageCapture(track); + // "error" can fire synchronously in `takePhoto` + const errorPromise = new Promise(r => imageCapture.onerror = r); + imageCapture.takePhoto(); + const blob = await Promise.race([ + new Promise(r => imageCapture.onphoto = r), + errorPromise.then(() => Promise.reject(new Error("Capture failed"))), + ]); + + const img = new Image(); + img.src = URL.createObjectURL(blob.data); + await new Promise((resolve, reject) => { + img.onload = resolve; + img.onerror = () => reject(new Error("Decode failed")); + }); +} + +// It should return an error event after disabling video track. +async function trackTest(track) { + const imageCapture = new ImageCapture(track); + track.enabled = false; + try { + const errorPromise = new Promise(r => imageCapture.onerror = r); + imageCapture.takePhoto(); + const error = await Promise.race([ + errorPromise, + new Promise(r => imageCapture.onphoto = r).then( + () => Promise.reject(new Error("Disabled video track should fail"))), + ]); + is(error.imageCaptureError.code, error.imageCaptureError.PHOTO_ERROR, + "Expected error code") + } finally { + track.enabled = true; + } +} + +async function init() { + // Use loopback camera if available, otherwise fake. + // MediaTrackGraph will be the backend of ImageCapture. + await setupGetUserMediaTestPrefs(); + let stream = await navigator.mediaDevices.getUserMedia({video: true}); + return stream.getVideoTracks()[0]; +} + +async function start() { + try { + const track = await init(); + info("ImageCapture track disable test."); + await trackTest(track); + info("ImageCapture blob test."); + await blobTest(track); + info("ImageCapture rapid takePhoto() test."); + await rapidTest(track); + info("ImageCapture multiple instances test."); + await gcTest(track); + } catch (e) { + ok(false, "Unexpected error during test: " + e); + } finally { + SimpleTest.finish(); + } +} + +SimpleTest.requestCompleteLog(); +SimpleTest.waitForExplicitFinish(); + +SpecialPowers.pushPrefEnv({ + "set": [ + ["dom.imagecapture.enabled", true], + ["media.navigator.permission.disabled", true], + ], +}, start); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_info_leak.html b/dom/media/test/test_info_leak.html new file mode 100644 index 0000000000..41b01c74c1 --- /dev/null +++ b/dom/media/test/test_info_leak.html @@ -0,0 +1,175 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=478957 +--> +<head> + <title>Test for Bug 478957</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=478957">Mozilla Bug 478957</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> + +<div id="log" style="font-size: small;"></div> + +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 478957 **/ + +// Tests whether we leak events and state change info when loading stuff from local files from a webserver. + +var manager = new MediaTestManager; + +var gEventTypes = [ 'loadstart', 'progress', 'suspend', 'abort', 'error', 'emptied', 'stalled', 'play', 'pause', 'loadedmetadata', 'loadeddata', 'waiting', 'playing', 'canplay', 'canplaythrough', 'seeking', 'seeked', 'timeupdate', 'ended', 'ratechange', 'durationchange', 'volumechange' ]; + +var gExpectedEvents = ['loadstart', 'error']; + +function createTestArray() { + var tests = []; + var tmpVid = document.createElement("video"); + + return makeInfoLeakTests().then(infoLeakTests => { + for (var testNum=0; testNum < infoLeakTests.length; testNum++) { + var test = infoLeakTests[testNum]; + if (!tmpVid.canPlayType(test.type)) { + continue; + } + + var t = {}; + t.name = test.src; + t.type = test.type; + + tests.push(t); + } + return tests; + }) +} + +function log(msg) { + info(msg); + var l = document.getElementById('log'); + // eslint-disable-next-line no-unsanitized/property + l.innerHTML += msg + "<br>"; +} + +function finish(v) { + log("finish: " + v.name); + clearInterval(v.checkStateInterval); + + for (var i=0; i<gEventTypes.length; i++) { + v.removeEventListener(gEventTypes[i], listener); + } + removeNodeAndSource(v); + + manager.finished(v.token); + v = null; +} + +function listener(evt) { + var v = evt.target; + log(filename(v.name) + ': got ' + evt.type); + + // On slow machines like B2G emulator, progress timer could time out before + // receiving any HTTP notification. We will ignore the 'stalled' event to + // pass the tests. + if (evt.type == 'stalled') { + return; + } + + ok(v.eventNum < gExpectedEvents.length, filename(v.name) + " Too many events received"); + var expected = (v.eventNum < gExpectedEvents.length) ? gExpectedEvents[v.eventNum] : "NoEvent"; + is(evt.type, expected, filename(v.name) + " Events received in wrong order"); + v.eventNum++; + if (v.eventNum == gExpectedEvents.length) { + // In one second, move onto the next test. This give a chance for any + // other events to come in. Note: we don't expect any events to come + // in, unless we've leaked some info, and 1 second should be enough time + // for the leak to show up. + setTimeout(function() {finish(v);}, 1000); + } +} + +function createMedia(type, src, token) { + var tag = getMajorMimeType(type); + var v = document.createElement(tag); + for (var i=0; i<gEventTypes.length; i++) { + v.addEventListener(gEventTypes[i], listener); + } + v.preload = "metadata"; + v.src = src; + v.name = src; + document.body.appendChild(v); + v.eventNum = 0; + v.token = token; + setTimeout( + function() { + v.checkStateInterval = setInterval(function(){checkState(v);},1); + }, 0); +} + +// Define our own ok() and is() functions. The mochitest ones take ages constructing the log +// of all the passes, so only report failures. +function test_ok(b, msg) { + if (!b) { + log("FAILED test_ok: " + msg); + ok(b, msg); + } +} + +function test_is(a, b, msg) { + if (a != b) { + log("FAILED test_is: " + msg); + is(a,b,msg); + } +} + +function filename(uri) { + return uri.substr(uri.lastIndexOf("/")+1); +} + +function checkState(v) { + test_ok(v.networkState <= HTMLMediaElement.NETWORK_LOADING || + v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE, + "NetworkState of " + v.networkState + " was leaked."); + test_ok(v.readyState == HTMLMediaElement.HAVE_NOTHING, + "Ready state of " + v.readyState + " was leaked"); + test_is(v.seeking, false, "Seeking leaked"); + test_is(v.currentTime, 0, "Leaked currentTime"); + test_ok(isNaN(v.duration), "Leaked duration"); + test_is(v.paused, true, "Paused leaked"); + test_is(v.ended, false, "Ended leaked"); + test_is(v.autoplay, false, "Autoplay leaked"); + test_is(v.controls, false, "Controls leaked"); + test_is(v.muted, false, "muted leaked"); + test_ok(v.error==null || v.error.code==MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, + "Error code should not exist or be SRC_NOT_SUPPORTED. v.error=" + + (v.error ? v.error.code : "null")); + test_ok(filename(v.currentSrc) == filename(v.name) || + v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE, + "currentSrc should match candidate uri, if we've got a valid source"); +} + + +function startTest(test, token) { + manager.started(token); + log("Testing: " + test.type + " @ " + test.name); + createMedia(test.type, test.name, token); +} + +SimpleTest.waitForExplicitFinish(); +createTestArray().then(testArray => { + manager.runTests(testArray, startTest); +}); + +</script> +</pre> + +</body> +</html> diff --git a/dom/media/test/test_invalid_reject.html b/dom/media/test/test_invalid_reject.html new file mode 100644 index 0000000000..583847fe12 --- /dev/null +++ b/dom/media/test/test_invalid_reject.html @@ -0,0 +1,58 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="UTF-8" /> + <title>Test rejection of invalid files</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var v = document.createElement('video'); + manager.started(token); + + // Set up event handlers. Seeing any of these is a failure. + function badEvent(type) { return function(e) { + ok(false, test.name + " should not fire '" + type + "' event"); + }}; + var events = [ + 'loadeddata', 'load', + 'canplay', 'canplaythrough', + 'playing' + ]; + events.forEach( function(e) { + v.addEventListener(e, badEvent(e)); + }); + + // Seeing a decoder error is a success. + v.addEventListener("error", function onerror(e) { + if (v.readyState == v.HAVE_NOTHING) { + is(v.error.code, v.error.MEDIA_ERR_SRC_NOT_SUPPORTED, + "decoder should reject " + test.name); + } else { + is(v.error.code, v.error.MEDIA_ERR_DECODE, + "decoder should reject " + test.name); + } + v.removeEventListener('error', onerror); + manager.finished(token); + }); + + // Now try to load and play the file, which should result in the + // error event handler above being called, terminating the test. + document.body.appendChild(v); + v.src = test.name; + v.play(); +} + +manager.runTests(gInvalidTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_invalid_reject_play.html b/dom/media/test/test_invalid_reject_play.html new file mode 100644 index 0000000000..3e658f94b8 --- /dev/null +++ b/dom/media/test/test_invalid_reject_play.html @@ -0,0 +1,44 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="UTF-8" /> + <title>Test rejection of invalid files during playback</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var v = document.createElement('video'); + manager.started(token); + + // Seeing a decoder error is a success. + v.addEventListener("error", function onerror(e) { + is(v.error.code, v.error.MEDIA_ERR_DECODE, + "decoder should reject " + test.name); + v.removeEventListener("error", onerror); + manager.finished(token); + }); + + v.addEventListener("ended", function onended(e) { + ok(false, "decoder should have rejected file before playback ended"); + v.removeEventListener("ended", onended); + manager.finished(token); + }); + + document.body.appendChild(v); + v.src = test.name; + v.play(); +} + +manager.runTests(gInvalidPlayTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_invalid_seek.html b/dom/media/test/test_invalid_seek.html new file mode 100644 index 0000000000..8aa514b977 --- /dev/null +++ b/dom/media/test/test_invalid_seek.html @@ -0,0 +1,32 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: invalide seek test</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<video id='v'></video> +<pre id="test"> +<script class="testbody" type="text/javascript"> +// http://www.whatwg.org/specs/web-apps/current-work/#dom-media-seek +// If the media element's readyState is HAVE_NOTHING, then the user agent +// must raise an InvalidStateError exception. +var v = document.getElementById('v'); +var passed = false; + +ok(v.readyState == HTMLMediaElement.HAVE_NOTHING, + "Invalid ready state in media element"); + +try { + v.seek(1); +} +catch(e) { + passed = true; +} + +ok(passed, "Video did not raise error during invalid seek"); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_load.html b/dom/media/test/test_load.html new file mode 100644 index 0000000000..de8fd63948 --- /dev/null +++ b/dom/media/test/test_load.html @@ -0,0 +1,217 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=479859 +--> +<head> + <title>Test for Bug 479859</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="application/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479859">Mozilla Bug 479859</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="text/javascript"> + +function log(msg) { + //document.getElementById('log').innerHTML += "<p>" + msg + "</p>"; +} + +// We don't track: progress, canplay, canplaythrough and stalled events, +// as these can be delivered out of order, and/or multiple times. +var gEventTypes = [ 'loadstart', 'abort', 'error', 'emptied', 'play', + 'pause', 'loadedmetadata', 'loadeddata', 'waiting', 'playing', 'seeking', + 'seeked', 'timeupdate', 'ended', 'ratechange', 'durationchange', 'volumechange' ]; + +var gEventNum = 0; +var gTestNum = 0; +var gTestFileNum = 0; +var gExpectedEvents = null; +var gTest = null; +var gTestName = "?"; + +function listener(evt) { + log('event ' + evt.type); + ok(gEventNum < gExpectedEvents.length, gTestName+" - corrent number of events received"); + var expected = (gEventNum < gExpectedEvents.length) ? gExpectedEvents[gEventNum] : "NoEvent"; + is(evt.type, expected, gTestName+" - events received in order"); + gEventNum++; + if (gEventNum == gExpectedEvents.length) { + setTimeout(nextTest, 0); + } +} + +function source_error(evt) { + log('event source_error'); + ok(evt.type == "error", "Should only get error events here"); + ok(gEventNum < gExpectedEvents.length, gTestName+" - corrent number of events received"); + var expected = (gEventNum < gExpectedEvents.length) ? gExpectedEvents[gEventNum] : "NoEvent"; + is("source_error", expected, gTestName+" - events received in order"); + gEventNum++; + if (gEventNum == gExpectedEvents.length) { + setTimeout(nextTest, 0); + } +} + +var gMedia = null; + +function createMedia(tag) { + gMedia = document.createElement(tag); + gMedia.preload = "metadata"; + for (var i=0; i<gEventTypes.length; i++) { + gMedia.addEventListener(gEventTypes[i], listener); + } +} + +function addSource(src, type) { + var s = document.createElement("source"); + s.addEventListener("error", source_error); + s.src = src; + s.type = type; + gMedia.appendChild(s); + return s; +} + +function prependSource(src, type) { + var s = document.createElement("source"); + s.addEventListener("error", source_error); + s.src = src; + s.type = type; + gMedia.insertBefore(s, gMedia.firstChild); + return s; +} + +var gTests = [ + { + // Test 0: adding video to doc, then setting src should load implicitly. + create(src, type) { + document.body.appendChild(gMedia); + gMedia.src = src; + }, + expectedEvents: ['loadstart', 'durationchange', 'loadedmetadata', 'loadeddata'] + }, { + // Test 1: adding video to doc, then adding source. + create(src, type) { + document.body.appendChild(gMedia); + addSource(src, type); + }, + expectedEvents: ['loadstart', 'durationchange', 'loadedmetadata', 'loadeddata'] + },{ + // Test 2: video with multiple source, the first of which are bad, we should load the last, + // and receive error events for failed loads on the source children. + create(src, type) { + document.body.appendChild(gMedia); + addSource("404a", type); + addSource("404b", type); + addSource(src, type); + }, + expectedEvents: ['loadstart', 'source_error', 'source_error', 'durationchange', 'loadedmetadata', 'loadeddata'] + }, { + // Test 3: video with bad src, good <source>, ensure that <source> aren't used. + create(src, type) { + gMedia.src = "404a"; + addSource(src, type); + document.body.appendChild(gMedia); + }, + expectedEvents: ['loadstart', 'error'] + }, { + // Test 4: video with only bad source, loading, then adding a good source + // - should resume load. + create(src, type) { + addSource("404a", type); + var s2 = addSource("404b", type); + s2.addEventListener("error", + function(e) { + // Should awaken waiting load, causing successful load. + addSource(src, type); + }); + document.body.appendChild(gMedia); + }, + expectedEvents: ['loadstart', 'source_error', 'source_error', 'durationchange', 'loadedmetadata', 'loadeddata'] + }, { + // Test 5: video with only 1 bad source, let it fail to load, then prepend + // a good <source> to the video, it shouldn't be selected, because the + // "pointer" should be after the last child - the bad source. + prepended: false, + create(src, type) { + var prepended = false; + addSource("404a", type); + var s2 = addSource("404b", type); + s2.addEventListener("error", + function(e) { + // Should awaken waiting load, causing successful load. + if (!prepended) { + prependSource(src, type); + prepended = true; + } + }); + document.body.appendChild(gMedia); + }, + expectedEvents: ['loadstart', 'source_error', 'source_error'] + }, { + // Test 6: (Bug 1165203) preload="none" then followed by an explicit + // call to load() should load metadata + create(src, type) { + gMedia.preload = "none"; + gMedia.src = src; + document.body.appendChild(gMedia); + addSource(src, type); + gMedia.load(); + }, + expectedEvents: ['emptied', 'loadstart', 'durationchange', 'loadedmetadata', 'loadeddata'] + } +]; + +function nextTest() { + if (gMedia) { + for (var i=0; i<gEventTypes.length; i++) { + gMedia.removeEventListener(gEventTypes[i], listener); + } + removeNodeAndSource(gMedia); + gMedia = null; + } + gEventNum = 0; + + if (gTestNum == gTests.length) { + gTestNum = 0; + ++gTestFileNum; + if (gTestFileNum == gSmallTests.length) { + SimpleTest.finish(); + return; + } + } + + var src = gSmallTests[gTestFileNum].name; + var type = gSmallTests[gTestFileNum].type; + + var t = gTests[gTestNum]; + gTestNum++; + + createMedia(type.match(/^audio\//) ? "audio" : "video"); + if (!gMedia.canPlayType(type)) { + // Unsupported type, skip to next test + nextTest(); + return; + } + + gTestName = "Test " + src + " " + (gTestNum - 1); + log("Starting " + gTestName); + gExpectedEvents = t.expectedEvents; + + t.create(src, type); +} + +addLoadEvent(nextTest); +SimpleTest.waitForExplicitFinish(); + +</script> +</pre> + +<div id="log" style="font-size: small"></div> +</body> +</html> diff --git a/dom/media/test/test_load_candidates.html b/dom/media/test/test_load_candidates.html new file mode 100644 index 0000000000..2bc1eb531b --- /dev/null +++ b/dom/media/test/test_load_candidates.html @@ -0,0 +1,84 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=465458 +--> +<head> + <title>Test for Bug 465458</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=465458">Mozilla Bug 465458</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 465458 **/ + +var manager = new MediaTestManager; + +function finish(evt) { + var v = evt.target; + is(v._error, 2, "Should have received 2 error events before loaded"); + v._finished = true; + // remove error event handler, because this would otherwise + // cause a failure on Windows 7, see bug 1024535 + v.onerror = null; + v.remove(); + manager.finished(v.token); +} + +function errorHandler(evt) { + evt.target.parentNode._error++; +} + +var extension = { + "audio/wav" : "wav", + "audio/x-wav": "wav", + "video/ogg" : "ogv", + "audio/ogg" : "oga", + "video/webm" : "webm" +}; + +function startTest(test, token) { + var v = document.createElement('video'); + v.preload = "auto"; + v.onerror = function(){ok(false,"Error events on source children should not bubble");} + v.token = token; + manager.started(token); + v._error = 0; + v._finished = false; + v._name = test.name; + + var s1 = document.createElement("source"); + s1.type = test.type; + s1.src = "404." + extension[test.type]; + s1.addEventListener("error", errorHandler); + v.appendChild(s1); + + var s2 = document.createElement("source"); + s2.type = test.type; + s2.src = "test_load_candidates.html"; // definitely an invalid media file, regardless of its actual mime type... + s2.addEventListener("error", errorHandler); + v.appendChild(s2); + + var s3 = document.createElement("source"); + s3.type = test.type; + s3.src = test.name; + v.appendChild(s3); + + v.addEventListener("loadeddata", finish); + document.body.appendChild(v); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_load_same_resource.html b/dom/media/test/test_load_same_resource.html new file mode 100644 index 0000000000..d3e16c88a5 --- /dev/null +++ b/dom/media/test/test_load_same_resource.html @@ -0,0 +1,106 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test loading of the same resource in multiple elements</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.requestCompleteLog(); +var manager = new MediaTestManager; + +function checkDuration(actual, expected, name) { + ok(Math.abs(actual - expected) < 0.1, + `${name} duration: ${actual} expected: ${expected}`); +} + +function cloneLoaded(event) { + var e = event.target; + ok(true, `${e.token} loaded OK`); + checkDuration(e.duration, e._expectedDuration, e.token); + removeNodeAndSource(e); + manager.finished(e.token); +} + +function tryClone(e) { + var clone = e.cloneNode(false); + clone.token = `${e.token}(cloned)`; + manager.started(clone.token); + manager.finished(e.token); + + // Log events for debugging. + var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(evt) { + var event = evt.target; + info(`${event.token} got ${evt.type}`); + } + events.forEach(function(event) { + clone.addEventListener(event, logEvent); + }); + + + checkDuration(e.duration, e._expectedDuration, e.token); + clone._expectedDuration = e._expectedDuration; + + clone.addEventListener("loadeddata", cloneLoaded, {once: true}); + clone.addEventListener("loadstart", function(evt) { + info(`${evt.target.token} starts loading.`); + // Since there is only one H264 decoder instance, we have to delete the + // decoder of the original element for the cloned element to load. However, + // we can't delete the decoder too early otherwise cloning decoder will + // fail to kick in. We wait for 'loadstart' event of the cloned element to + // know when the decoder is already cloned and we can delete the decoder of + // the original element. + removeNodeAndSource(e); + }, {once: true}); +} + +// This test checks that loading the same URI twice in different elements at the same time +// uses the same resource without doing another network fetch. One of the gCloneTests +// uses dynamic_resource.sjs to return one resource on the first fetch and a different resource +// on the second fetch. These resources have different lengths, so if the cloned element +// does a network fetch it will get a resource with the wrong length and we get a test +// failure. + +async function initTest(test, token) { + var e = document.createElement("video"); + e.preload = "auto"; + e.src = test.name; + e._expectedDuration = test.duration; + ok(true, `Trying to load ${test.name}, duration=${test.duration}`); + e.token = token; + manager.started(token); + + // Since 320x240.ogv is less than 32KB, we need to wait for the + // 'suspend' event to ensure the partial block is flushed to the cache + // otherwise the cloned resource will create a new channel when it + // has no data to read at position 0. The new channel will download + // a different file than the original resource and fail the duration + // test. + let p1 = once(e, "loadeddata"); + let p2 = once(e, "suspend"); + await p1; + await p2; + tryClone(e); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + {"set": [ + ["logging.MediaDecoder", "Debug"], + ["logging.nsMediaElement", "Debug"], + ["logging.MediaCache", "Debug"], + ]}, + manager.runTests.bind(manager, gCloneTests, initTest)); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_load_source.html b/dom/media/test/test_load_source.html new file mode 100644 index 0000000000..95a925b61f --- /dev/null +++ b/dom/media/test/test_load_source.html @@ -0,0 +1,76 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=534571 +--> +<head> + <title>Test for Bug 534571</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=534571">Mozilla Bug 534571</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 534571 **/ + +// Test that when we load a video from a source child and then change the +// source's src attribute and load again, that the subsequent loads work. + +var v = null; +var s = null; + +function finish(event) { + ok(true, "Should have played both videos"); + SimpleTest.finish(); +} + +var first = null; +var second = null; + +function ended(event) { + s.type = second.type; + s.src = second.name; + v.removeEventListener("ended", ended); + v.addEventListener("ended", finish); + v.load(); +} + +// Find 2 videos we can play. +v = document.createElement('video'); +for (var i=0; i<gPlayTests.length; i++) { + if (!v.canPlayType(gPlayTests[i].type)) + continue; + if (!first) { + first = gPlayTests[i]; + } else if (!second) { + second = gPlayTests[i]; + break; + } +} + +if (first && second) { + s = document.createElement('source'); + s.type = first.type; + s.src = first.name; + v.appendChild(s); + v.autoplay = true; + v.addEventListener("ended", ended); + document.body.appendChild(v); + SimpleTest.waitForExplicitFinish(); +} else { + todo(false, "Need at least two media of supported types for this test!"); +} + + + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_load_source_empty_type.html b/dom/media/test/test_load_source_empty_type.html new file mode 100644 index 0000000000..9ceee9af46 --- /dev/null +++ b/dom/media/test/test_load_source_empty_type.html @@ -0,0 +1,36 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Load resource from source element with empty type</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<video id="type"><source src="gizmo.mp4" type></video> +<video id="type-and-equal"><source src="gizmo.mp4" type=></video> +<video id="type-and-equal-quotation-marks"><source src="gizmo.mp4" type=""></video> +<script class="testbody" type="text/javascript"> +/** + * This test is used to ensure that media element can start loading when its + * child source element with empty type property. We would test following forms, + * `type`, `type=` and `type=""`. + */ +SimpleTest.waitForExplicitFinish(); + +const videos = document.getElementsByTagName("video"); +const videoNum = videos.length; +let loadedElementsNum = 0; + +for (let video of videos) { + video.addEventListener("loadeddata", + () => { + ok(true, `loaded element '${video.id}'`); + if (++loadedElementsNum == videoNum) { + SimpleTest.finish(); + } + }, { once: true}); +} +</script> +</body> +</html> diff --git a/dom/media/test/test_loop.html b/dom/media/test/test_loop.html new file mode 100644 index 0000000000..943059edad --- /dev/null +++ b/dom/media/test/test_loop.html @@ -0,0 +1,57 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test looping support</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + manager.started(token); + var v = document.createElement('video'); + v.token = token; + v.src = test.name; + v.name = test.name; + v.playCount = 0; + v.seekingCount = 0; + v.seekedCount = 0; + v.loop = true; + + v.addEventListener("play", function (e) { + e.target.playCount += 1; + ok(e.target.playCount == 1, "Should get exactly one play event."); + }); + + v.addEventListener("seeking", function (e) { + e.target.seekingCount += 1; + }); + + v.addEventListener("seeked", function (e) { + e.target.seekedCount += 1; + if (e.target.seekedCount == 2) { + ok(e.target.seekingCount == 2, "Expect matched pairs of seeking/seeked events."); + e.target.loop = false; + } + }); + + v.addEventListener("ended", function (e) { + ok(!e.target.loop, "Shouldn't get ended event while looping."); + removeNodeAndSource(e.target); + manager.finished(e.target.token); + }); + + document.body.appendChild(v); + v.play(); +} + +manager.runTests(gSmallTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_looping_eventsOrder.html b/dom/media/test/test_looping_eventsOrder.html new file mode 100644 index 0000000000..7d070de72f --- /dev/null +++ b/dom/media/test/test_looping_eventsOrder.html @@ -0,0 +1,52 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Looping events order</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<script type="text/javascript"> +/** + * This test is used to ensure the events order when media is looping back to + * the start position. We should see events in following order. + * 'seeking' -> 'timeupdate' -> 'seeked'. + */ +SimpleTest.waitForExplicitFinish(); + +var tests = [ + { name:"small-shot.ogg", type:"audio/ogg" }, + { name:"seek-short.webm", type:"video/webm" } +]; + +async function testTimeupdateChanged({name, type}) { + info(`- start testPlay for name=${name} -`); + const element = document.createElement(getMajorMimeType(type)); + element.src = name; + element.loop = true; + + await once(element, "canplay"); + ok(await element.play().then(() => true, () => false), `start playing ${name}`); + + let gotTimeUpdated = false; + await once(element, "seeking"); + element.addEventListener("timeupdate", function() { + gotTimeUpdated = true; + }, {once: true}); + await once(element, "seeked"); + ok(gotTimeUpdated, "Got timeupdate between seeking and seeked."); + + removeNodeAndSource(element); +} + +(async function startTest() { + for (let test of tests) { + await testTimeupdateChanged(test); + } + SimpleTest.finish(); +})(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_media_selection.html b/dom/media/test/test_media_selection.html new file mode 100644 index 0000000000..42f5a9bd43 --- /dev/null +++ b/dom/media/test/test_media_selection.html @@ -0,0 +1,142 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: media selection</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="application/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function maketest(attach_media, name, type, check_metadata) { + return function (token) { + var e = document.createElement('video'); + e.preload = "metadata"; + token = name + "-" + token; + manager.started(token); + var errorRun = false; + if (check_metadata) { + e.addEventListener('loadedmetadata', function () { + ok(e.readyState >= HTMLMediaElement.HAVE_METADATA, + 'test ' + token + ' readyState ' + e.readyState + ' expected >= ' + HTMLMediaElement.HAVE_METADATA); + is(e.currentSrc.substring(e.currentSrc.length - name.length), name, 'test ' + token); + // The load can go idle due to cache size limits + ok(e.networkState >= HTMLMediaElement.NETWORK_IDLE, + 'test ' + token + ' networkState = ' + e.networkState + ' expected >= ' + HTMLMediaElement.NETWORK_IDLE); + check_metadata(e); + removeNodeAndSource(e); + manager.finished(token); + }); + } else { + e.addEventListener('error', function onerror(event) { + is(errorRun, false, "error handler should run once only!"); + errorRun = true; + is(e.readyState, HTMLMediaElement.HAVE_NOTHING, + 'test ' + token + ' readyState should be HAVE_NOTHING when load fails.'); + e.removeEventListener('error', onerror, true); + removeNodeAndSource(e); + manager.finished(token); + }, true); + } + attach_media(e, name, type); + } +} + +function set_src(element, name, type) { + element.src = name; + document.body.appendChild(element); +} + +function add_source(element, name, type) { + do_add_source(element, name, type); + document.body.appendChild(element); +} + +function do_add_source(element, name, type) { + var source = document.createElement('source'); + if (type) { + source.type = type; + } + source.src = name; + element.appendChild(source); +} + +function add_sources_last(element, name, type) { + do_add_source(element, name, 'unsupported/type'); + do_add_source(element, name, type); + document.body.appendChild(element); +} + +function add_sources_first(element, name, type) { + do_add_source(element, name, type); + do_add_source(element, name, 'unsupported/type'); + document.body.appendChild(element); +} + +function late_add_sources_last(element, name, type) { + document.body.appendChild(element); + do_add_source(element, name, 'unsupported/type'); + do_add_source(element, name, type); +} + +function late_add_sources_first(element, name, type) { + document.body.appendChild(element); + do_add_source(element, name, type); + do_add_source(element, name, 'unsupported/type'); +} + +var nextTest = 0; +var subtests = [ + maketest(add_source, 'unknown.raw', 'bogus/type', null) +]; + +var tmpVid = document.createElement('video'); + +for (var i = 0; i < gSmallTests.length; ++i) { + var test = gSmallTests[i]; + var src = test.name; + var type = test.type; + + if (!tmpVid.canPlayType(type)) + continue; + + // The following nested function hack is to ensure that 'test' is correctly + // captured in the closure and we don't end up getting the value 'test' + // had in the last iteration of the loop. I blame Brendan. + var check = function(t) { return function (e) { + checkMetadata(t.name, e, t); + }}(test); + + var otherType = type.match(/^video\//) ? "audio/x-wav" : "video/ogg"; + subtests.push(maketest(set_src, src, null, check), + maketest(add_source, src, null, check), + maketest(add_source, src, type, check), + maketest(add_sources_last, src, null, check), + maketest(add_sources_first, src, type, check), + + // type hint matches a decoder, actual type matches different decoder + maketest(add_source, src, otherType, check), + maketest(add_source, 'unknown.raw', type, null), + + // should not start loading, type excludes it from media candiate list + maketest(add_source, src, 'bogus/type', null), + + // element doesn't notice source children attached later, needs bug 462455 fixed + maketest(late_add_sources_last, src, type, check), + maketest(late_add_sources_first, src, type, check)); +} + +function startTest(t, token) { + t(token); +} + +manager.runTests(subtests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_media_sniffer.html b/dom/media/test/test_media_sniffer.html new file mode 100644 index 0000000000..68e2bba8af --- /dev/null +++ b/dom/media/test/test_media_sniffer.html @@ -0,0 +1,67 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: sniffing</title> + <meta charset='utf-8'> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function finish_test(element) { + element.removeEventListener("error", onerror); + removeNodeAndSource(element); + manager.finished(element.token); +} + +function onmetadataloaded(e) { + var t = e.target; + ++t.srcIndex; + ok(true, "The media loads when loaded via " + t.src); + if (t.srcIndex < t.srcList.length) { + t.src = t.srcList[t.srcIndex]; + } else { + finish_test(t); + } +} + +function onerror(e) { + var t = e.target; + t.removeEventListener('error', onerror); + ok(false, "The media could not be loaded." + t.src + "\n"); + finish_test(t); +} + +function startTest(test, token) { + var elemType = /^audio/.test(test.type) ? "audio" : "video"; + var element = document.createElement(elemType); + // This .sjs file serve the media file without Content-Type header, or with a + // specific Content-Type header. + var baseSrc = 'contentType.sjs?file=' + test.name; + element.srcList = [ + baseSrc + "&nomime", + baseSrc + "&type=application/octet-stream", + baseSrc + "&type=audio/wav", + baseSrc + "&type=text/html", + baseSrc + "&type=absolute_nonsense" + ]; + element.srcIndex = 0; + element.src = element.srcList[element.srcIndex]; + element.token = token; + element.controls = true; + element.preload = "metadata"; + document.body.appendChild(element); + manager.started(token); + element.addEventListener("loadedmetadata", onmetadataloaded); + element.addEventListener("error", onerror); +} + +manager.runTests(gSnifferTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediacapabilities_resistfingerprinting.html b/dom/media/test/test_mediacapabilities_resistfingerprinting.html new file mode 100644 index 0000000000..1f07d1c707 --- /dev/null +++ b/dom/media/test/test_mediacapabilities_resistfingerprinting.html @@ -0,0 +1,69 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test for Bug 1369309</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1461454">Mozilla Bug 1461454</a> +<a target="_blank" href="https://gitlab.torproject.org/tpo/applications/tor-browser/-/issues/13543">Tor Issue 13543</a> + +<script type="application/javascript"> +async function testWhetherSpoofed(resistFingerprinting) { + + var unsupportedVideoConfiguration = { + contentType: 'video/bogus', + width: 800, + height: 600, + bitrate: 3000, + framerate: 24, + }; + var supportedVideoConfiguration = { + contentType: 'video/webm; codecs="vp09.00.10.08"', + width: 800, + height: 600, + bitrate: 3000, + framerate: 24, + }; + + await SpecialPowers.pushPrefEnv({ + "set": [ + ["privacy.resistFingerprinting", resistFingerprinting] + ], + }); + + var result; + + result = await navigator.mediaCapabilities.decodingInfo({ + type: 'file', + video: unsupportedVideoConfiguration + }); + is(result.supported, false, "video/bogus should be unsupported."); + is(result.smooth, false, "smooth is false when unsupported."); + is(result.powerEfficient, false, "powerEfficient is false when unsupported."); + + result = await navigator.mediaCapabilities.decodingInfo({ + type: 'file', + video: supportedVideoConfiguration + }); + is(result.supported, true, "'video/webm; codecs=\"vp09.00.10.08\"' should be supported."); + if (resistFingerprinting) { + is(result.smooth, true, "smooth should be spoofed to true in RFP mode."); + is(result.powerEfficient, false, "powerEfficient should be spoofed to false in RFP mode."); + } +} + +async function start() { + await testWhetherSpoofed(true); + await testWhetherSpoofed(false); + SimpleTest.finish(); +} + +SimpleTest.waitForExplicitFinish(); +start(); +</script> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_avoid_recursion.html b/dom/media/test/test_mediarecorder_avoid_recursion.html new file mode 100644 index 0000000000..0c733b9fc2 --- /dev/null +++ b/dom/media/test/test_mediarecorder_avoid_recursion.html @@ -0,0 +1,61 @@ +<html> +<head> + <title>MediaRecorder infinite recursion with requestData() calls in "dataavailable" event</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=897776">Mozill +a Bug 897776</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> +async function startTest() { + try { + await setupGetUserMediaTestPrefs(); + let stream = await navigator.mediaDevices.getUserMedia({audio: true}); + let mediaRecorder = new MediaRecorder(stream); + let count = 0; + mediaRecorder.onerror = function () { + ok(false, 'Unexpected onerror callback fired'); + SimpleTest.finish(); + }; + mediaRecorder.onwarning = function () { + ok(false, 'Unexpected onwarning callback fired'); + }; + mediaRecorder.ondataavailable = function (e) { + ++count; + info("got ondataavailable data size = " + e.data.size); + // no more requestData() to prevent busy main thread from starving + // the encoding thread + if (count == 30) { + info("track.stop"); + stream.getTracks()[0].stop(); + } else if (count < 30 && mediaRecorder.state == 'recording') { + info("requestData again"); + mediaRecorder.requestData(); + } + }; + mediaRecorder.onstop = function () { + ok(true, "requestData within ondataavailable successfully avoided infinite recursion"); + SimpleTest.finish(); + }; + mediaRecorder.start(); + info("mediaRecorder start"); + mediaRecorder.requestData(); + info("mediaRecorder requestData"); + } catch (e) { + ok(false, 'Unexpected error fired with: ' + e); + SimpleTest.finish(); + } +} + +SimpleTest.waitForExplicitFinish(); +startTest(); + +</script> +</pre> +</body> +</html> + diff --git a/dom/media/test/test_mediarecorder_bitrate.html b/dom/media/test/test_mediarecorder_bitrate.html new file mode 100644 index 0000000000..693b27e3bf --- /dev/null +++ b/dom/media/test/test_mediarecorder_bitrate.html @@ -0,0 +1,127 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Bitrate</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; +var results = []; + +/** + * Starts a test on every media recorder file included to check that + * the bitrate control works + */ +function startTest(test, token) { + manager.started(token); + runTest(test, token, 1000000); + runTest(test, token, 100000); +} + +function runTest(test, token, bitrate) { + var element = document.createElement('video'); + + element.token = token; + + element.src = test.name; + element.preload = "metadata"; + element.onloadedmetadata = function () { + info("loadedmetadata"); + const stream = element.mozCaptureStreamUntilEnded(); + element.onloadedmetadata = null; + element.play(); + + const mediaRecorder = new MediaRecorder(stream, {videoBitsPerSecond: bitrate}); + mediaRecorder.start(); + is(mediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mediaRecorder.stream, stream, + 'Media recorder stream = element stream at the start of recording'); + + var onStopFired = false; + var onDataAvailableFired = false; + var encoded_size = 0; + + mediaRecorder.onerror = function () { + ok(false, 'Unexpected onerror callback fired'); + }; + + mediaRecorder.onwarning = function () { + ok(false, 'Unexpected onwarning callback fired'); + }; + + // This handler verifies that only a single onstop event handler is fired. + mediaRecorder.onstop = function () { + if (onStopFired) { + ok(false, 'onstop unexpectedly fired more than once'); + } else { + onStopFired = true; + + // ondataavailable should always fire before onstop + if (onDataAvailableFired) { + ok(true, 'onstop fired after ondataavailable'); + info("test " + test.name + " encoded@" + bitrate + "=" + encoded_size); + if (results[test.name]) { + var big, small, temp; + big = {}; + big.bitrate = bitrate; + big.size = encoded_size; + small = results[test.name]; + // Don't assume the order that these will finish in + if (results[test.name].bitrate > bitrate) { + temp = big; + big = small; + small = temp; + } + // Ensure there is a big enough difference in the encoded + // sizes + ok(small.size*1.25 < big.size, + test.name + ' encoded@' + big.bitrate + '=' + big.size + + ' > encoded@' + small.bitrate + '=' + small.size); + manager.finished(token); + } else { + results[test.name] = {}; + results[test.name].bitrate = bitrate; + results[test.name].size = encoded_size; + } + } else { + ok(false, 'onstop fired without an ondataavailable event first'); + } + } + }; + + // This handler verifies that only a single ondataavailable event handler + // is fired with the blob generated having greater than zero size + // and a correct mime type. + mediaRecorder.ondataavailable = function (evt) { + if (onDataAvailableFired) { + ok(false, 'ondataavailable unexpectedly fired more than once'); + } else { + onDataAvailableFired = true; + + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + ok(evt.data.size > 0, + 'Blob data received should be greater than zero'); + encoded_size = evt.data.size; + + // onstop should not have fired before ondataavailable + if (onStopFired) { + ok(false, 'ondataavailable unexpectedly fired later than onstop'); + manager.finished(token); + } + } + }; + }; +} + +manager.runTests(gMediaRecorderVideoTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_creation.html b/dom/media/test/test_mediarecorder_creation.html new file mode 100644 index 0000000000..cd23cb7a96 --- /dev/null +++ b/dom/media/test/test_mediarecorder_creation.html @@ -0,0 +1,45 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Creation</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +/** + * Starts a test on every media recorder file included to check that + * a media recorder object created with a stream derived from a media + * element with that file produces the correct starting attribute values. + */ +function startTest(test, token) { + var element = document.createElement('audio'); + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStreamUntilEnded(); + + var mediaRecorder = new MediaRecorder(element.stream); + + is(mediaRecorder.stream, element.stream, + 'Stream should be provided stream on creation'); + is(mediaRecorder.mimeType, '', + 'mimeType should be an empty string on creation'); + is(mediaRecorder.state, 'inactive', + 'state should be inactive on creation'); + + manager.finished(token); +} + +manager.runTests(gMediaRecorderTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_creation_fail.html b/dom/media/test/test_mediarecorder_creation_fail.html new file mode 100644 index 0000000000..f411c2d3e1 --- /dev/null +++ b/dom/media/test/test_mediarecorder_creation_fail.html @@ -0,0 +1,50 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record with media.ogg.enabled = false</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +function testThrows(stream, options) { + try { + new MediaRecorder(stream, options); + return false; + } catch(e) { + return e.name; + } +} +(async () => { + SimpleTest.waitForExplicitFinish(); + await SpecialPowers.pushPrefEnv({set: [ + ["media.ogg.enabled", false], + ["media.encoder.webm.enabled", false], + ]}); + const stream = new AudioContext().createMediaStreamDestination().stream; + is(testThrows(stream, {mimeType: "audio/ogg"}), "NotSupportedError", + "Creating an ogg recorder without ogg support throws"); + is(testThrows(stream, {mimeType: "audio/webm"}), "NotSupportedError", + "Creating a webm recorder without webm support throws"); + is(testThrows(stream, {mimeType: "video/webm"}), "NotSupportedError", + "Creating a webm recorder without webm support throws"); + is(testThrows(stream, {mimeType: "apa/bepa"}), "NotSupportedError", + "Creating a recorder for a bogus mime type throws"); + is(testThrows(stream, {}), false, + "Creating a default recorder never throws, even without container support"); + const recorder = new MediaRecorder(stream); + try { + recorder.start(); + ok(false, "Starting a recorder without container support should throw"); + } catch(e) { + is(e.name, "NotSupportedError", + "Starting a recorder without container support throws"); + } + SimpleTest.finish(); +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html b/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html new file mode 100644 index 0000000000..537e1dbb47 --- /dev/null +++ b/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html @@ -0,0 +1,45 @@ +<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Bug 1395022 - MediaRecorder fires start event when erroring.</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1395022">Mozilla Bug 1395022</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+function startTest() {
+ let audioContext = new AudioContext();
+ let destination1 = audioContext.createMediaStreamDestination();
+ let oscilator = audioContext.createOscillator();
+ oscilator.connect(destination1);
+ oscilator.start();
+
+ let destination2 = audioContext.createMediaStreamDestination();
+
+ let rec = new MediaRecorder(destination1.stream);
+
+ let numStartEvents = 0;
+
+ rec.onstart = () => {
+ numStartEvents += 1;
+ is(numStartEvents, 1, "One start event should be fired by the recorder");
+ // Trigger an error in the recorder
+ destination1.stream.addTrack(destination2.stream.getTracks()[0]);
+ };
+
+ rec.onerror = () => {
+ is(numStartEvents, 1, "One start event should have been fired by the recorder");
+ SimpleTest.finish();
+ };
+
+ rec.start();
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</head>
+</html>
\ No newline at end of file diff --git a/dom/media/test/test_mediarecorder_onerror_pause.html b/dom/media/test/test_mediarecorder_onerror_pause.html new file mode 100644 index 0000000000..55ee5fb535 --- /dev/null +++ b/dom/media/test/test_mediarecorder_onerror_pause.html @@ -0,0 +1,107 @@ +<html> +<head> + <title>Bug 957439 - Media Recording - Assertion fail at Pause if unsupported input stream.</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=957439">Mozilla Bug 957439</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +function unexpected(e) { + ok(false, `Got unexpected ${e.type} event`); +} + + +async function startTest() { + // also do general checks on mimetype support for audio-only + ok(MediaRecorder.isTypeSupported("audio/ogg"), + 'Should support audio/ogg'); + ok(MediaRecorder.isTypeSupported('audio/ogg; codecs=opus'), + 'Should support audio/ogg+opus'); + ok(!MediaRecorder.isTypeSupported('audio/ogg; codecs=foobar'), + 'Should not support audio/ogg + unknown_codec'); + ok(MediaRecorder.isTypeSupported("video/webm"), + 'Should support video/webm'); + ok(!MediaRecorder.isTypeSupported("video/mp4"), + 'Should not support video/mp4'); + + try { + await setupGetUserMediaTestPrefs(); + const expectedMimeType = 'video/webm; codecs="vp8, opus"'; + const stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true}); + const [audioTrack] = stream.getAudioTracks(); + + // Expected event sequence should be: + // 1. start + // 2. error (from removed track) + // 3. dataavailable + // 4. stop + const mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, 'Stream should be provided on creation'); + + mediaRecorder.onstart = unexpected; + mediaRecorder.onerror = unexpected; + mediaRecorder.ondataavailable = unexpected; + mediaRecorder.onstop = unexpected; + + mediaRecorder.start(); + is(mediaRecorder.state, 'recording', 'state should be recording'); + is(mediaRecorder.mimeType, '', 'mimetype should be unset'); + + await new Promise(r => mediaRecorder.onstart = r); + mediaRecorder.onstart = unexpected; + ok(true, 'start event fired'); + is(mediaRecorder.mimeType, expectedMimeType, 'mimetype should be set'); + + // Trigger an error + stream.removeTrack(audioTrack); + + const err = await new Promise(r => mediaRecorder.onerror = r); + mediaRecorder.onerror = unexpected; + ok(true, 'error event fired'); + is(err.error.name, 'InvalidModificationError', + 'Error name should be InvalidModificationError.'); + ok(err.error.stack.includes('test_mediarecorder_onerror_pause.html'), + 'Events fired from onerror should include an error with a stack trace indicating ' + + 'an error in this test'); + is(mediaRecorder.mimeType, '', 'mimetype should be unset'); + is(mediaRecorder.state, 'inactive', 'state is inactive'); + + try { + mediaRecorder.pause(); + ok(false, 'pause should fire an exception if called on an inactive recorder'); + } catch(e) { + ok(e instanceof DOMException, 'pause should fire an exception ' + + 'if called on an inactive recorder'); + is(e.name, 'InvalidStateError', 'Exception name should be InvalidStateError'); + } + + const evt = await new Promise(r => mediaRecorder.ondataavailable = r); + mediaRecorder.ondataavailable = unexpected; + ok(true, 'dataavailable event fired'); + isnot(evt.data.size, 0, 'data size should not be zero'); + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.data.type, expectedMimeType, 'blob mimeType is set'); + + await new Promise(r => mediaRecorder.onstop = r); + mediaRecorder.onstop = unexpected; + ok(true, 'onstop event fired'); + is(mediaRecorder.state, 'inactive', 'state should be inactive'); + } catch (err) { + ok(false, `Unexpected error fired with: ${err}`); + } finally { + SimpleTest.finish(); + } +} + +SimpleTest.waitForExplicitFinish(); +window.onload = startTest; + +</script> +</head> +</html> diff --git a/dom/media/test/test_mediarecorder_pause_resume_video.html b/dom/media/test/test_mediarecorder_pause_resume_video.html new file mode 100644 index 0000000000..a9b3d6a034 --- /dev/null +++ b/dom/media/test/test_mediarecorder_pause_resume_video.html @@ -0,0 +1,130 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording doesn't record during pause</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> + <canvas id="video-src-canvas"></canvas> + <video id="recorded-video"></video> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + // Setup canvas and take a stream from it + let canvas = document.getElementById("video-src-canvas"); + + let canvas_size = 100; + let new_canvas_size = 50; + + canvas.width = canvas.height = canvas_size; + + let helper = new CaptureStreamTestHelper2D(100, 100); + helper.drawColor(canvas, helper.red); + + let canvasStream = canvas.captureStream(); + // Canvas set up + + // Check values for events + let numDataAvailabledRaised = 0; + // Recorded data that will be playback. + let blob; + + let mediaRecorder = new MediaRecorder(canvasStream); + is(mediaRecorder.stream, canvasStream, + "Media recorder stream = canvas stream at the start of recording"); + + mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired"); + + mediaRecorder.onerror = () => ok(false, "Recording failed"); + + mediaRecorder.ondataavailable = ev => { + info("Got 'dataavailable' event"); + ++numDataAvailabledRaised; + // Save recorded data for playback + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info("Got 'start' event"); + // We just want one frame encoded before we pause + mediaRecorder.pause(); + // We may rewrite this once we settle Bug 1363915, could listen for pause event instead + is(mediaRecorder.state, 'paused', 'Media recorder should be paused'); + + // Wait a while, then while paused draw blue at another size, then again + // green at the original size and resume. + let numberOfPaintsSincePause = 0; + window.requestAnimationFrame(function draw() { + numberOfPaintsSincePause++; + if (numberOfPaintsSincePause == 8) { + canvas.width = canvas.height = new_canvas_size; + helper.drawColor(canvas, helper.blue); + } else if (numberOfPaintsSincePause == 60) { + canvas.width = canvas.height = canvas_size; + helper.drawColor(canvas, helper.green); + } else if (numberOfPaintsSincePause == 68) { + // Waited 8 draws since changing canvas to green, should be safe to resume + mediaRecorder.resume(); + } else if (numberOfPaintsSincePause > 120) { + mediaRecorder.stop(); + return; // Early return, we don't want to request any more animation frames + } + window.requestAnimationFrame(draw); + }); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + is(mediaRecorder.state, 'inactive', 'Media recorder should be incative after stop'); + is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event"); + + ok(blob, "Should have gotten a data blob"); + let video = document.getElementById("recorded-video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + // Setup a check to make sure we don't play back any blue + let checkVideoHasNoBlue = () => { + if(helper.isPixel(helper.getPixel(video), helper.blue, 128)) { + ok(false, "Video should have no blue frames"); + // Remove handler so we don't spam the log + video.ontimeupdate = null; + } + }; + video.ontimeupdate = checkVideoHasNoBlue; + video.onerror = () => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + video.onended = () => { + ok(helper.isPixel(helper.getPixel(video), helper.green, 128), "Last frame should be green"); + SimpleTest.finish(); + }; + // The video will resize once it loads its metadata, only listen for resizes after that + video.onloadedmetadata = () => { + ok(video.videoWidth === canvas_size && video.videoHeight === canvas_size, + "video element should be same size as canvas once metadata is loaded"); + // We shouldn't have any resize events once the video is loaded + video.onresize = () => { + ok(false, "Should not have any resize events!"); + }; + }; + + video.play(); + }; + + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder should be recording"); +} + +SimpleTest.waitForExplicitFinish(); +startTest(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_playback_can_repeat.html b/dom/media/test/test_mediarecorder_playback_can_repeat.html new file mode 100644 index 0000000000..4cf2735088 --- /dev/null +++ b/dom/media/test/test_mediarecorder_playback_can_repeat.html @@ -0,0 +1,87 @@ +<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording creates videos that can playback more than once</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+ <canvas id="video-src-canvas"></canvas>
+ <video id="recorded-video" loop></video>
+</div>
+<script class="testbody" type="text/javascript">
+
+(async function() {
+ try {
+ SimpleTest.waitForExplicitFinish();
+ await startTest();
+ } catch(e) {
+ ok(false, `Got error msg '${e}'`);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+
+async function startTest() {
+ let canvas = document.getElementById("video-src-canvas");
+
+ let canvas_size = 100;
+ canvas.width = canvas.height = canvas_size;
+ let helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size);
+ helper.drawColor(canvas, helper.red);
+
+ let stream = canvas.captureStream();
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the beginning of recording");
+ new Promise(r => mediaRecorder.onerror = err => r(err)).then(
+ err => Promise.reject(`MediaRecorder: error unexpectedly fired. Code ${err.name}`));
+
+ mediaRecorder.start();
+ await new Promise(r => mediaRecorder.onstart = r);
+ info("Media recorder started");
+ // Feed some more data into the recorder so the output has a non trivial duration.
+ // This avoids the case where we have only 1 frame in the output, which breaks
+ // looping (this is expected as a WebM with 1 video frame has no duration).
+ let counter = 0;
+ let draw = () => {
+ counter++;
+ helper.drawColor(canvas, helper.blue);
+ if(counter > 2) {
+ mediaRecorder.stop();
+ return;
+ }
+ requestAnimationFrame(draw);
+ };
+ requestAnimationFrame(draw);
+
+ // When recorder is stopped get recorded data.
+ const data = await new Promise(r => mediaRecorder.ondataavailable = ev => r(ev));
+ info(`Media recorder get availiable data`);
+ const blob = data.data;
+
+ await new Promise(r => mediaRecorder.onstop = r);
+ info("Media recorder stopped");
+ ok(blob, "Should have gotten a data blob");
+ const video = document.getElementById("recorded-video");
+ new Promise(r => video.onerror = err => r(err)).then(
+ err => Promise.reject(`Video: error unexpectedly fired. Code ${err.code}`));
+ video.src = URL.createObjectURL(blob);
+ video.play();
+ await new Promise(r => video.onplaying = r);
+ for (let idx = 0; idx < 2; idx++) {
+ await new Promise(r => video.onseeking = r);
+ ok(true, `waiting until video finishes seeking`);
+ await new Promise(r => video.onseeked = r);
+ ok(true, "video finished seeked");
+ }
+ video.pause();
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_principals.html b/dom/media/test/test_mediarecorder_principals.html new file mode 100644 index 0000000000..00dcaa5a5b --- /dev/null +++ b/dom/media/test/test_mediarecorder_principals.html @@ -0,0 +1,132 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=489415 +--> +<head> + <title>Test for MediaRecorder Reaction to Principal Change</title> + <script type="application/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<div> + <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1018299">Test for MediaRecorder Principal Handling</a> +</div> + +<pre id="test"> +<script type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +let throwOutside = e => setTimeout(() => { throw e; }); + +// Loading data from a resource that changes origins while streaming should +// be detected by the media cache and result in a null principal so that the +// MediaRecorder usages below fail. + +// This test relies on midflight-redirect.sjs returning the the first quarter of +// the resource as a byte range response, and then hanging up, and when Firefox +// requests the remainder midflight-redirect.sjs serves a redirect to another origin. + +async function testPrincipals(resource) { + if (!resource) { + todo(false, "No types supported"); + return; + } + await testPrincipals1(resource); + await testPrincipals2(resource); +} + +function makeVideo() { + let video = document.createElement("video"); + video.preload = "metadata"; + video.controls = true; + document.body.appendChild(video); + return video; +} + +// First test: Load file from same-origin first, then get redirected to +// another origin before attempting to record stream. +async function testPrincipals1(resource) { + let video = makeVideo(); + video.src = + "http://mochi.test:8888/tests/dom/media/test/midflight-redirect.sjs" + + "?resource=" + resource.name + "&type=" + resource.type; + + let errorBarrier = once(video, "error"); + // Wait for the video to load to metadata. We can then start capturing. + // Must also handle the download bursting and hitting the error before we + // reach loadedmetadata. Normally we reach loadedmetadata first, but + // rarely we hit the redirect first. + await Promise.race([once(video, "loadedmetadata"), errorBarrier]); + + let rec = new MediaRecorder(video.mozCaptureStreamUntilEnded()); + video.play(); + + // Wait until we hit a playback error. This means our download has hit the redirect. + await errorBarrier; + + // Try to record, it should be blocked with a security error. + try { + rec.start(); + ok(false, "mediaRecorder.start() must throw SecurityError, but didn't throw at all"); + } catch (ex) { + is(ex.name, "SecurityError", "mediaRecorder.start() must throw SecurityError"); + } + removeNodeAndSource(video); +} + +// Second test: Load file from same-origin first, but record ASAP, before +// getting redirected to another origin. +async function testPrincipals2(resource) { + let video = makeVideo(); + video.src = + "http://mochi.test:8888/tests/dom/media/test/midflight-redirect.sjs" + + "?resource=" + resource.name + "&type=" + resource.type; + + // Wait for the video to load to metadata. We can then start capturing. + // Must also handle the download bursting and hitting the error before we + // reach loadedmetadata. Normally we reach loadedmetadata first, but + // rarely we hit the redirect first. + await Promise.race([once(video, "loadedmetadata"), once(video, "error")]); + + let ended = false; + once(video, "ended", () => ended = true); + + // Start capturing. It should work. + let rec; + let errorBarrier; + try { + rec = new MediaRecorder(video.mozCaptureStreamUntilEnded()); + errorBarrier = nextEvent(rec, "error"); + rec.start(); + ok(true, "mediaRecorder.start() should not throw here, and didn't"); + } catch (ex) { + ok(false, "mediaRecorder.start() unexpectedly threw " + ex.name + " (" + ex.message + ")"); + } + + // Play the video, this should result in a SecurityError on the recorder. + let hasStopped = once(rec, "stop"); + video.play(); + let error = (await errorBarrier).error; + is(error.name, "SecurityError", "mediaRecorder.onerror must fire SecurityError"); + ok(error.stack.includes('test_mediarecorder_principals.html'), + 'Events fired from onerror should include an error with a stack trace indicating ' + + 'an error in this test'); + is(ended, false, "Playback should not have reached end"); + await hasStopped; + is(ended, false, "Playback should definitely not have reached end"); + + removeNodeAndSource(video); +} + +testPrincipals({ name:"pixel_aspect_ratio.mp4", type:"video/mp4", duration:28 }) + .catch(e => throwOutside(e)) + .then(() => SimpleTest.finish()); + +</script> +</pre> + +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html b/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html new file mode 100644 index 0000000000..99f57f181e --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html @@ -0,0 +1,76 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record AudioContext with four channels</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + +<script class="testbody" type="text/javascript"> + +function startTest() { + var context = new AudioContext(); + var buffer = context.createBuffer(4, 80920, context.sampleRate); + for (var i = 0; i < 80920; ++i) { + for(var j = 0; j < 4; ++j) { + buffer.getChannelData(j)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate); + } + } + + var source = context.createBufferSource(); + source.buffer = buffer; + source.loop = true; + var dest = context.createMediaStreamDestination(); + var stopTriggered = false; + var onstopTriggered = false; + dest.channelCount = 4; + var expectedMimeType = 'audio/ogg; codecs=opus'; + source.channelCountMode = 'explicit'; + source.connect(dest); + var elem = document.createElement('audio'); + elem.srcObject = dest.stream; + source.start(0); + elem.play(); + let mMediaRecorder = new MediaRecorder(dest.stream); + mMediaRecorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + mMediaRecorder.onerror = function() { + ok(false, 'onerror unexpectedly fired'); + }; + + mMediaRecorder.onstop = function() { + ok(true, 'onstop fired successfully'); + is(mMediaRecorder.state, 'inactive', 'check recording status is inactive'); + onstopTriggered = true; + SimpleTest.finish(); + }; + mMediaRecorder.ondataavailable = function (e) { + ok(e.data.size > 0, 'check blob has data'); + is(e.data.type, expectedMimeType, 'blob should have expected mimetype'); + if (!stopTriggered) { + is(mMediaRecorder.mimeType, expectedMimeType, 'recorder should have expected mimetype'); + mMediaRecorder.stop(); + is(mMediaRecorder.mimeType, '', 'recorder should have reset its mimetype'); + stopTriggered = true; + } else if (onstopTriggered) { + ok(false, 'ondataavailable should come before onstop event'); + } + }; + try { + mMediaRecorder.start(1000); + is('recording', mMediaRecorder.state, "check record state recording"); + } catch (e) { + ok(false, 'Can t record audio context'); + } +} + +startTest(); +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_addtracked_stream.html b/dom/media/test/test_mediarecorder_record_addtracked_stream.html new file mode 100644 index 0000000000..046bf768ce --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_addtracked_stream.html @@ -0,0 +1,184 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder recording a constructed MediaStream</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <script src="/tests/dom/media/webrtc/tests/mochitests/head.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script> +SimpleTest.waitForExplicitFinish(); +runTestWhenReady(async () => { + const canvas = document.createElement("canvas"); + canvas.width = canvas.height = 100; + document.getElementById("content").appendChild(canvas); + + const helper = new CaptureStreamTestHelper2D(100, 100); + helper.drawColor(canvas, helper.red); + + const audioCtx = new AudioContext(); + const osc = audioCtx.createOscillator(); + osc.frequency.value = 1000; + osc.start(); + const dest = audioCtx.createMediaStreamDestination(); + osc.connect(dest); + + const stream = dest.stream; + + // Timeouts are experienced intermittently in Linux due to no sound in the + // destination. As a workaround wait for the source sound to arrive. + const sourceAnalyser = new AudioStreamAnalyser(audioCtx, stream); + const sourceAudioReady = sourceAnalyser.waitForAnalysisSuccess(array => { + const freq = osc.frequency.value; + const lowerFreq = freq / 2; + const upperFreq = freq + 1000; + const lowerAmp = array[sourceAnalyser.binIndexForFrequency(lowerFreq)]; + const freqAmp = array[sourceAnalyser.binIndexForFrequency(freq)]; + const upperAmp = array[sourceAnalyser.binIndexForFrequency(upperFreq)]; + info("Analysing source audio. " + + lowerFreq + ": " + lowerAmp + ", " + + freq + ": " + freqAmp + ", " + + upperFreq + ": " + upperAmp); + return lowerAmp < 50 && freqAmp > 200 && upperAmp < 50; + }); + await sourceAudioReady; + info("Source Audio content ok"); + + canvas.captureStream(0).getVideoTracks().forEach(t => stream.addTrack(t)); + + const blobs = []; + + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = constructed stream at the start of recording"); + + + mediaRecorder.ondataavailable = evt => { + info("ondataavailable fired"); + + is(mediaRecorder.state, "inactive", "Blob received after stopping"); + is(blobs.length, 0, "This is the first and only blob"); + ok(evt instanceof BlobEvent, + "Events fired from ondataavailable should be BlobEvent"); + is(evt.type, "dataavailable", + "Event type should dataavailable"); + ok(evt.data.size >= 0, + "Blob data size received is greater than or equal to zero"); + + blobs.push(evt.data); + }; + + const stopped = haveEvent(mediaRecorder, "stop", wait(5000, new Error("Timeout"))); + const stoppedNoErrors = Promise.all([ + stopped, + haveNoEvent(mediaRecorder, "warning", stopped), + haveNoEvent(mediaRecorder, "error", stopped) + ]); + + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder should be recording"); + + await haveEvent(mediaRecorder, "start", wait(5000, new Error("Timeout"))); + info("onstart fired"); + + // The recording can be too short to cause any checks with + // waitForAnalysisSuccess(). Waiting a bit here solves this. + await wait(500); + + is(mediaRecorder.state, "recording", + "Media recorder is recording before being stopped"); + mediaRecorder.stop(); + is(mediaRecorder.state, "inactive", + "Media recorder is inactive after being stopped"); + is(mediaRecorder.stream, stream, + "Media recorder stream = constructed stream post recording"); + + await stoppedNoErrors; + info("Got 'stop' event"); + + ok(blobs.length == 1, "Should have gotten one data blob"); + + // Clean up recording sources + osc.stop(); + stream.getTracks().forEach(t => t.stop()); + + // Sanity check the recording + const video = document.createElement("video"); + document.getElementById("content").appendChild(video); + video.id = "recorded-video"; + + const blob = new Blob(blobs); + ok(blob.size > 0, "Recorded blob should contain data"); + + video.src = URL.createObjectURL(blob); + video.preload = "metadata"; + + info("Waiting for metadata to be preloaded"); + + await haveEvent(video, "loadedmetadata", wait(5000, new Error("Timeout"))); + info("Playback of recording loaded metadata"); + + const recordingStream = video.mozCaptureStream(); + is(recordingStream.getVideoTracks().length, 1, + "Recording should have one video track"); + is(recordingStream.getAudioTracks().length, 1, + "Recording should have one audio track"); + + const ended = haveEvent(video, "ended", wait(5000, new Error("Timeout"))); + const endedNoError = Promise.all([ + ended, + haveNoEvent(video, "error", ended), + ]); + + const analyser = new AudioStreamAnalyser(audioCtx, recordingStream); + const audioReady = analyser.waitForAnalysisSuccess(array => { + const freq = osc.frequency.value; + const lowerFreq = freq / 2; + const upperFreq = freq + 1000; + const lowerAmp = array[analyser.binIndexForFrequency(lowerFreq)]; + const freqAmp = array[analyser.binIndexForFrequency(freq)]; + const upperAmp = array[analyser.binIndexForFrequency(upperFreq)]; + info("Analysing audio. " + + lowerFreq + ": " + lowerAmp + ", " + + freq + ": " + freqAmp + ", " + + upperFreq + ": " + upperAmp); + return lowerAmp < 50 && freqAmp > 200 && upperAmp < 50; + }, endedNoError.then(() => new Error("Audio check failed"))); + + const videoReady = helper.pixelMustBecome( + video, helper.red, { + threshold: 128, + infoString: "Should become red", + cancelPromise: endedNoError.then(() => new Error("Video check failed")), + }); + + video.play(); + + try { + await endedNoError; + } finally { + analyser.disconnect(); + let url = video.src; + video.src = ""; + URL.revokeObjectURL(url); + } + + info("Playback of recording ended without error"); + + await audioReady; + info("Audio content ok"); + + await videoReady; + info("Video content ok"); + + SimpleTest.finish(); +}); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_audiocontext.html b/dom/media/test/test_mediarecorder_record_audiocontext.html new file mode 100644 index 0000000000..686faaeb6b --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_audiocontext.html @@ -0,0 +1,65 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record AudioContext</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + +<script class="testbody" type="text/javascript"> + +function startTest() { + var context = new AudioContext(); + var buffer = context.createBuffer(1, 80920, context.sampleRate); + for (var i = 0; i < 80920; ++i) { + buffer.getChannelData(0)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate); + } + + var source = context.createBufferSource(); + source.buffer = buffer; + source.loop = true; + + var dest = context.createMediaStreamDestination(); + source.connect(dest); + var elem = document.createElement('audio'); + elem.srcObject = dest.stream; + source.start(0); + elem.play(); + let mMediaRecorder = new MediaRecorder(dest.stream); + mMediaRecorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + mMediaRecorder.onerror = function() { + ok(false, 'onerror unexpectedly fired'); + }; + + mMediaRecorder.onstop = function() { + ok(true, 'onstop fired successfully'); + is(mMediaRecorder.state, 'inactive', 'check recording status is inactive'); + SimpleTest.finish(); + }; + mMediaRecorder.ondataavailable = function (e) { + if (mMediaRecorder.state == 'recording') { + is(mMediaRecorder.mimeType, 'audio/ogg; codecs=opus', 'Expected MediaRecorder mimetype'); + is(e.data.type, 'audio/ogg; codecs=opus', 'Expected Blob mimetype'); + ok(e.data.size > 0, 'check blob has data'); + mMediaRecorder.stop(); + } + }; + try { + mMediaRecorder.start(1000); + is('recording', mMediaRecorder.state, "check record state recording"); + } catch (e) { + ok(false, 'Can t record audio context'); + } +} + +startTest(); +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html b/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html new file mode 100644 index 0000000000..4128702aef --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html @@ -0,0 +1,24 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>capture for possible memory leak when record AudioContext</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=973765">Mozill +a Bug 973765</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + // This test case want to capture the memory leak if exit the browser after running those script. + var ac = new window.AudioContext(); + var destStream = ac.createMediaStreamDestination().stream; + var recorder = new MediaRecorder(destStream); + recorder.start(1000); + is(recorder.state, 'recording', 'Media recorder should be recording'); + is(recorder.stream, destStream, + 'Media recorder stream = element stream at the start of recording'); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_audionode.html b/dom/media/test/test_mediarecorder_record_audionode.html new file mode 100644 index 0000000000..8a57437b81 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_audionode.html @@ -0,0 +1,135 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record AudioContext Node</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=968109">Mozilla Bug 968109</a> + +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function setUpSource(contextType, nodeType) { + // Use contextType to choose offline or real-time context. + const context = contextType == "offline"? + new OfflineAudioContext(2 , 80920, 44100) : new AudioContext(); + const buffer = context.createBuffer(2, 80920, context.sampleRate); + for (let i = 0; i < 80920; ++i) { + buffer.getChannelData(0)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate); + buffer.getChannelData(1)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate); + } + + const source = context.createBufferSource(); + source.buffer = buffer; + source.loop = true; + + source.start(0); + + // nodeType decides which node in graph should be the source of recording. + let node; + if (nodeType == "source") { + node = source; + } else if (nodeType == "splitter") { + const splitter = context.createChannelSplitter(); + source.connect(splitter); + node = splitter; + } else if (nodeType == "destination") { + source.connect(context.destination); + node = context.destination; + } + // Explicitly start offline context. + if (contextType == "offline") { + context.startRendering(); + } + + return node; +} + +async function testRecord(source, mimeType) { + const isOffline = source.context instanceof OfflineAudioContext; + const recorder = new MediaRecorder(source, 0, {mimeType}); + is(recorder.mimeType, mimeType, "Mime type is set"); + const extendedMimeType = `${mimeType || "audio/ogg"}; codecs=opus`; + + recorder.onwarning = () => ok(false, "should not fire onwarning"); + recorder.onerror = () => ok(false, "should not fire onerror"); + if (!isOffline) { + recorder.onstop = () => ok(false, "should not fire stop yet"); + } + + recorder.start(1000); + is("recording", recorder.state, "state should become recording after calling start()"); + is(recorder.mimeType, mimeType, "Mime type is not changed by start()"); + + await new Promise(r => recorder.onstart = r); + is(recorder.mimeType, extendedMimeType, "Mime type is fully defined"); + + const chunks = []; + let {data} = await new Promise(r => recorder.ondataavailable = r); + if (!isOffline) { + is(recorder.state, "recording", "Expected to still be recording"); + } + is(data.type, extendedMimeType, "Blob has fully defined mimetype"); + isnot(data.size, 0, "should get data and its length should be > 0"); + chunks.push(data); + + if (isOffline) { + await new Promise(r => recorder.onstop = r); + is(recorder.state, "inactive", "Offline context should end by itself"); + } else { + is(recorder.state, "recording", "Expected to still be recording"); + recorder.stop(); + ({data} = await new Promise(r => recorder.ondataavailable = r)); + is(recorder.state, "inactive", "Expected to be inactive after last blob"); + isnot(data.size, 0, "should get data and its length should be > 0"); + chunks.push(data); + + await new Promise(r => recorder.onstop = r); + is(recorder.state, "inactive", "state should remain inactive after stop event"); + } + return new Blob(chunks, {type: chunks[0].type}); +} + +addLoadEvent(async () => { + const src = setUpSource(); + let didThrow = false; + try { + new MediaRecorder(src); + } catch (e) { + didThrow = true; + } + ok(didThrow, "MediaRecorder(AudioNode) should be hidden behind a pref"); + + await SpecialPowers.pushPrefEnv({set: [ + ["media.recorder.audio_node.enabled", true], + ]}); + + // Test with various context and source node types. + for (const mimeType of [ + "audio/ogg", + "audio/webm", + "video/webm", + "", + ]) { + for (const {context, node} of [ + {context: "", node: "source"}, + {context: "", node: "splitter"}, + {context: "offline", node: "destination"}, + ]) { + info(`Testing recording ${context || "regular"} context and ${node} ` + + `node with mimeType '${mimeType}'`); + await testRecord(setUpSource(context, node), mimeType); + } + } + + SimpleTest.finish(); +}); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_canvas_captureStream.html b/dom/media/test/test_mediarecorder_record_canvas_captureStream.html new file mode 100644 index 0000000000..0b6cd6dbb5 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_canvas_captureStream.html @@ -0,0 +1,75 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording canvas stream</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + var canvas = document.createElement("canvas"); + canvas.width = canvas.height = 100; + document.getElementById("content").appendChild(canvas); + + var helper = new CaptureStreamTestHelper2D(100, 100); + helper.drawColor(canvas, helper.red); + + var stream = canvas.captureStream(0); + + var blob; + + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = canvas stream at the start of recording"); + + mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired"); + + mediaRecorder.onerror = () => ok(false, "Recording failed"); + + mediaRecorder.ondataavailable = ev => { + is(blob, undefined, "Should only get one dataavailable event"); + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info("Got 'start' event"); + // We just want one frame encoded, to see that the recorder produces something readable. + mediaRecorder.stop(); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + ok(blob, "Should have gotten a data blob"); + + var video = document.createElement("video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + video.play(); + video.onerror = err => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + document.getElementById("content").appendChild(video); + helper.pixelMustBecome(video, helper.red, { + threshold: 128, + infoString: "Should become red" + }).then(SimpleTest.finish); + }; + + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder should be recording"); +} + +SimpleTest.waitForExplicitFinish(); +startTest(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_changing_video_resolution.html b/dom/media/test/test_mediarecorder_record_changing_video_resolution.html new file mode 100644 index 0000000000..d6354ee5a1 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_changing_video_resolution.html @@ -0,0 +1,174 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording canvas stream that dynamically changes resolution</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + let canvas = document.createElement("canvas"); + const resolution_change = [ + {width: 100, height: 100, color: "red"}, + {width: 150, height: 150, color: "blue"}, + {width: 100, height: 100, color: "red"}, + ]; + canvas.width = resolution_change[0].width; + canvas.height = resolution_change[0].height; + + let ctx = canvas.getContext("2d"); + ctx.fillStyle = resolution_change[0].color; + ctx.fillRect(0, 0, canvas.width, canvas.height); + + // The recorded stream coming from canvas. + let stream = canvas.captureStream(); + + // Check values for events + let numDataAvailabledRaised = 0; + let numResizeRaised = 0; + // Recorded data that will be playback. + let blob; + + // Media recorder for VP8 and canvas stream. + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = canvas stream at the start of recording"); + + // Not expected events. + mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired"); + mediaRecorder.onerror = err => { + ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name); + SimpleTest.finish(); + }; + + // When recorder is stopped get recorded data. + mediaRecorder.ondataavailable = ev => { + info("Got 'dataavailable' event"); + ++numDataAvailabledRaised; + is(blob, undefined, "Should only get one dataavailable event"); + // Save recorded data for playback + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info('onstart fired successfully'); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + is(numDataAvailabledRaised, 1, "Should have gotten 1 dataavailable event"); + // Playback stream and verify resolution changes. + ok(blob, "Should have gotten a data blob"); + + let video = document.createElement("video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + video.preload = "metadata"; + + video.onerror = err => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + + // Check that the encoded frames have the correct sizes. + video.onresize = function() { + if (numResizeRaised < resolution_change.length) { + is(video.videoWidth, resolution_change[numResizeRaised].width, + "onresize width should be as expected"); + is(video.videoHeight, resolution_change[numResizeRaised].height, + "onresize height should be as expected"); + } else { + ok(false, "Got more resize events than expected"); + } + ++numResizeRaised; + }; + + video.onloadedmetadata = function() { + info("loadedmetadata"); + seekThroughFrames(); + }; + + video.onended = function() { + is(numResizeRaised, resolution_change.length, "Expected number of resize events"); + SimpleTest.finish(); + // This shouldn't be needed, however video.ended may not be set after + // seeking to the final frame. This can result in seekToNextFrame being + // called again by seekThroughFrames and onended being invoked again, + // resulting in multiple finish() calls. + // FIXME: https://bugzilla.mozilla.org/show_bug.cgi?id=1386489 + video.onended = null; + }; + + document.getElementById("content").appendChild(video); + + function seekThroughFrames() { + info("Seeking to next frame"); + video.seekToNextFrame() + .then(() => { + info("Seeking to next frame finished. width=" + video.videoWidth + + ", height=" + video.videoHeight); + + if (video.ended) { + return; + } + + // After seeking finished we queue the next seek task on the event + // loop so it gets in the same queue as the "resize" events. + setTimeout(seekThroughFrames, 0); + }) + .catch(error => { + ok(false, "seekToNextFrame rejected: " + error); + }); + } + }; + + // Start here by stream recorder. + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder should be recording"); + requestAnimationFrame(draw); + + // Change resolution in every frame + // Stop recorder on last frame + let countFrames = 0; + let previous_time = performance.now(); + function draw(timestamp) { + if (timestamp - previous_time < 100) { + requestAnimationFrame(draw); + return; + } + previous_time = timestamp; + + if (countFrames == resolution_change.length) { + mediaRecorder.stop(); + return; + } + + canvas.width = resolution_change[countFrames].width; + canvas.height = resolution_change[countFrames].height; + ctx.fillStyle = resolution_change[countFrames].color; + // Resize and draw canvas + ctx.fillRect(0, 0, canvas.width, canvas.height); + // Register draw to be called on next rendering + requestAnimationFrame(draw); + countFrames++; + } +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.seekToNextFrame.enabled", true ], + ["media.video-queue.send-to-compositor-size", 1] + ] + }, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_downsize_resolution.html b/dom/media/test/test_mediarecorder_record_downsize_resolution.html new file mode 100644 index 0000000000..f9422a3897 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_downsize_resolution.html @@ -0,0 +1,148 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording canvas dynamically changes to greater resolution</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + var canvas = document.createElement("canvas"); + var canvas_size = 100; + var new_canvas_size = 50; + canvas.width = canvas.height = canvas_size; + + var helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size); + helper.drawColor(canvas, helper.red); + + // The recorded stream coming from canvas. + var stream = canvas.captureStream(); + + // Check values for events + var numDataAvailabledRaised = 0; + var numResizeRaised = 0; + // Recorded data that will be playback. + var blob; + + // Media recorder for VP8 and canvas stream. + var mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = canvas stream at the beginning of recording"); + + // Not expected events. + mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired"); + mediaRecorder.onerror = err => { + ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name); + SimpleTest.finish(); + } + + // When recorder is stopped get recorded data. + mediaRecorder.ondataavailable = ev => { + info("Got 'dataavailable' event"); + ++numDataAvailabledRaised; + is(blob, undefined, "On dataavailable event blob is undefined"); + // Save recorded data for playback + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info('onstart fired successfully'); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event"); + + // Playback stream and verify resolution changes. + ok(blob, "Should have gotten a data blob"); + var video = document.createElement("video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + video.onerror = err => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + + // Check here that resize is correct in the playback stream. + video.onresize = function() { + ++numResizeRaised; + if (numResizeRaised == 1) { + is(this.videoWidth, canvas_size, "1st resize event original width"); + is(this.videoHeight, canvas_size, "1st resize event original height "); + } else if (numResizeRaised == 2) { + is(this.videoWidth, new_canvas_size, "2nd resize event new width"); + is(this.videoHeight, new_canvas_size, "2nd resize event new height"); + } else { + ok(false, "Only 2 numResizeRaised events are expected"); + } + }; + + video.onended = () => { + is(numResizeRaised, 2, "Expected 2 resize event"); + }; + document.getElementById("content").appendChild(video); + video.play(); + + // Check last color + helper.pixelMustBecome(video, helper.red, { + threshold: 128, + infoString: "Should become red", + }).then(() => { + video.onresize = {}; + video.onended = {}; + SimpleTest.finish(); + }); + }; + + // Start here by stream recorder. + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder started"); + requestAnimationFrame(draw); + + // Change resolution every 100ms + var countFrames=0; + var previous_time = performance.now(); + function draw(timestamp) { + if (timestamp - previous_time < 100) { + requestAnimationFrame(draw); + return; + } + previous_time = timestamp; + + var size = 0; + var color = ""; + if (countFrames < 1) { + // Initial size + size = canvas_size; + color = helper.blue; + } else if (countFrames < 2) { + // upsize + size = new_canvas_size; + color = helper.red; + } else { + // Stop recoredr on last frame + mediaRecorder.stop(); + return; + } + // Resize and draw canvas + canvas.width = canvas.height = size; + helper.drawColor(canvas, color); + // Register next draw on every call + requestAnimationFrame(draw); + countFrames++; + } +} + +SimpleTest.waitForExplicitFinish(); +startTest(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_getdata_afterstart.html b/dom/media/test/test_mediarecorder_record_getdata_afterstart.html new file mode 100644 index 0000000000..3b181ed8db --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_getdata_afterstart.html @@ -0,0 +1,81 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Bug 951008 Test MediaRecorder Record has start event</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + + var element = document.createElement('audio'); + var hasonstart = false; + var hasondataavailable = false; + var mMediaRecorder; + + element.token = token; + manager.started(token); + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStream(); + + mMediaRecorder = new MediaRecorder(element.stream); + is(mMediaRecorder.mimeType, '', 'Expected MediaRecorder mimetype'); + mMediaRecorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + mMediaRecorder.onerror = function() { + ok(false, 'onerror unexpectedly fired'); + }; + + mMediaRecorder.onstart = function() { + info('onstart fired successfully'); + hasonstart = true; + is(mMediaRecorder.mimeType, 'audio/ogg; codecs=opus', + "MediaRecorder mimetype as expected"); + mMediaRecorder.requestData(); + }; + + mMediaRecorder.onstop = function() { + info('onstop fired successfully'); + ok(hasondataavailable, "should have ondataavailable before onstop"); + is(mMediaRecorder.state, 'inactive', 'check recording status is inactive'); + SimpleTest.finish(); + }; + + mMediaRecorder.ondataavailable = function (e) { + info('ondataavailable fired successfully'); + if (mMediaRecorder.state == 'recording') { + hasondataavailable = true; + ok(hasonstart, "should have had start event first"); + is(e.data.type, mMediaRecorder.mimeType, + "blob's mimeType matches the recorder's"); + mMediaRecorder.stop(); + } + }; + + // Start recording once metadata are parsed. + element.onloadedmetadata = function() { + element.oncanplaythrough = null; + mMediaRecorder.start(0); + is(mMediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mMediaRecorder.stream, element.stream, + 'Media recorder stream = element stream at the start of recording'); + }; + + element.play(); +} + +manager.runTests(gMediaRecorderTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html b/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html new file mode 100644 index 0000000000..961a9644b2 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html @@ -0,0 +1,94 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record gUM video with Timeslice</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<pre id="test"> +<div id="content" style="display: none"> +</div> +<script class="testbody" type="text/javascript"> + +async function startTest() { + try { + await setupGetUserMediaTestPrefs(); + let stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true}); + let dataAvailableCount = 0; + let onDataAvailableFirst = false; + const expectedMimeType = 'video/webm; codecs="vp8, opus"'; + + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + 'Media recorder stream = element stream at the start of recording'); + mediaRecorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + mediaRecorder.onerror = function() { + ok(false, 'onerror unexpectedly fired'); + }; + + mediaRecorder.onstop = function() { + ok(false, 'Unexpected onstop callback fired'); + }; + + mediaRecorder.onstart = function() { + is(mediaRecorder.mimeType, expectedMimeType, 'Expected mime type'); + }; + + mediaRecorder.ondataavailable = function (evt) { + info('ondataavailable fired'); + dataAvailableCount++; + + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + ok(evt.data.size >= 0, + 'Blob data size ' + evt.data.size + ' received is greater than or equal to zero'); + is(evt.data.type, expectedMimeType, 'Expected blob mime type'); + + // We'll stop recording upon the 1st blob being received + if (dataAvailableCount === 1) { + mediaRecorder.onstop = function (event) { + info('onstop fired'); + + if (!onDataAvailableFirst) { + ok(false, 'onstop unexpectedly fired before ondataavailable'); + } + + ok(true, 'onstop fired successfully'); + is(mediaRecorder.state, 'inactive', + 'check recording status is inactive'); + SimpleTest.finish(); + }; + + mediaRecorder.stop(); + is(mediaRecorder.state, 'inactive', + 'Media recorder is inactive after being stopped'); + + } else if (dataAvailableCount === 2) { + // Ensure we've received at least two ondataavailable events before + // onstop + onDataAvailableFirst = true; + } + }; + + mediaRecorder.start(250); + is(mediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mediaRecorder.mimeType, '', 'Expected mime type'); + } catch (err) { + ok(false, 'Unexpected error fired with: ' + err); + SimpleTest.finish(); + } +} + +SimpleTest.waitForExplicitFinish(); +startTest(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html b/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html new file mode 100644 index 0000000000..af607280f6 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html @@ -0,0 +1,100 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record gUM video with Timeslice, and playback of mixed memory and file blobs</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<pre id="test"> +<script type="text/javascript"> +function unexpected({type}) { + ok(false, `${type} unexpectedly fired`); +} + +(async () => { + SimpleTest.waitForExplicitFinish(); + let blobUrl = null; + let stream = null; + try { + // This is the memory limit per blob. If a blob is larger than this, + // MediaRecorder will put it in a file. For this test we need to get at + // least one blob under, and one blob over the limit. + const memoryLimit = 3000; + await SpecialPowers.pushPrefEnv({set: [ + ["media.recorder.max_memory", memoryLimit], + ]}); + // We always use fake devices since the loopback ones don't make enough + // pixels change per frame to make the encoded frames large enough. + await pushGetUserMediaTestPrefs({fakeAudio: true, fakeVideo: true}); + stream = await navigator.mediaDevices.getUserMedia( + {audio: true, video: true}); + const blobs = []; + + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = element stream at the start of recording"); + mediaRecorder.start(); + mediaRecorder.addEventListener("warning", unexpected); + mediaRecorder.addEventListener("error", unexpected); + mediaRecorder.addEventListener("stop", unexpected); + await new Promise(r => mediaRecorder.onstart = r); + + for (let hasMemory = false; !hasMemory;) { + mediaRecorder.requestData(); + const {data} = await new Promise(r => mediaRecorder.ondataavailable = r); + blobs.push(data); + ok(data.size < memoryLimit, "Blob should be small enough at start"); + hasMemory = data.size > 0 && data.size < memoryLimit; + info(`Blob is ${data.size} bytes.${hasMemory ? " In memory." : ""}`); + } + info("Got a memory blob"); + + SimpleTest.requestFlakyTimeout("Wait for file blob"); + for (let hasFile = false, waitTimeMs = 500; !hasFile; waitTimeMs *= 4) { + info(`Waiting ${waitTimeMs/1000} seconds for file blob`); + await new Promise(r => setTimeout(r, waitTimeMs)); + mediaRecorder.requestData(); + const {data} = await new Promise(r => mediaRecorder.ondataavailable = r); + blobs.push(data); + hasFile = data.size > memoryLimit; + info(`Blob is ${data.size} bytes. In ${hasFile ? "file" : "memory"}.`); + } + info("Got a file blob"); + + mediaRecorder.stop(); + const {data} = await new Promise(r => mediaRecorder.ondataavailable = r); + blobs.push(data); + mediaRecorder.removeEventListener("stop", unexpected); + await new Promise(r => mediaRecorder.onstop = r); + + const video = document.createElement("video"); + const blob = new Blob(blobs); + blobUrl = URL.createObjectURL(blob); + video.src = blobUrl; + info(`Starting playback. Blob-size=${blob.size}`); + video.play(); + + await Promise.race([ + new Promise(res => video.onended = res), + new Promise((res, rej) => video.onerror = () => rej(video.error.message)), + ]); + } catch (e) { + ok(false, e); + } finally { + if (stream) { + for (const t of stream.getTracks()) { + t.stop(); + } + } + if (blobUrl) { + URL.revokeObjectURL(blobUrl); + } + SimpleTest.finish(); + } +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_immediate_stop.html b/dom/media/test/test_mediarecorder_record_immediate_stop.html new file mode 100644 index 0000000000..8ed7c321a0 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_immediate_stop.html @@ -0,0 +1,115 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Immediate Stop</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +/** + * Stops the media recorder immediately after starting the recorder. This test + * verifies whether the media recorder can handle this scenario nicely. The + * return blob size should be greater than zero, but its duration would be zero + * length when play. + */ +function startTest(test, token) { + var element = document.createElement('audio'); + var expectedMimeType = test.type; + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStreamUntilEnded(); + + var mediaRecorder = + new MediaRecorder(element.stream, {mimeType: expectedMimeType}); + var onStopFired = false; + var onDataAvailableFired = false; + + mediaRecorder.onerror = function () { + ok(false, 'Unexpected onerror callback fired'); + }; + + mediaRecorder.onwarning = function () { + ok(false, 'Unexpected onwarning callback fired'); + }; + + // This handler verifies that only a single onstop event handler is fired. + mediaRecorder.onstop = function () { + if (onStopFired) { + ok(false, 'onstop unexpectedly fired more than once'); + } else { + onStopFired = true; + + // ondataavailable should always fire before onstop + if (onDataAvailableFired) { + manager.finished(token); + } else { + ok(false, 'onstop fired without an ondataavailable event first'); + } + } + }; + + // This handler verifies that only a single ondataavailable event handler + // is fired with the blob generated having greater than zero size + // and a correct mime type. + mediaRecorder.ondataavailable = function (evt) { + if (onDataAvailableFired) { + ok(false, 'ondataavailable unexpectedly fired more than once'); + } else { + onDataAvailableFired = true; + + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + + // The initialization of encoder can be cancelled. + // On some platforms, the stop method may run after media stream track + // available, so the blob can contain the header data. + is(evt.data.type, expectedMimeType, + 'Blob data received and should have mime type'); + is(mediaRecorder.mimeType, expectedMimeType, + 'Media Recorder mime type in ondataavailable = ' + expectedMimeType); + ok(evt.data.size >= 0, 'Blob size can not be negative'); + + // onstop should not have fired before ondataavailable + if (onStopFired) { + ok(false, 'ondataavailable unexpectedly fired later than onstop'); + manager.finished(token); + } + } + }; + + // This handler completes a start and stop of recording and verifies + // respective media recorder state. + element.onloadedmetadata = function () { + element.onloadedmetadata = null; + element.play(); + mediaRecorder.start(); + is(mediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mediaRecorder.stream, element.stream, + 'Media recorder stream = element stream at the start of recording'); + + mediaRecorder.stop(); + is(mediaRecorder.state, 'inactive', + 'Media recorder is inactive after being stopped'); + is(mediaRecorder.stream, element.stream, + 'Media recorder stream = element stream post recording'); + }; + + element.preload = "metadata"; +} + +manager.runTests(gMediaRecorderTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_no_timeslice.html b/dom/media/test/test_mediarecorder_record_no_timeslice.html new file mode 100644 index 0000000000..6ed148a108 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_no_timeslice.html @@ -0,0 +1,106 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record No Timeslice</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +/** + * Starts a test on every media recorder file included to check that a + * stream derived from the file can be recorded with no time slice provided. + */ +function startTest(test, token) { + var element = document.createElement('audio'); + var expectedMimeType = test.type; + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStreamUntilEnded(); + + var mediaRecorder = + new MediaRecorder(element.stream, {mimeType: expectedMimeType}); + var onStopFired = false; + var onDataAvailableFired = false; + + mediaRecorder.onerror = function () { + ok(false, 'Unexpected onerror callback fired'); + }; + + mediaRecorder.onwarning = function () { + ok(false, 'Unexpected onwarning callback fired'); + }; + + // This handler verifies that only a single onstop event handler is fired. + mediaRecorder.onstop = function () { + if (onStopFired) { + ok(false, 'onstop unexpectedly fired more than once'); + } else { + onStopFired = true; + + // ondataavailable should always fire before onstop + if (onDataAvailableFired) { + ok(true, 'onstop fired after ondataavailable'); + manager.finished(token); + } else { + ok(false, 'onstop fired without an ondataavailable event first'); + } + } + }; + + // This handler verifies that only a single ondataavailable event handler + // is fired with the blob generated having greater than zero size + // and a correct mime type. + mediaRecorder.ondataavailable = function (evt) { + if (onDataAvailableFired) { + ok(false, 'ondataavailable unexpectedly fired more than once'); + } else { + onDataAvailableFired = true; + + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + ok(evt.data.size > 0, + 'Blob data received should be greater than zero'); + is(evt.data.type, expectedMimeType, + 'Blob data received should have type = ' + expectedMimeType); + + is(mediaRecorder.mimeType, expectedMimeType, + 'Mime type in ondataavailable = ' + expectedMimeType); + + // onstop should not have fired before ondataavailable + if (onStopFired) { + ok(false, 'ondataavailable unexpectedly fired later than onstop'); + manager.finished(token); + } + } + }; + + element.preload = "metadata"; + + element.onloadedmetadata = function () { + element.onloadedmetadata = null; + mediaRecorder.start(); + is(mediaRecorder.state, 'recording', + 'Media recorder should be recording'); + is(mediaRecorder.stream, element.stream, + 'Media recorder stream = element stream at the start of recording'); + + element.play(); + } +} + +manager.runTests(gMediaRecorderTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_session.html b/dom/media/test/test_mediarecorder_record_session.html new file mode 100644 index 0000000000..88795d82d0 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_session.html @@ -0,0 +1,75 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=909670 +--> +<head> + <meta charset="utf-8"> + <title>Test for Media Recoder recording session</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function startTest(test, token) { + var element = document.createElement('audio'); + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStream(); + + var mStopCount = 0; + // Start and stop recording session three times continuously. + var mExpectStopCount = 3; + var mediaRecorder = new MediaRecorder(element.stream); + + // Stop callback. + // Suppose to receive mExpectStopCount + mediaRecorder.onstop = function stopCallback() { + mStopCount++; + + info("MediaRecorder.onstop callback: (" + mStopCount + ")"); + + if (mExpectStopCount === mStopCount) + { + manager.finished(token); + } + } + + // data avaliable. + mediaRecorder.ondataavailable = function(evt) {} + + mediaRecorder.onerror = function(err) { + ok(false, 'Unexpected error fired with:' + err); + } + + mediaRecorder.onwarning = function() { + ok(false, 'Unexpected warning fired'); + } + + element.preload = "metadata"; + + element.onloadedmetadata = function () { + element.onloadedmetadata = null; + element.play(); + for (var i = 0; i < mExpectStopCount; i++) { + mediaRecorder.start(1000); + mediaRecorder.stop(); + } + } + +} + +manager.runTests(gMediaRecorderTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_startstopstart.html b/dom/media/test/test_mediarecorder_record_startstopstart.html new file mode 100644 index 0000000000..b4cc62c709 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_startstopstart.html @@ -0,0 +1,75 @@ + <!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder crash on sequence start stop start method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content" style="display: none"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + var ac = new window.AudioContext(); + var dest = ac.createMediaStreamDestination(); + var recorder = new MediaRecorder(dest.stream); + var stopCount = 0; + var dataavailable = 0; + var expectedMimeType = 'audio/ogg; codecs=opus'; + recorder.onstop = function (e) { + info('onstop fired'); + is(recorder.stream, dest.stream, + 'Media recorder stream = element stream post recording'); + stopCount++; + if (stopCount == 2) { + if (dataavailable >= 2) { + SimpleTest.finish(); + } else { + ok(false, 'Should have at least two dataavailable events'); + } + } + } + recorder.ondataavailable = function (evt) { + info('ondataavailable fired'); + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + // If script runs slower, it may generate header data in blob from encoder + if (evt.data.size > 0) { + info('blob size = ' + evt.data.size); + is(evt.data.type, expectedMimeType, + 'Blob data received should have type = ' + expectedMimeType); + } + dataavailable++; + } + recorder.onerror = function (e) { + ok(false, 'it should execute normally without exception'); + } + recorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + recorder.start(2000); + is(recorder.state, 'recording', 'Media recorder should be recording'); + recorder.stop(); + is(recorder.state, 'inactive', 'check recording status is inactive'); + recorder.start(10000); // This bug would crash on this line without this fix. + is(recorder.state, 'recording', 'check recording status is recording'); + // Simulate delay stop, only delay stop no no stop can trigger crash. + setTimeout(function() { + recorder.stop(); + is(recorder.state, 'inactive','check recording status is recording'); + }, 1000); +} + +SimpleTest.requestFlakyTimeout("untriaged"); +startTest(); +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +</body> +</html> + diff --git a/dom/media/test/test_mediarecorder_record_timeslice.html b/dom/media/test/test_mediarecorder_record_timeslice.html new file mode 100644 index 0000000000..3e547e77b4 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_timeslice.html @@ -0,0 +1,105 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Record Timeslice</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +/** + * Starts a test on every media recorder file included to check that a stream + * derived from the file can be recorded with a timeslice provided + */ +function startTest(test, token) { + var element = document.createElement('audio'); + var expectedMimeType = test.type; + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.preload = "auto"; + + // Set up MediaRecorder once loadedmetadata fires and tracks are available. + element.onloadedmetadata = function() { + element.onloadedmetadata = null; + + const stream = element.mozCaptureStream(); + const mediaRecorder = + new MediaRecorder(stream, {mimeType: expectedMimeType}); + + mediaRecorder.onerror = function () { + ok(false, 'Unexpected onerror callback fired'); + }; + + mediaRecorder.onwarning = function () { + ok(false, 'Unexpected onwarning callback fired'); + }; + + mediaRecorder.onstop = function () { + ok(false, 'Unexpected onstop callback fired'); + }; + + var dataAvailableCount = 0; + var onDataAvailableFirst = false; + + // This handler fires every 250ms to generate a blob. + mediaRecorder.ondataavailable = function (evt) { + info('ondataavailable fired'); + dataAvailableCount++; + + ok(evt instanceof BlobEvent, + 'Events fired from ondataavailable should be BlobEvent'); + is(evt.type, 'dataavailable', + 'Event type should dataavailable'); + ok(evt.data.size >= 0, + 'Blob data size received is greater than or equal to zero'); + + is(evt.data.type, expectedMimeType, + 'Blob data received should have type = ' + expectedMimeType); + is(mediaRecorder.mimeType, expectedMimeType, + 'Mime type in ondataavailable = ' + mediaRecorder.mimeType); + + // We'll stop recording upon the 1st blob being received + if (dataAvailableCount === 1) { + mediaRecorder.onstop = function (event) { + info('onstop fired'); + + if (!onDataAvailableFirst) { + ok(false, 'onstop unexpectedly fired before ondataavailable'); + } + element.pause(); + manager.finished(token); + }; + + mediaRecorder.stop(); + is(mediaRecorder.state, 'inactive', + 'Media recorder is inactive after being stopped'); + is(mediaRecorder.stream, stream, + 'Media recorder stream = element stream post recording'); + + } else if (dataAvailableCount === 2) { + // Ensure we've received at least two ondataavailable events before onstop + onDataAvailableFirst = true; + } + }; + + mediaRecorder.start(1); + element.play(); + is(mediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mediaRecorder.stream, stream, + 'Media recorder stream = element stream at the start of recording'); + }; +} + +manager.runTests(gMediaRecorderTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_record_upsize_resolution.html b/dom/media/test/test_mediarecorder_record_upsize_resolution.html new file mode 100644 index 0000000000..d02fd08e44 --- /dev/null +++ b/dom/media/test/test_mediarecorder_record_upsize_resolution.html @@ -0,0 +1,148 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording canvas dynamically changes to greater resolution</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + var canvas = document.createElement("canvas"); + var canvas_size = 100; + var new_canvas_size = 150; + canvas.width = canvas.height = canvas_size; + + var helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size); + helper.drawColor(canvas, helper.red); + + // The recorded stream coming from canvas. + var stream = canvas.captureStream(); + + // Check values for events + var numDataAvailabledRaised = 0; + var numResizeRaised = 0; + // Recorded data that will be playback. + var blob; + + // Media recorder for VP8 and canvas stream. + var mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = canvas stream at the beginning of recording"); + + // Not expected events. + mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired"); + mediaRecorder.onerror = err => { + ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name); + SimpleTest.finish(); + } + + // When recorder is stopped get recorded data. + mediaRecorder.ondataavailable = ev => { + info("Got 'dataavailable' event"); + ++numDataAvailabledRaised; + is(blob, undefined, "On dataavailable event blob is undefined"); + // Save recorded data for playback + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info('onstart fired successfully'); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event"); + + // Playback stream and verify resolution changes. + ok(blob, "Should have gotten a data blob"); + var video = document.createElement("video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + video.onerror = err => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + + // Check here that resize is correct in the playback stream. + video.onresize = function() { + ++numResizeRaised; + if (numResizeRaised == 1) { + is(this.videoWidth, canvas_size, "1st resize event original width"); + is(this.videoHeight, canvas_size, "1st resize event original height "); + } else if (numResizeRaised == 2) { + is(this.videoWidth, new_canvas_size, "2nd resize event new width"); + is(this.videoHeight, new_canvas_size, "2nd resize event new height"); + } else { + ok(false, "Only 2 numResizeRaised events are expected"); + } + }; + + video.onended = () => { + is(numResizeRaised, 2, "Expected 2 resize event"); + }; + document.getElementById("content").appendChild(video); + video.play(); + + // Check last color + helper.pixelMustBecome(video, helper.red, { + threshold: 128, + infoString: "Should become red", + }).then(() => { + video.onresize = {}; + video.onended = {}; + SimpleTest.finish(); + }); + }; + + // Start here by stream recorder. + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder started"); + requestAnimationFrame(draw); + + // Change resolution every 100 ms + var countFrames=0; + var previous_time = performance.now(); + function draw(timestamp) { + if (timestamp - previous_time < 100) { + requestAnimationFrame(draw); + return; + } + previous_time = timestamp; + + var size = 0; + var color = ""; + if (countFrames < 1) { + // Initial size + size = canvas_size; + color = helper.blue; + } else if (countFrames < 2) { + // upsize + size = new_canvas_size; + color = helper.red; + }else { + // Stop recoredr on last frame + mediaRecorder.stop(); + return; + } + // Resize and draw canvas + canvas.width = canvas.height = size; + helper.drawColor(canvas, color); + // Register next draw on every call + requestAnimationFrame(draw); + countFrames++; + } +} + +SimpleTest.waitForExplicitFinish(); +startTest(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_reload_crash.html b/dom/media/test/test_mediarecorder_reload_crash.html new file mode 100644 index 0000000000..f6d008261f --- /dev/null +++ b/dom/media/test/test_mediarecorder_reload_crash.html @@ -0,0 +1,29 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that reloading media recorder object</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=894348">Mozill +a Bug 894348</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + for (let i = 0; i< 5; i++) { + /* eslint-disable no-undef */ + try { o0 = document.createElement('audio') } catch(e) { } + try { o0.src = "sound.ogg" } catch(e) { } + try { (document.body || document.documentElement).appendChild(o0) } catch(e) { } + try { o1 = o0.mozCaptureStreamUntilEnded(); } catch(e) { } + try { o2 = new MediaRecorder(o1) } catch(e) { } + try { o2.start(0) } catch(e) { } + /* eslint-enable no-undef */ + SpecialPowers.gc(); + } + ok(true, "pass the crash test"); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_state_event_order.html b/dom/media/test/test_mediarecorder_state_event_order.html new file mode 100644 index 0000000000..feba055f4d --- /dev/null +++ b/dom/media/test/test_mediarecorder_state_event_order.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder fires an event after changing state</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +/** + * The flow of test is start()=>pause()=>resume()=>stop(). In each steps, + * this test verifies whether each MediaRecorder methods properly change + * its state before firing an event. Checking the state is done in the + * corresponding event handlers. + */ +function startTest(test, token) { + var element = document.createElement('audio'); + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStream(); + + var mediaRecorder = new MediaRecorder(element.stream); + + mediaRecorder.onwarning = function() { + ok(false, 'onwarning unexpectedly fired'); + }; + + mediaRecorder.onerror = function() { + ok(false, 'onerror unexpectedly fired'); + }; + + mediaRecorder.onstart = function() { + info('onstart fired successfully'); + is(mediaRecorder.state, 'recording', + 'Media Recorder changes state to recording before firing a start event'); + mediaRecorder.pause(); + }; + + mediaRecorder.onpause = function() { + info('onpause fired successfully'); + is(mediaRecorder.state, 'paused', + 'Media Recorder changes state to paused before firing a pause event'); + mediaRecorder.resume(); + }; + + mediaRecorder.onresume = function() { + info('onresume fired successfully'); + is(mediaRecorder.state, 'recording', + 'Media Recorder changes state to recording before firing a resume event'); + mediaRecorder.stop(); + }; + + mediaRecorder.onstop = function() { + info('onstop fired successfully'); + is(mediaRecorder.state, 'inactive', + 'Media Recorder changes state to inactive before firing a stop event'); + SimpleTest.finish(); + }; + + // Call start() once metadata are parsed. + element.onloadedmetadata = function() { + element.play(); + mediaRecorder.start(); + is(mediaRecorder.state, 'recording', 'Media recorder should be recording'); + is(mediaRecorder.stream, element.stream, + 'Media recorder stream = element stream at the start of recording'); + }; +} + +manager.runTests(gMediaRecorderTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_state_transition.html b/dom/media/test/test_mediarecorder_state_transition.html new file mode 100644 index 0000000000..5d6483b7de --- /dev/null +++ b/dom/media/test/test_mediarecorder_state_transition.html @@ -0,0 +1,280 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder State Transition</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +// List of operation tests for media recorder objects to verify if running +// these operations should result in an exception or not +var operationTests = [ + { + operations: ['stop'], + isValid: true + }, + { + operations: ['requestData'], + isValid: false + }, + { + operations: ['pause'], + isValid: false + }, + { + operations: ['resume'], + isValid: false + }, + { + operations: ['start'], + isValid: true + }, + { + operations: ['start'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'pause'], + isValid: true + }, + { + operations: ['start', 'pause'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'start'], + isValid: false + }, + { + operations: ['start', 'resume'], + isValid: true + }, + { + operations: ['pause', 'start'], + isValid: false + }, + { + operations: ['resume', 'start'], + isValid: false + }, + { + operations: ['requestData', 'start'], + isValid: false + }, + { + operations: ['stop', 'start'], + isValid: true + }, + { + operations: ['start', 'stop'], + isValid: true + }, + { + operations: ['start', 'stop'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'requestData'], + isValid: true + }, + { + operations: ['start', 'requestData'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'pause', 'stop'], + isValid: true + }, + { + operations: ['start', 'pause', 'start'], + isValid: false + }, + { + operations: ['start', 'pause', 'pause'], + isValid: true + }, + { + operations: ['start', 'pause', 'requestData'], + isValid: true + }, + { + operations: ['start', 'pause', 'resume'], + isValid: true + }, + { + operations: ['start', 'pause', 'resume'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'resume', 'resume'], + isValid: true + }, + { + operations: ['start', 'resume', 'resume'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'resume', 'stop'], + isValid: true + }, + { + operations: ['start', 'resume', 'stop'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'start'], + isValid: true + }, + { + operations: ['start', 'stop', 'start'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'pause'], + isValid: false + }, + { + operations: ['start', 'stop', 'resume'], + isValid: false + }, + { + operations: ['start', 'stop', 'requestData'], + isValid: false + }, + { + operations: ['start', 'stop', 'stop'], + isValid: true + }, + { + operations: ['start', 'pause', 'resume', 'resume'], + isValid: true + }, + { + operations: ['start', 'pause', 'resume', 'resume'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'pause', 'pause', 'resume'], + isValid: true + }, + { + operations: ['start', 'pause', 'pause', 'resume'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'start', 'stop'], + isValid: true + }, + { + operations: ['start', 'stop', 'start', 'stop'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'start', 'pause'], + isValid: true + }, + { + operations: ['start', 'stop', 'start', 'pause'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'start', 'resume'], + isValid: true + }, + { + operations: ['start', 'stop', 'start', 'resume'], + isValid: true, + timeSlice: 200 + }, + { + operations: ['start', 'stop', 'start', 'requestData'], + isValid: true + }, + { + operations: ['start', 'stop', 'start', 'requestData'], + isValid: true, + timeSlice: 200 + }, +]; + +/** + * Runs through each available state transition test by running all + * available operations on a media recorder object. Then, we report + * back if the test was expected through an exception or not. + * + * @param {MediaStream} testStream the media stream used for media recorder + * operation tests + */ +function runStateTransitionTests(testStream) { + for (const operationTest of operationTests) { + var mediaRecorder = new MediaRecorder(testStream); + var operationsString = operationTest.operations.toString(); + + try { + for (const operation of operationTest.operations) { + if (operationTest.timeSlice && operation === 'start') { + operationsString += ' with timeslice ' + operationTest.timeSlice; + mediaRecorder[operation](operationTest.timeSlice); + } else { + mediaRecorder[operation](); + } + } + + ok(operationTest.isValid, `${operationsString} should succeed`); + } catch (err) { + if (operationTest.isValid) { + ok(false, `${operationsString} failed unexpectedly with ${err.name}`); + } else { + is(err.name, "InvalidStateError", + `${operationsString} expected to fail with InvalidStateError`); + } + } + } +} + +/** + * Starts a test on every media recorder file included to check that various + * state transition flows that can happen in the media recorder object throw + * exceptions when they are expected to and vice versa. + */ +function startTest(test, token) { + var element = document.createElement('audio'); + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.stream = element.mozCaptureStream(); + + element.oncanplaythrough = function () { + element.oncanplaythrough = null; + runStateTransitionTests(element.stream); + manager.finished(token); + }; + + element.play(); +} + +manager.runTests(gMediaRecorderTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediarecorder_webm_support.html b/dom/media/test/test_mediarecorder_webm_support.html new file mode 100644 index 0000000000..6b115ee33a --- /dev/null +++ b/dom/media/test/test_mediarecorder_webm_support.html @@ -0,0 +1,56 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media Recording - test WebM MIME support</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +ok(MediaRecorder.isTypeSupported('audio/webm'), + 'Should support audio/webm'); +ok(MediaRecorder.isTypeSupported('AUDIO/WEBM'), + 'Should support audio/webm, upper case'); +ok(MediaRecorder.isTypeSupported('AuDiO/wEbM'), + 'Should support audio/webm, mixed case'); + +ok(MediaRecorder.isTypeSupported('audio/webm;codecs=opus'), + 'Should support audio/webm;codecs=opus'); +ok(MediaRecorder.isTypeSupported('AUDIO/WEBM;CODECS=opus'), + 'Should support audio/webm;codecs=opus, upper case'); +ok(MediaRecorder.isTypeSupported('AuDiO/wEbM;cOdEcS=opus'), + 'Should support audio/webm;codecs=opus, mixed case'); + +ok(MediaRecorder.isTypeSupported('video/webm'), + 'Should support video/webm'); +ok(MediaRecorder.isTypeSupported('VIDEO/WEBM'), + 'Should support video/webm, upper case'); +ok(MediaRecorder.isTypeSupported('vIdEo/WeBm'), + 'Should support video/webm, mixed case'); + +ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8"'), + 'Should support video/webm; codecs="vp8"'); +ok(MediaRecorder.isTypeSupported('VIDEO/WEBM; CODECS="vp8"'), + 'Should support video/webm; codecs="vp8", upper case'); +ok(MediaRecorder.isTypeSupported('vIdEo/WeBm; CoDeCs="vp8"'), + 'Should support video/webm; codecs="vp8", mixed case'); + +ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8.0"'), + 'Should support video/webm; codecs="vp8.0"'); +ok(MediaRecorder.isTypeSupported('VIDEO/WEBM; CODECS="vp8.0"'), + 'Should support video/webm; codecs="vp8.0", upper case'); +ok(MediaRecorder.isTypeSupported('vIdEo/WeBm; CoDeCs="vp8.0"'), + 'Should support video/webm; codecs="vp8.0", mixed case'); + +ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp8, vorbis"'), + 'Should not support video/webm + vp8/vorbis'); +ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp9, vorbis"'), + 'Should not support video/webm + vp9/vorbis'); +ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8, opus"'), + 'Should support video/webm + vp8/opus'); +ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp9, opus"'), + 'Should not support video/webm + vp9/opus'); +</script> +</head> +</html> diff --git a/dom/media/test/test_mediatrack_consuming_mediaresource.html b/dom/media/test/test_mediatrack_consuming_mediaresource.html new file mode 100644 index 0000000000..515df5c053 --- /dev/null +++ b/dom/media/test/test_mediatrack_consuming_mediaresource.html @@ -0,0 +1,198 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test track interfaces when consuming media resources</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +const manager = new MediaTestManager; + +function startTest(test, token) { + const elemType = getMajorMimeType(test.type); + const element = document.createElement(elemType); + + let audioOnchange = 0; + let audioOnaddtrack = 0; + let audioOnremovetrack = 0; + let videoOnchange = 0; + let videoOnaddtrack = 0; + let videoOnremovetrack = 0; + let isPlaying = false; + + isnot(element.audioTracks, undefined, + 'HTMLMediaElement::AudioTracks() property should be available.'); + isnot(element.videoTracks, undefined, + 'HTMLMediaElement::VideoTracks() property should be available.'); + + element.audioTracks.onaddtrack = function(e) { + audioOnaddtrack++; + } + + element.audioTracks.onremovetrack = function(e) { + audioOnremovetrack++; + } + + element.audioTracks.onchange = function(e) { + audioOnchange++; + } + + element.videoTracks.onaddtrack = function(e) { + videoOnaddtrack++; + } + + element.videoTracks.onremovetrack = function(e) { + videoOnremovetrack++; + } + + element.videoTracks.onchange = function(e) { + videoOnchange++; + } + + function checkTrackNotRemoved() { + is(audioOnremovetrack, 0, 'Should have no calls of onremovetrack on audioTracks.'); + is(videoOnremovetrack, 0, 'Should have no calls of onremovetrack on videoTracks.'); + if (isPlaying) { + is(element.audioTracks.length, test.hasAudio ? 1 : 0, + 'Expected length of audioTracks.'); + is(element.videoTracks.length, test.hasVideo ? 1 : 0, + 'Expected length of videoTracks.'); + } + } + + function checkTrackRemoved() { + is(element.audioTracks.length, 0, 'The length of audioTracks should be 0.'); + is(element.videoTracks.length, 0, 'The length of videoTracks should be 0.'); + if (isPlaying) { + is(audioOnremovetrack, test.hasAudio ? 1 : 0, + 'Expected calls of onremovetrack on audioTracks.'); + is(videoOnremovetrack, test.hasVideo ? 1 : 0, + 'Expected calls of onremovetrack on videoTracks.'); + } + } + + function onended() { + ok(true, 'Event ended is expected to be fired on element.'); + checkTrackNotRemoved(); + element.onended = null; + element.onplaying = null; + element.onpause = null; + element.src = ""; + is(element.audioTracks.length, 0, 'audioTracks have been forgotten'); + is(element.videoTracks.length, 0, 'videoTracks have been forgotten'); + is(audioOnremovetrack, 0, 'No audio removetrack events yet'); + is(videoOnremovetrack, 0, 'No video removetrack events yet'); + setTimeout(() => { + checkTrackRemoved(); + manager.finished(element.token); + }, 100); + } + + function checkTrackAdded() { + isPlaying = true; + if (test.hasAudio) { + is(audioOnaddtrack, 1, 'Calls of onaddtrack on audioTracks should be 1.'); + is(element.audioTracks.length, 1, 'The length of audioTracks should be 1.'); + ok(element.audioTracks[0].enabled, 'Audio track should be enabled as default.'); + } + if (test.hasVideo) { + is(videoOnaddtrack, 1, 'Calls of onaddtrack on videoTracks should be 1.'); + is(element.videoTracks.length, 1, 'The length of videoTracks should be 1.'); + is(element.videoTracks.selectedIndex, 0, + 'The first video track is set selected as default.'); + } + } + + function setTrackEnabled(enabled) { + if (test.hasAudio) { + element.audioTracks[0].enabled = enabled; + } + if (test.hasVideo) { + element.videoTracks[0].selected = enabled; + } + } + + function checkTrackChanged(calls, enabled) { + if (test.hasAudio) { + is(audioOnchange, calls, 'Calls of onchange on audioTracks should be '+calls); + is(element.audioTracks[0].enabled, enabled, + 'Enabled value of the audio track should be ' +enabled); + } + if (test.hasVideo) { + is(videoOnchange, calls, 'Calls of onchange on videoTracks should be '+calls); + is(element.videoTracks[0].selected, enabled, + 'Selected value of the video track should be ' +enabled); + var index = enabled ? 0 : -1; + is(element.videoTracks.selectedIndex, index, + 'SelectedIndex of video tracks should be ' +index); + } + } + + function onpause() { + element.onpause = null; + if (element.ended) { + return; + } + if (steps == 1) { + setTrackEnabled(false); + element.onplaying = onplaying; + element.play(); + steps++; + } else if (steps == 2) { + setTrackEnabled(true); + element.onplaying = onplaying; + element.play(); + steps++; + } + } + + function onplaying() { + element.onplaying = null; + if (element.ended) { + return; + } + if (steps == 1) { + element.onpause = onpause; + element.pause(); + checkTrackAdded(); + } else if (steps == 2) { + element.onpause = onpause; + element.pause(); + checkTrackChanged(1, false); + } else if (steps == 3) { + checkTrackChanged(2, true); + } + } + + var steps = 0; + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.onplaying = onplaying; + element.onended = onended; + element.play(); + steps++; +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.track.enabled", true] + ] + }, + function() { + manager.runTests(gTrackTests, startTest); + }); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediatrack_consuming_mediastream.html b/dom/media/test/test_mediatrack_consuming_mediastream.html new file mode 100644 index 0000000000..b930ca4fdc --- /dev/null +++ b/dom/media/test/test_mediatrack_consuming_mediastream.html @@ -0,0 +1,146 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test track interfaces when consuming a MediaStream from gUM</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +async function startTest() { + let steps = 0; + let audioOnchange = 0; + let audioOnaddtrack = 0; + let videoOnchange = 0; + let videoOnaddtrack = 0; + let isPlaying = false; + let element = document.createElement("video"); + let stream; + try { + await setupGetUserMediaTestPrefs(); + stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true}); + } catch (err) { + ok(false, 'Unexpected error fired with: ' + err); + SimpleTest.finish(); + return; + } + + element.audioTracks.onaddtrack = function(e) { + audioOnaddtrack++; + }; + + element.audioTracks.onchange = function(e) { + audioOnchange++; + }; + + element.videoTracks.onaddtrack = function(e) { + videoOnaddtrack++; + }; + + element.videoTracks.onchange = function(e) { + videoOnchange++; + }; + + function checkTrackRemoved() { + if (isPlaying) { + is(element.audioTracks.length, 0, 'The length of audioTracks should be 0.'); + is(element.videoTracks.length, 0, 'The length of videoTracks should be 0.'); + } + } + + element.onended = function() { + ok(true, 'Event ended is expected to be fired on element.'); + checkTrackRemoved(); + element.onended = null; + element.onplaying = null; + element.onpause = null; + SimpleTest.finish(); + } + + function checkTrackAdded() { + isPlaying = true; + is(audioOnaddtrack, 1, 'Calls of onaddtrack on audioTracks should be 1.'); + is(element.audioTracks.length, 1, 'The length of audioTracks should be 1.'); + ok(element.audioTracks[0].enabled, 'Audio track should be enabled as default.'); + is(videoOnaddtrack, 1, 'Calls of onaddtrack on videoTracks should be 1.'); + is(element.videoTracks.length, 1, 'The length of videoTracks should be 1.'); + is(element.videoTracks.selectedIndex, 0, + 'The first video track is set selected as default.'); + } + + function setTrackEnabled(enabled) { + element.audioTracks[0].enabled = enabled; + element.videoTracks[0].selected = enabled; + } + + function checkTrackChanged(calls, enabled) { + is(audioOnchange, calls, 'Calls of onchange on audioTracks should be '+calls); + is(element.audioTracks[0].enabled, enabled, + 'Enabled value of the audio track should be ' +enabled); + is(videoOnchange, calls, 'Calls of onchange on videoTracks should be '+calls); + is(element.videoTracks[0].selected, enabled, + 'Selected value of the video track should be ' +enabled); + var index = enabled ? 0 : -1; + is(element.videoTracks.selectedIndex, index, + 'SelectedIndex of video tracks should be ' +index); + } + + function onpause() { + element.onpause = null; + if (element.ended) { + return; + } + if (steps == 1) { + setTrackEnabled(false); + element.onplaying = onplaying; + element.play(); + steps++; + } else if (steps == 2) { + setTrackEnabled(true); + element.onplaying = onplaying; + element.play(); + steps++; + } + } + + function onplaying() { + element.onplaying = null; + if (element.ended) { + return; + } + if (steps == 1) { + element.onpause = onpause; + element.pause(); + checkTrackAdded(); + } else if (steps == 2) { + element.onpause = onpause; + element.pause(); + checkTrackChanged(1, false); + } else if (steps == 3) { + checkTrackChanged(2, true); + stream.getTracks().forEach(t => t.stop()); + } + } + + element.onplaying = onplaying; + element.srcObject = stream; + + steps++; + await element.play(); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.track.enabled", true] + ] + }, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediatrack_events.html b/dom/media/test/test_mediatrack_events.html new file mode 100644 index 0000000000..5eae94f804 --- /dev/null +++ b/dom/media/test/test_mediatrack_events.html @@ -0,0 +1,135 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test events of media track interfaces</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +async function startTest() { + let steps = 0; + let element = document.createElement("video"); + let stream; + try { + await setupGetUserMediaTestPrefs(); + stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true}); + } catch (err) { + ok(false, 'Unexpected error fired with: ' + err); + SimpleTest.finish(); + return; + } + + function verifyEvent(e, type) { + is(e.type, type, "Event type should be " + type); + ok(e.isTrusted, "Event should be trusted."); + ok(!e.bubbles, "Event shouldn't bubble."); + ok(!e.cancelable, "Event shouldn't be cancelable."); + } + + element.audioTracks.onaddtrack = function(e) { + ok(e instanceof TrackEvent, "Event fired from onaddtrack should be a TrackEvent"); + ok(true, 'onaddtrack is expected to be called from audioTracks.'); + verifyEvent(e, "addtrack"); + }; + + element.audioTracks.onremovetrack = function(e) { + ok(e instanceof TrackEvent, "Event fired from onremovetrack should be a TrackEvent"); + ok(true, 'onremovetrack is expected to be called from audioTracks.'); + verifyEvent(e, "removetrack"); + }; + + element.audioTracks.onchange = function(e) { + ok(e instanceof window.Event, "Event fired from onchange should be a simple event."); + ok(true, 'onchange is expected to be called from audioTracks.'); + verifyEvent(e, "change"); + }; + + element.videoTracks.onaddtrack = function(e) { + ok(e instanceof TrackEvent, "Event fired from onaddtrack should be a TrackEvent"); + ok(true, 'onaddtrack is expected to be called from videoTracks.'); + verifyEvent(e, "addtrack"); + }; + + element.videoTracks.onremovetrack = function(e) { + ok(e instanceof TrackEvent, "Event fired from onremovetrack should be a TrackEvent"); + ok(true, 'onremovetrack is expected to be called from videoTracks.'); + verifyEvent(e, "removetrack"); + }; + + element.videoTracks.onchange = function(e) { + ok(e instanceof window.Event, "Event fired from onchange should be a simple event."); + ok(true, 'onchange is expected to be called from videoTracks.'); + verifyEvent(e, "change"); + }; + + element.onended = function() { + ok(true, 'Event ended is expected to be fired on element.'); + element.onended = null; + element.onplaying = null; + element.onpause = null; + //This helps to prevent these events from firing after SimpleTest.finish() + //on B2G ICS Emulator, but not sure they have been run at all, then + element.audioTracks.onremovetrack = null; + element.audioTracks.onaddtrack = null; + element.audioTracks.onchange = null; + element.videoTracks.onremovetrack = null; + element.videoTracks.onaddtrack = null; + element.videoTracks.onchange = null; + SimpleTest.finish(); + } + + function onpause() { + element.onpause = null; + if (element.ended) { + return; + } + if (steps == 1) { + element.audioTracks[0].enabled = false; + element.videoTracks[0].selected = false; + element.onplaying = onplaying; + element.play(); + steps++; + } + } + + function onplaying() { + element.onplaying = null; + if (element.ended) { + return; + } + if (steps == 1) { + element.onpause = onpause; + element.pause(); + } else if (steps == 2) { + stream.getTracks().forEach(t => t.stop()); + } + } + + element.onplaying = onplaying; + element.srcObject = stream; + + isnot(element.audioTracks, undefined, + 'HTMLMediaElement::AudioTracks() property should be available.'); + isnot(element.videoTracks, undefined, + 'HTMLMediaElement::VideoTracks() property should be available.'); + + steps++; + await element.play(); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.track.enabled", true] + ] + }, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediatrack_parsing_ogg.html b/dom/media/test/test_mediatrack_parsing_ogg.html new file mode 100644 index 0000000000..aabd40e2a3 --- /dev/null +++ b/dom/media/test/test_mediatrack_parsing_ogg.html @@ -0,0 +1,72 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test events of media track interfaces</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function localCheckMetadata(msg, e) { + ok(msg in gOggTrackInfoResults, "File: " + msg + " is in pre-parsed gOggTrackInfoResults list"); + var r = gOggTrackInfoResults[msg]; + + var hasExpectedAudio = r && r.hasOwnProperty("audio_id"); + var hasExpectedVideo = r && r.hasOwnProperty("video_id"); + + var hasParsedAudio = e.audioTracks.length >= 1; + var hasParsedVideo = e.videoTracks.length >= 1; + + ok(!(hasExpectedAudio ^ hasParsedAudio), "Check availability of expected/parsed audio"); + ok(!(hasExpectedVideo ^ hasParsedVideo), "Check availability of expected/parsed video"); + if (hasParsedAudio) { + is(e.audioTracks.length, 1, "The length of audio track should be 1"); + is(e.audioTracks[0].id, r.audio_id, "File: " + msg + ", Audio track id"); + is(e.audioTracks[0].kind, r.audio_kind, "File: " + msg + ", Audio track kind"); + is(e.audioTracks[0].language, r.audio_language, "File: " + msg + ", Audio track language"); + is(e.audioTracks[0].label, r.audio_label, "File: " + msg + ", Audio track label"); + } + if (hasParsedVideo) { + is(e.videoTracks.length, 1, "The length of video track should be 1"); + is(e.videoTracks[0].id, r.video_id, "File: " + msg + ", Video track id"); + is(e.videoTracks[0].kind, r.video_kind, "File: " + msg + ", Video track kind"); + is(e.videoTracks[0].language, r.video_language, "File: " + msg + ", Video track language"); + is(e.videoTracks[0].label, r.video_label, "File: " + msg + ", Video track label"); + } +} + +function startTest(test, token) { + var v = document.createElement('video'); + v.preload = "metadata"; + v.token = token; + manager.started(token); + + v.src = test.name; + v.name = test.name; + + v.onloadedmetadata = function(evt) { + localCheckMetadata(evt.target.name, evt.target); + evt.target.finished = true; + evt.target.onloadedmetadata = null; + removeNodeAndSource(evt.target); + manager.finished(evt.target.token); + }; + + document.body.appendChild(v); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.track.enabled", true]]}, + function() { + manager.runTests(gMultitrackInfoOggPlayList, startTest); + } +); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mediatrack_replay_from_end.html b/dom/media/test/test_mediatrack_replay_from_end.html new file mode 100644 index 0000000000..16b0cbeb97 --- /dev/null +++ b/dom/media/test/test_mediatrack_replay_from_end.html @@ -0,0 +1,160 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test media tracks if replay after playback has ended</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +const manager = new MediaTestManager; + +function startTest(test, token) { + // Scenario to test: + // 1. Audio tracks and video tracks should be added to the track list when + // metadata has loaded, and all tracks should remain even after we seek to + // the end. + // 2. No tracks should be added back to the list if we replay from the end, + // and no tracks should be removed from the list after we seek to the end. + // 3. After seek to the middle from end of playback, all tracks should remain + // in the list if we play from here, and no tracks should be removed from + // the list after we seek to the end. + // 4. Unsetting the media element's src attribute should remove all tracks. + + const elemType = getMajorMimeType(test.type); + const element = document.createElement(elemType); + + let audioOnaddtrack = 0; + let audioOnremovetrack = 0; + let videoOnaddtrack = 0; + let videoOnremovetrack = 0; + let isPlaying = false; + let steps = 0; + + element.audioTracks.onaddtrack = function(e) { + audioOnaddtrack++; + } + + element.audioTracks.onremovetrack = function(e) { + audioOnremovetrack++; + } + + element.videoTracks.onaddtrack = function(e) { + videoOnaddtrack++; + } + + element.videoTracks.onremovetrack = function(e) { + videoOnremovetrack++; + } + + function testExpectedAddtrack(expectedCalls) { + if (test.hasAudio) { + is(audioOnaddtrack, expectedCalls, + 'Calls of onaddtrack on audioTracks should be '+expectedCalls+' times.'); + } + if (test.hasVideo) { + is(videoOnaddtrack, expectedCalls, + 'Calls of onaddtrack on videoTracks should be '+expectedCalls+' times.'); + } + } + + function testExpectedRemovetrack(expectedCalls) { + if (test.hasAudio) { + is(audioOnremovetrack, expectedCalls, + 'Calls of onremovetrack on audioTracks should be '+expectedCalls+' times.'); + } + if (test.hasVideo) { + is(videoOnremovetrack, expectedCalls, + 'Calls of onremovetrack on videoTracks should be '+expectedCalls+' times.'); + } + } + + function finishTesting() { + element.onpause = null; + element.onseeked = null; + element.onplaying = null; + element.onended = null; + manager.finished(element.token); + } + + function onended() { + if (isPlaying) { + switch(steps) { + case 1: + testExpectedAddtrack(1); + testExpectedRemovetrack(0); + element.onplaying = onplaying; + element.play(); + steps++; + break; + case 2: + testExpectedAddtrack(1); + testExpectedRemovetrack(0); + element.currentTime = element.duration * 0.5; + element.onplaying = onplaying; + element.play(); + steps++; + break; + case 3: + testExpectedAddtrack(1); + testExpectedRemovetrack(0); + element.src = ""; + setTimeout(() => { + testExpectedAddtrack(1); + testExpectedRemovetrack(1); + finishTesting(); + }, 0); + break; + } + } else { + ok(true, 'Finish the test anyway if ended is fired before other events.'); + finishTesting(); + } + } + + function seekToEnd() { + element.onpause = null; + element.currentTime = element.duration * 1.1; + } + + function onseeked() { + element.onseeked = null; + element.onpause = seekToEnd; + element.pause(); + } + + function onplaying() { + isPlaying = true; + element.onplaying = null; + element.onseeked = onseeked; + } + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + element.onplaying = onplaying; + element.onended = onended; + element.play(); + steps++; +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.track.enabled", true] + ] + }, + function() { + manager.runTests(gTrackTests, startTest); + }); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_metadata.html b/dom/media/test/test_metadata.html new file mode 100644 index 0000000000..08f28f5f47 --- /dev/null +++ b/dom/media/test/test_metadata.html @@ -0,0 +1,82 @@ +<!DOCTYPE HTML> +<html> +<head> + <meta charset="utf-8"> + <title>Test returning metadata from media files with mozGetMetadata()</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<div id="output"></div> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var a = document.createElement('audio'); + a.preload = "metadata"; + a.token = token; + manager.started(token); + + a.src = test.name; + a.name = test.name; + + // Tags should not be available immediately. + var exception_fired = false; + try { + a.mozGetMetadata(); + } catch (e) { + is(e.name, 'InvalidStateError', + "early mozGetMetadata() should throw InvalidStateError"); + exception_fired = true; + } + ok(exception_fired, + "mozGetMetadata() should throw an exception before HAVE_METADATA"); + + // Wait until metadata has loaded. + a.addEventListener('loadedmetadata', function() { + // read decoded tags + let tags = a.mozGetMetadata(); + ok(tags, "mozGetMetadata() should return a truthy value"); + // Dump them out. + var d = document.getElementById('output'); + var html = '<table>\n'; + html += '<caption><p>Called getMozMetadata()' + html += ' on '+test.name+'</p></caption>\n'; + html += '<tr><th>tag</th>'; + html += '<th>decoded value</th><th>expected value</th></tr>\n'; + for (let tag in tags) { + html += '<tr><td>'+tag+'</td>'; + html += '<td>'+tags[tag]+'</td>'; + html += '<td>'+test.tags[tag]+'</td>'; + html += '</tr>\n'; + } + if (!Object.keys(tags).length) { + html += '<tr><td colspan=3 align=center><em>no tags</em></td></tr>\n'; + } + html += '</table>\n'; + var div = document.createElement('div'); + // eslint-disable-next-line no-unsanitized/property + div.innerHTML = html; + d.appendChild(div); + // Verify decoded tag values. + for (let tag in tags) { + is(tags[tag], test.tags[tag], "Tag '"+tag+"' should match"); + } + // Verify expected tag values + for (let tag in test.tags) { + is(tags[tag], test.tags[tag], "Tag '"+tag+"' should match"); + } + removeNodeAndSource(a); + manager.finished(token); + }); +} + +manager.runTests(gMetadataTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_midflight_redirect_blocked.html b/dom/media/test/test_midflight_redirect_blocked.html new file mode 100644 index 0000000000..55b96ccd38 --- /dev/null +++ b/dom/media/test/test_midflight_redirect_blocked.html @@ -0,0 +1,96 @@ +<!DOCTYPE HTML> +<html> + <head> + <title>Test mid-flight cross site redirects are blocked</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + </head> + <body> + <pre id='test'> + <script class="testbody" type='application/javascript'> + + function testIfLoadsToMetadata(test, useCors) { + return new Promise(function(resolve, reject) { + var elemType = getMajorMimeType(test.type); + var element = document.createElement(elemType); + + if (useCors) { + element.crossOrigin = "anonymous"; + } + + // Log events for debugging. + [ + "suspend", "play", "canplay", "canplaythrough", "loadstart", + "loadedmetadata", "loadeddata", "playing", "ended", "error", + "stalled", "emptied", "abort", "waiting", "pause" + ].forEach((eventName) => { + element.addEventListener(eventName, (event)=> { + info(test.name + " " + event.type); + }); + }); + + element.addEventListener("loadedmetadata", ()=>{ + resolve(true); + removeNodeAndSource(element); + }); + + element.addEventListener("error", ()=>{ + resolve(false); + removeNodeAndSource(element); + }); + + // Note: request redirect before the end of metadata, otherwise we won't + // error before metadata has loaded if mixed origins are allowed. + element.src = "midflight-redirect.sjs?resource=" + test.name + + (useCors ? "&cors" : "") + + "&type=" + test.type + + "&redirectAt=200"; + element.preload = "metadata"; + document.body.appendChild(element); + element.load() + }); + } + + let v = document.createElement("video"); + const testCases = gSmallTests.filter(t => v.canPlayType(t.type)); + + async function testMediaLoad(expectedToLoad, message, useCors) { + for (let test of testCases) { + let loaded = await testIfLoadsToMetadata(test, useCors); + is(loaded, expectedToLoad, test.name + " " + message); + } + } + + async function runTest() { + try { + SimpleTest.info("Allowing midflight redirects..."); + await SpecialPowers.pushPrefEnv({'set': [["media.block-midflight-redirects", false]]}); + + SimpleTest.info("Test that all media plays..."); + await testMediaLoad(true, "expected to load", false); + + SimpleTest.info("Blocking midflight redirects..."); + await SpecialPowers.pushPrefEnv({'set': [["media.block-midflight-redirects", true]]}); + + SimpleTest.info("Test that all media no longer play..."); + await testMediaLoad(false, "expected to be blocked", false); + + SimpleTest.info("Test that all media play if CORS used..."); + await testMediaLoad(true, "expected to play with CORS", true); + } catch (e) { + info("Exception " + e.message); + ok(false, "Threw exception " + e.message); + } + SimpleTest.finish(); + } + + SimpleTest.waitForExplicitFinish(); + + addLoadEvent(runTest); + + </script> + </pre> + </body> +</html> diff --git a/dom/media/test/test_mixed_principals.html b/dom/media/test/test_mixed_principals.html new file mode 100644 index 0000000000..c00d65b983 --- /dev/null +++ b/dom/media/test/test_mixed_principals.html @@ -0,0 +1,83 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=489415 +--> +<head> + <title>Test for Bug 489415</title> + <script type="application/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <style> + video { + width: 40%; + border: solid black 1px; + } + </style> +</head> + +<body> + <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=489415">Mozilla Bug 489415</a> + <p id="display"></p> + <pre id="test"> + <script type="text/javascript"> + SimpleTest.waitForExplicitFinish(); + + var pushPrefs = (...p) => SpecialPowers.pushPrefEnv({ set: p }); + var count = 0; + + function canReadBack(video) { + var c = document.createElement("canvas"); + var ctx = c.getContext("2d"); + ctx.drawImage(video, 0, 0); + try { + c.toDataURL(); + return true; + } catch (ex) { + return false; + } + } + + function runTest(origin, shouldReadBackOnLoad) { + return new Promise(function (resolve, reject) { + // Load will redirect mid-flight, which will be detected and should error, + // and we should no longer be able to readback. + var video = document.createElement("video"); + video.preload = "metadata"; + video.controls = true; + var url = "http://" + origin + "/tests/dom/media/test/midflight-redirect.sjs" + + "?resource=pixel_aspect_ratio.mp4&type=video/mp4"; + SimpleTest.info("Loading from " + url); + video.src = url; + document.body.appendChild(video); + + once(video, "loadeddata", () => { + is(canReadBack(video), shouldReadBackOnLoad, "Should be able to readback"); + video.play(); + }); + + once(video, "error", () => { + is(canReadBack(video), false, "Should not be able to readback after error"); + removeNodeAndSource(video); + resolve(); + }); + + once(video, "ended", () => { + ok(false, "Should not be able to playback to end, we should have errored!"); + removeNodeAndSource(video); + resolve(); + }); + + }); + } + + Promise.all([ + runTest("mochi.test:8888", true), + runTest("example.org", false), + ]).then(SimpleTest.finish); + + </script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mozHasAudio.html b/dom/media/test/test_mozHasAudio.html new file mode 100644 index 0000000000..c25873c786 --- /dev/null +++ b/dom/media/test/test_mozHasAudio.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of media files that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function onloadedmetadata(e) { + var t = e.target; + is(t.mozHasAudio, t.hasAudio, "The element reports the wrong audio property." + t.token); + manager.finished(t.token); +} + +function startTest(test, token) { + var elemType = /^audio/.test(test.type) ? "audio" : "video"; + var element = document.createElement(elemType); + element.preload = "auto"; + + element.token = token; + manager.started(token); + + element.src = test.name; + element.name = test.name; + element.hasAudio = elemType == "video" ? test.hasAudio : undefined; + element.addEventListener("loadedmetadata", onloadedmetadata); + + element.load(); +} + +manager.runTests(gTrackTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_mp3_with_multiple_ID3v2.html b/dom/media/test/test_mp3_with_multiple_ID3v2.html new file mode 100644 index 0000000000..1f2f946520 --- /dev/null +++ b/dom/media/test/test_mp3_with_multiple_ID3v2.html @@ -0,0 +1,30 @@ +<!DOCTYPE HTML> +<html> +<head> +<title>Play mp3 file with multiple ID3v2 tags</title> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +<script type="application/javascript"> + +add_task(async function testPlayMP3WithMultipleID3Tags() { + info(`adjust cache size`); + await SpecialPowers.pushPrefEnv( + // The second ID3v2 header is huge (4622361 bytes) so the first audio samle + // in this file is in the position 4945370, so we have to extend the size of + // the cache. + {"set": [["media.cache_size", 5000000]]} + ); + + info(`create audio and wait its loading`); + let audio = document.createElement('audio'); + audio.src = "multi_id3v2.mp3"; + document.body.appendChild(audio); + await new Promise(r => audio.onloadeddata = r); + ok(true, `finishing loading data.`) +}); + +</script> +</head> +<body> +</body> +</html> diff --git a/dom/media/test/test_multiple_mediastreamtracks.html b/dom/media/test/test_multiple_mediastreamtracks.html new file mode 100644 index 0000000000..fb28d8652e --- /dev/null +++ b/dom/media/test/test_multiple_mediastreamtracks.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test the ability of MediaStream with multiple MediaStreamTracks</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +async function startTest() { + try { + await setupGetUserMediaTestPrefs(); + let orgStream = await navigator.mediaDevices.getUserMedia({audio: true, video: true}); + let a = orgStream.getAudioTracks()[0]; + let v = orgStream.getVideoTracks()[0]; + let stream = new MediaStream([a, a, a, a, v, v, v].map(track => track.clone())); + let element = document.createElement("video"); + + element.onloadedmetadata = function() { + is(stream.getAudioTracks().length, 4, 'Length of audio tracks should be 4.'); + is(stream.getVideoTracks().length, 3, 'Length of vudio tracks should be 3.'); + SimpleTest.finish(); + }; + + element.srcObject = stream; + element.play(); + } catch (err) { + ok(false, 'Unexpected error fired with: ' + err); + SimpleTest.finish(); + } +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.track.enabled", true] + ] + }, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_networkState.html b/dom/media/test/test_networkState.html new file mode 100644 index 0000000000..8001ca514c --- /dev/null +++ b/dom/media/test/test_networkState.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: networkState</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onunload="mediaTestCleanup();"> +<video id='v1'></video><audio id='a1'></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +"use strict"; +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var passed = "truthy"; + +try { + v1.networkState = 0; +} catch (e) { + passed = !passed; +} +try { + a1.networkState = 0; +} catch (e) { + passed = !passed; +} +ok(passed === true, + "Setting networkState throws in strict mode (readonly attribute)"); +</script> + +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var passed = false; + +var oldv1ns = v1.networkState, olda1ns = a1.networkState; +try { + v1.networkState = 0; + a1.networkState = 0; + passed = v1.networkState === oldv1ns && a1.networkState === olda1ns; +} catch (e) { } +ok(passed, "Should not be able to modify networkState (readonly attribute)"); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_new_audio.html b/dom/media/test/test_new_audio.html new file mode 100644 index 0000000000..e1f8964f73 --- /dev/null +++ b/dom/media/test/test_new_audio.html @@ -0,0 +1,48 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=528566 +--> +<head> + <title>Test for Bug 528566</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=528566">Mozilla Bug 528566</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 528566 **/ + +var manager = new MediaTestManager; + +var player = new Audio(); + +function startTest(test, token) { + if (!player.canPlayType(test.type)) { + return; + } + manager.started(token); + var a = new Audio(test.name); + a.autoplay = true; + document.body.appendChild(a); + a.addEventListener("ended", + function(e){ + ok(true, "[" + a.src + "]We should get to the end. Oh look we did."); + a.remove(); + manager.finished(token); + }); +} + +manager.runTests(gAudioTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_no_load_event.html b/dom/media/test/test_no_load_event.html new file mode 100644 index 0000000000..1e2717d983 --- /dev/null +++ b/dom/media/test/test_no_load_event.html @@ -0,0 +1,53 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=715469 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug 715469</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="start();"> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=715469">Mozilla Bug 715469</a> +<p id="display"></p> +<div id="content" style="display: none"> +</div> +<pre id="test"> + +<script type="application/javascript"> + +/** Test for Bug 715469 **/ + +var gotLoadEvent = false; + +function start() { + var resource = getPlayableVideo(gSmallTests); + if (resource == null) { + todo(false, "No types supported"); + } else { + SimpleTest.waitForExplicitFinish(); + var v = document.createElement("video"); + v.src = resource.name; + + v.addEventListener("load", function() { + gotLoadEvent = true; + }); + + v.addEventListener("ended", finished); + document.body.appendChild(v); + v.play(); + } +} + +function finished() { + is(gotLoadEvent, false, "Should not receive a load on the video element"); + SimpleTest.finish(); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html b/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html new file mode 100644 index 0000000000..f1b8cc8b15 --- /dev/null +++ b/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html @@ -0,0 +1,46 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Do not reset playback rate when removing non-loaded media from a document</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<script class="testbody" type="text/javascript"> +/** + * When removing media from a document, it should only trigger internal pause, + * instead of the pause method, which would trigger loading process and reset + * the media's playback rate for non-loaded media. + */ +async function startTest() { + info(`create a media and append it to a document`); + const audio = document.createElement("audio"); + document.body.appendChild(audio); + + info(`change audio's playbackRate and remove it from a document`); + const expectedRate = 0.1; + audio.playbackRate = expectedRate; + await once(audio, "ratechange"); + is(audio.playbackRate, expectedRate, + `${audio.playbackRate} is equal to ${expectedRate}`); + audio.remove(); + + info(`queue a macrotask to check if the playback rate is still unchanged`); + setTimeout(() => { + // If we unexpectedly reset the playback rate, it would happen in a + // microtask when removing media from a document [1] (Await a stable state), + // which would always be run before the macrotask. + // [1] https://html.spec.whatwg.org/#playing-the-media-resource:remove-an-element-from-a-document + is(audio.playbackRate, expectedRate, + `${audio.playbackRate} is equal to ${expectedRate}`); + SimpleTest.finish(); + }, 0); +} + +SimpleTest.waitForExplicitFinish(); +onload = startTest; + +</script> +</body> +</html> diff --git a/dom/media/test/test_paused.html b/dom/media/test/test_paused.html new file mode 100644 index 0000000000..17af4d3898 --- /dev/null +++ b/dom/media/test/test_paused.html @@ -0,0 +1,21 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: paused</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<video id='v1'></video><audio id='a1'></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +ok(v1.paused, "v1.paused must initially be true"); +ok(a1.paused, "a1.paused must initially be true"); +mediaTestCleanup(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_paused_after_ended.html b/dom/media/test/test_paused_after_ended.html new file mode 100644 index 0000000000..83f27a921f --- /dev/null +++ b/dom/media/test/test_paused_after_ended.html @@ -0,0 +1,53 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: paused</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function ended(evt) { + var v = evt.target; + v.removeEventListener("ended", ended); + is(v.gotPause, true, "We should have received a \"pause\" event.") + is(v.paused, true, v._name + " must be paused after end"); + manager.finished(v.token); + removeNodeAndSource(v); +} + +function pause(evt) { + var v = evt.target; + v.removeEventListener("pause", pause); + v.gotPause = true; +} + +function startTest(test, token) { + var v = document.createElement('video'); + document.body.appendChild(v); + v.token = token; + manager.started(v.token); + v.src = test.name; + v._name = test.name; + v._finished = false; + v.load(); + is(v.paused, true, v._name + " must be paused at start"); + + v.play(); + is(v.paused, false, v._name + " must not be paused after play"); + + v.addEventListener("pause", pause); + v.addEventListener("ended", ended); +} + +manager.runTests(gPausedAfterEndedTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_play_events.html b/dom/media/test/test_play_events.html new file mode 100644 index 0000000000..44b819b488 --- /dev/null +++ b/dom/media/test/test_play_events.html @@ -0,0 +1,61 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> + +var manager = new MediaTestManager; + +var tokens = { + 0: ["play"], + "play": ["canplay"], + "canplay": ["playing"], + "playing": ["canplay", "canplaythrough"], + "canplaythrough": ["canplay", "canplaythrough"] +}; + +function gotPlayEvent(event) { + var v = event.target; + ok(tokens[v._state].includes(event.type), + "Check expected event got " + event.type + " at " + v._state + " for " + v.src + + " tokens["+v._state+"]=" + tokens[v._state] + + " tokens["+v._state+"].indexOf(event.type)=" + tokens[v._state].indexOf(event.type)); + v._state = event.type; +} + +function ended(event) { + var v = event.target; + removeNodeAndSource(v); + manager.finished(v.token); +} + +function initTest(test, token) { + var v = document.createElement('video'); + v.token = token; + manager.started(token); + v._state = 0; + + ["play", "canplay", "playing", "canplaythrough"].forEach(function (e) { + v.addEventListener(e, gotPlayEvent); + }); + + v.addEventListener("ended", ended); + + v.src = test.name; + document.body.appendChild(v); // Causes load. + v.play(); +} + +manager.runTests(gSmallTests, initTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_play_events_2.html b/dom/media/test/test_play_events_2.html new file mode 100644 index 0000000000..022bb5373a --- /dev/null +++ b/dom/media/test/test_play_events_2.html @@ -0,0 +1,60 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: play() method via DOM 0 handlers</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +var manager = new MediaTestManager; + +var tokens = { + 0: ["play"], + "play": ["canplay"], + "canplay": ["playing"], + "playing": ["canplay", "canplaythrough"], + "canplaythrough": ["canplay", "canplaythrough"] +}; + +function gotPlayEvent(event) { + var v = event.target; + ok(tokens[v._state].includes(event.type), + "Check expected event got " + event.type + " at " + v._state + " for " + v.src); + v._state = event.type; +} + +function ended(event) { + var v = event.target; + v._finished = true; + removeNodeAndSource(v); + manager.finished(v.token); +} + +function startTest(test, token) { + var v = document.createElement('video'); + v.token = token; + manager.started(token); + v._state = 0; + v._finished = false; + + ["play", "canplay", "playing", "canplaythrough"].forEach(function (e) { + v["on" + e] = gotPlayEvent; + }); + + v.onended = ended; + + v.src = test.name; + document.body.appendChild(v); // Causes load. + v.play(); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_play_promise_1.html b/dom/media/test/test_play_promise_1.html new file mode 100644 index 0000000000..ce4e287f34 --- /dev/null +++ b/dom/media/test/test_play_promise_1.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playBeforeCanPlay +// Case: invoke play() on an element that doesn't have enough data +// Expected result: resolve the promise + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = test.name; + ok(element.readyState == HTMLMediaElement.HAVE_NOTHING); + + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_10.html b/dom/media/test/test_play_promise_10.html new file mode 100644 index 0000000000..1c0096986c --- /dev/null +++ b/dom/media/test/test_play_promise_10.html @@ -0,0 +1,40 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadRejectsPendingPromises +// Case: invoke load() on an element with pending promises. +// Expected result: reject all the pending promises with AbortError DOM exception. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "AbortError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + element.load(); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_11.html b/dom/media/test/test_play_promise_11.html new file mode 100644 index 0000000000..a59baf5657 --- /dev/null +++ b/dom/media/test/test_play_promise_11.html @@ -0,0 +1,40 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: newSrcRejectPendingPromises +// Case: change src of an element with pending promises. +// Expected result: reject all the pending promises. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "AbortError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + element.src = test.name; +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_12.html b/dom/media/test/test_play_promise_12.html new file mode 100644 index 0000000000..50972885eb --- /dev/null +++ b/dom/media/test/test_play_promise_12.html @@ -0,0 +1,45 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: pausePlayAfterPlaybackStarted +// Case: invoke pause() and then play() on an element that is already playing. +// Expected result: resolve the promise with undefined. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + element.play(); + once(element, "playing").then(() => { + element.pause(); + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_13.html b/dom/media/test/test_play_promise_13.html new file mode 100644 index 0000000000..aacab88895 --- /dev/null +++ b/dom/media/test/test_play_promise_13.html @@ -0,0 +1,49 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadAlgorithmDoesNotCancelTasks +// Case: re-invoke the load() on an element which had dispatched a task to resolve a promise. +// Expected result: the already dispatched promise should still be resolved with undefined. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + + // We must wait for "canplay" event; otherwise, invoke play() will lead to a + // pending promise and will then be rejected by the following load(). + once(element, "canplay").then(() => { + // The play() promise will be queued to be resolved immediately, which means + // the promise is not in the pending list. + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + element.src = test.name; // Re-invoke the load algorithm again. + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_14.html b/dom/media/test/test_play_promise_14.html new file mode 100644 index 0000000000..066138ecad --- /dev/null +++ b/dom/media/test/test_play_promise_14.html @@ -0,0 +1,56 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadAlgorithmKeepPromisesPendingWhenNotPausing +// Case: step1: create an element with its paused member to be fause and networkState to be NETWORK_EMPTY. +// stpe2: invoke load() on the element and the load() leaves the promise pending. +// Expected result: the pending promise should finally be resolved. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + // Invoke play() -> (1) the promise will be left pending. + // (2) invoke load() -> (1) set the networkState to be NETWORK_NO_SOURCE. + // (2) queue a task to run resouce selection algorithm. + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + + once(element, "play").then(() => { + // The resouce selection algorithm has been done. + // -> set the networkState to be NETWORK_EMPTY because there is no valid resource to load. + ok(element.networkState == HTMLMediaElement.NETWORK_EMPTY); + + // Invoke load() again and since the networkState is NETWORK_EMPTY, the load() does not reject the pending promise. + // The load() will queue a task to run resouce selection algorithm which will change the readyState and finally resolve the pending promise. + element.src = test.name; + + // Since the networkState is NETWORK_EMPTY, the load() does not set paused to be true. + ok(!element.paused, `loadAlgorithmKeepPromisesPendingWhenNotPausing(${token}).paused should be false.`); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_15.html b/dom/media/test/test_play_promise_15.html new file mode 100644 index 0000000000..b8bce48651 --- /dev/null +++ b/dom/media/test/test_play_promise_15.html @@ -0,0 +1,51 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadAlgorithmRejectPromisesWhenPausing +// Case: step1: create an element with its paused member to be fause and networkState to be NETWORK_NO_SOURCE. +// stpe2: invoke load() on the element and the load() rejects the pending promise. +// Expected result: reject the pending promise with AbortError DOM exception. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + // Invoke play() -> (1) the promise will be left pending. + // (2) invoke load() -> (1) set the networkState to be NETWORK_NO_SOURCE. + // (2) queue a task to run resouce selection algorithm. + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "AbortError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + + ok(element.networkState == HTMLMediaElement.NETWORK_NO_SOURCE); + + // Invoke load() again and since the networkState is NETWORK_NO_SOURCE, the load() rejects the pending promise. + element.src = test.name; + + // Since the networkState is not NETWORK_EMPTY, the load() sets paused to be true. + ok(element.paused, `loadAlgorithmRejectPromisesWhenPausing(${token}).paused should be true.`); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_16.html b/dom/media/test/test_play_promise_16.html new file mode 100644 index 0000000000..a643e7fff8 --- /dev/null +++ b/dom/media/test/test_play_promise_16.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadAlgorithmResolveOrdering +// Case: invoke load() on an element should resolve pending promises in order. +// Expected result: the pending promises are resolved in order. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + once(element, "canplay").then(() => { + let firstPromiseResolved = false; + + // play + element.play().then( + () => { firstPromiseResolved = true; }, + () => { ok(false, `loadAlgorithmResolveOrdering(${token}) should not be rejected.`); } + ); + + // play again + element.play().then( + () => { ok(firstPromiseResolved, `loadAlgorithmResolveOrdering(${token}), the first play should already be resolved.`); }, + () => { ok(false, `loadAlgorithmResolveOrdering(${token}) should not be rejected.`); } + ).then( () => { manager.finished(token); } ); + + // triger load + element.src = test.name; + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_17.html b/dom/media/test/test_play_promise_17.html new file mode 100644 index 0000000000..5e5eb71fd5 --- /dev/null +++ b/dom/media/test/test_play_promise_17.html @@ -0,0 +1,43 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: loadAlgorithmRejectOrdering +// Case: invoke load() on an element should reject pending promises in order. +// Expected result: the pending promises are rejected in order. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + let firstPromiseRejected = false; + + // play + element.play().then( + () => { ok(false, `loadAlgorithmRejectOrdering(${token}) should not be resolved.`); }, + () => { firstPromiseRejected = true; } + ); + + // play again + element.play().then( + () => { ok(false, `loadAlgorithmRejectOrdering(${token}) should not be resolved.`); }, + () => { ok(firstPromiseRejected, `loadAlgorithmRejectOrdering(${token}), the first play should already be rejected.`); } + ).then( () => { manager.finished(token); } ); + + // triger load + element.src = test.name; +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_18.html b/dom/media/test/test_play_promise_18.html new file mode 100644 index 0000000000..e0c0837fdf --- /dev/null +++ b/dom/media/test/test_play_promise_18.html @@ -0,0 +1,46 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: 'playing' event should come before promise resolving +// Case: invoke play() on an element which has valis source +// Expected result: receive a 'playing' event before the promise is resolved + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = test.name; + element.receivedPlayingEvent = false; + + element.addEventListener("playing", () => { + element.receivedPlayingEvent = true; + }); + + element.play().then( + (result) => { + if (result == undefined && element.receivedPlayingEvent) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result} and receivedPlayingEvent = ${element.receivedPlayingEvent}`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_2.html b/dom/media/test/test_play_promise_2.html new file mode 100644 index 0000000000..c4befdbe91 --- /dev/null +++ b/dom/media/test/test_play_promise_2.html @@ -0,0 +1,43 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playWhenCanPlay +// Case: invoke play() on an element that has enough data +// Expected result: resolve the promise + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + once(element, "canplay").then(() => { + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_3.html b/dom/media/test/test_play_promise_3.html new file mode 100644 index 0000000000..8ffacdae03 --- /dev/null +++ b/dom/media/test/test_play_promise_3.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playAfterPlaybackStarted +// Case: invoke play() on an element that is already playing +// Expected result: resolve the promise + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + once(element, "playing").then(() => { + ok(element.readyState >= HTMLMediaElement.HAVE_FUTURE_DATA, `playAfterPlaybackStarted(${token})`); + ok(!element.paused, `playAfterPlaybackStarted(${token})`); + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + }); + + element.play(); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_4.html b/dom/media/test/test_play_promise_4.html new file mode 100644 index 0000000000..dfcd96e0b9 --- /dev/null +++ b/dom/media/test/test_play_promise_4.html @@ -0,0 +1,41 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="play_promise.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playNotSupportedContent +// Case: invoke play() on an element with an unsupported content +// Expected result: reject the promise with NotSupportedError DOM exception + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = getNotSupportedFile(test.name); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "NotSupportedError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_5.html b/dom/media/test/test_play_promise_5.html new file mode 100644 index 0000000000..a4c3eeb936 --- /dev/null +++ b/dom/media/test/test_play_promise_5.html @@ -0,0 +1,44 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="play_promise.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playWithErrorAlreadySet +// Case: invoke play() on an element with MEDIA_ERR_SRC_NOT_SUPPORTED has been set +// Expected result: reject the promise with NotSupportedError DOM exception + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = getNotSupportedFile(test.name); + once(element, "error").then(() => { + ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "NotSupportedError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_6.html b/dom/media/test/test_play_promise_6.html new file mode 100644 index 0000000000..dcbea3f108 --- /dev/null +++ b/dom/media/test/test_play_promise_6.html @@ -0,0 +1,45 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="play_promise.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playSwitchToValidSrcAfterError +// Case: invoke play() on an element which had its source changed (to a valid source) after suffering from an error +// Expected result: resolve the promise with undefined + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = getNotSupportedFile(test.name); + once(element, "error").then(() => { + ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED); + element.src = test.name; + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_7.html b/dom/media/test/test_play_promise_7.html new file mode 100644 index 0000000000..1abbe42f33 --- /dev/null +++ b/dom/media/test/test_play_promise_7.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="play_promise.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playSwitchToInvalidSrcAfterError +// Case: invoke play() on an element which had its source changed (to a invalid source) after suffering from an error +// Expected result: reject the promise with NotSupportedError DOM exception + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.src = getNotSupportedFile(test.name); + once(element, "error").then(() => { + ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED); + element.src = getNotSupportedFile(test.name); + ok(element.error == null); + ok(element.readyState == HTMLMediaElement.HAVE_NOTHING); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "NotSupportedError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_8.html b/dom/media/test/test_play_promise_8.html new file mode 100644 index 0000000000..d61e12c9fe --- /dev/null +++ b/dom/media/test/test_play_promise_8.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playAndPauseWhenCanplay +// Case: invlke play() and then pause() on an element that already has enough data to play +// Expected result: resolve the promise +// Note: the pause() doesn't cancel the play() because it was alredy been queued to be resolved. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + element.preload = "auto"; + element.src = test.name; + once(element, "canplay").then(() => { + element.play().then( + (result) => { + if (result == undefined) { + ok(true, `${token} is resolved with ${result}.`); + } else { + ok(false, `${token} is resolved with ${result}.`); + } + }, + (error) => { + ok(false, `${token} is rejected with ${error.name}.`); + } + ).then( () => { manager.finished(token); } ); + ok(!element.paused, `playAndPauseWhenCanplay(${token}) element should not be paused.`); + element.pause(); + ok(element.paused, `playAndPauseWhenCanplay(${token}) element should be paused.`); + }); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_promise_9.html b/dom/media/test/test_play_promise_9.html new file mode 100644 index 0000000000..c3a3856515 --- /dev/null +++ b/dom/media/test/test_play_promise_9.html @@ -0,0 +1,44 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: promise-based play() method</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> +// Name: playAndPauseBeforeCanPlay +// Case: invlke play() and then pause() on an element that deoen't have enough data to play. +// Expected result: reject the promise with AbortError DOM exception. +// Note: the pause() cancels the play() because the promise is still pending. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + let element = document.createElement(getMajorMimeType(test.type)); + ok(element.readyState == HTMLMediaElement.HAVE_NOTHING); + element.play().then( + (result) => { + ok(false, `${token} is resolved with ${result}.`); + }, + (error) => { + if (error.name == "AbortError") { + ok(true, `${token} is rejected with ${error.name}.`); + } else { + ok(false, `${token} is rejected with ${error.name}.`); + } + } + ).then( () => { manager.finished(token); } ); + ok(!element.paused, `playAndPauseBeforeCanPlay(${token}) element should not be paused.`); + element.pause(); + ok(element.paused, `playAndPauseBeforeCanPlay(${token}) element should be paused.`); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_play_twice.html b/dom/media/test/test_play_twice.html new file mode 100644 index 0000000000..e94f28a031 --- /dev/null +++ b/dom/media/test/test_play_twice.html @@ -0,0 +1,95 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of media files that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.token = token; + manager.started(token); + video.src = test.name; + video.name = test.name; + video.playingCount = 0; + video._playedOnce = false; + + var check = function(t, v) { return function() { + checkMetadata(t.name, v, test); + }}(test, video); + + var noLoad = function(t, v) { return function() { + ok(false, t.name + " should not fire 'load' event"); + }}(test, video); + + function finish(v) { + removeNodeAndSource(v); + manager.finished(v.token); + } + + function mayFinish(v) { + if (v.seenEnded && v.seenSuspend) { + finish(v); + } + } + + var checkEnded = function(t, v) { return function() { + if (t.duration) { + ok(Math.abs(v.currentTime - t.duration) < 0.1, + t.name + " current time at end: " + v.currentTime); + } + + is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState"); + ok(v.ended, t.name + " checking playback has ended"); + ok(v.playingCount > 0, "Expect at least one playing event"); + v.playingCount = 0; + + if (v._playedOnce) { + v.seenEnded = true; + mayFinish(v); + } else { + v._playedOnce = true; + v.play(); + } + }}(test, video); + + var checkSuspended = function(t, v) { return function() { + if (v.seenSuspend) { + return; + } + + v.seenSuspend = true; + ok(true, t.name + " got suspend"); + mayFinish(v); + }}(test, video); + + var checkPlaying = function(t, v) { return function() { + is(t.name, v.name, "Should be testing file we think we're testing..."); + v.playingCount++; + }}(test, video); + + video.addEventListener("load", noLoad); + video.addEventListener("loadedmetadata", check); + video.addEventListener("playing", checkPlaying); + + // We should get "ended" and "suspend" events for every resource + video.addEventListener("ended", checkEnded); + video.addEventListener("suspend", checkSuspended); + + document.body.appendChild(video); + video.play(); +} + +manager.runTests(gReplayTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_playback.html b/dom/media/test/test_playback.html new file mode 100644 index 0000000000..5e28861e93 --- /dev/null +++ b/dom/media/test/test_playback.html @@ -0,0 +1,108 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of media files that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.preload = "metadata"; + video.token = token; + video.prevTime = 0; + video.seenEnded = false; + video.seenSuspend = false; + + var handler = { + "ontimeout": function() { + Log(token, "timed out: ended=" + video.seenEnded + ", suspend=" + video.seenSuspend); + } + }; + manager.started(token, handler); + + video.src = test.name; + video.name = test.name; + + var check = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #1"); + checkMetadata(t.name, v, t); + }}(test, video); + + var noLoad = function(t, v) { return function() { + ok(false, t.name + " should not fire 'load' event"); + }}(test, video); + + var noError = function(t, v) { return function() { + ok(false, t.name + " should not fire 'error' event " + v.error.message); + }}(test, video); + + var finish = function() { + video.finished = true; + video.removeEventListener("timeupdate", timeUpdate); + removeNodeAndSource(video); + manager.finished(video.token); + } + + // We should get "ended" and "suspend" events to finish the test. + var mayFinish = function() { + if (video.seenEnded && video.seenSuspend) { + finish(); + } + } + + var checkEnded = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #2"); + checkMetadata(t.name, v, test); + is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState"); + ok(v.ended, t.name + " checking playback has ended"); + ok(!v.finished, t.name + " shouldn't be finished"); + ok(!v.seenEnded, t.name + " shouldn't be ended"); + + v.seenEnded = true; + mayFinish(); + }}(test, video); + + var checkSuspended = function(t, v) { return function() { + if (v.seenSuspend) { + return; + } + is(t.name, v.name, t.name + ": Name should match #3"); + + v.seenSuspend = true; + mayFinish(); + }}(test, video); + + var timeUpdate = function(t, v) { return function() { + if (v.prevTime > v.currentTime) { + ok(false, t.name + " time should run forwards: p=" + + v.prevTime + " c=" + v.currentTime); + } + v.prevTime = v.currentTime; + }}(test, video); + + video.addEventListener("load", noLoad); + video.addEventListener("error", noError); + video.addEventListener("loadedmetadata", check); + video.addEventListener("timeupdate", timeUpdate); + + // We should get "ended" and "suspend" events for every resource + video.addEventListener("ended", checkEnded); + video.addEventListener("suspend", checkSuspended); + + document.body.appendChild(video); + video.play(); +} + +manager.runTests(gPlayTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_playback_errors.html b/dom/media/test/test_playback_errors.html new file mode 100644 index 0000000000..7b3f046099 --- /dev/null +++ b/dom/media/test/test_playback_errors.html @@ -0,0 +1,48 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of media files that should have errors</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + manager.started(token); + video._errorCount = 0; + video._ignore = false; + function endedTest(v) { + if (v._ignore) + return; + v._ignore = true; + v.remove(); + manager.finished(token); + } + var checkError = function(t, v) { return function(evt) { + v._errorCount++; + is(v._errorCount, 1, t.name + " only one error fired"); + endedTest(v); + }}(test, video); + var checkEnded = function(t, v) { return function() { + ok(false, t.name + " successfully played"); + endedTest(v); + }}(test, video); + video.addEventListener("error", checkError); + video.addEventListener("ended", checkEnded); + video.src = test.name; + document.body.appendChild(video); + video.play(); +} + +manager.runTests(gErrorTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_playback_hls.html b/dom/media/test/test_playback_hls.html new file mode 100644 index 0000000000..7d5c777fc1 --- /dev/null +++ b/dom/media/test/test_playback_hls.html @@ -0,0 +1,91 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback of HLS with simple m3u8 that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.preload = "metadata"; + video.token = token; + video.prevTime = 0; + video.seenEnded = false; + + var handler = { + "ontimeout": function() { + Log(token, "timed out: ended=" + video.seenEnded); + } + }; + manager.started(token, handler); + + video.src = test.name; + video.name = test.name; + + var check = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #1"); + checkMetadata(t.name, v, t); + }}(test, video); + + var noLoad = function(t, v) { return function() { + ok(false, t.name + " should not fire 'load' event"); + }}(test, video); + + var finish = function() { + video.finished = true; + video.removeEventListener("timeupdate", timeUpdate); + removeNodeAndSource(video); + manager.finished(video.token); + } + + // We should get "ended" events to finish the test. + var mayFinish = function() { + if (video.seenEnded) { + finish(); + } + } + + var checkEnded = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #2"); + checkMetadata(t.name, v, test); + is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState"); + ok(v.ended, t.name + " checking playback has ended"); + ok(!v.finished, t.name + " shouldn't be finished"); + ok(!v.seenEnded, t.name + " shouldn't be ended"); + + v.seenEnded = true; + mayFinish(); + }}(test, video); + + var timeUpdate = function(t, v) { return function() { + if (v.prevTime > v.currentTime) { + ok(false, t.name + " time should run forwards: p=" + + v.prevTime + " c=" + v.currentTime); + } + v.prevTime = v.currentTime; + }}(test, video); + + video.addEventListener("load", noLoad); + video.addEventListener("loadedmetadata", check); + video.addEventListener("timeupdate", timeUpdate); + + // We should get "ended" events for the hls resource + video.addEventListener("ended", checkEnded); + + document.body.appendChild(video); + video.play(); +} + +manager.runTests(gHLSTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_playback_rate.html b/dom/media/test/test_playback_rate.html new file mode 100644 index 0000000000..a704b9a1ba --- /dev/null +++ b/dom/media/test/test_playback_rate.html @@ -0,0 +1,175 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for the playbackRate property </title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type='application/javascript'> + +let manager = new MediaTestManager; + +function rangeCheck(lhs, rhs, threshold) { + var diff = Math.abs(lhs - rhs); + if (diff < threshold) { + return true; + } + return false; +} + +function checkPlaybackRate(wallclock, media, expected, threshold) { + if (rangeCheck(media / wallclock, expected, threshold)) { + return true; + } + return false; +} + +// Those value are expected to match those at the top of HTMLMediaElement.cpp. +let VERY_SLOW_RATE = 1 / 32, + SLOW_RATE = 1 / 16, + FAST_RATE = 16, + VERY_FAST_RATE = 20, + NULL_RATE = 0.0; + +function ontimeupdate(e) { + var t = e.target; + // Skip short files for SoundTouch doesn't work well on small number of samples. + if (t.gotEnded || t.duration < 2) { + return; + } + t.testedForSlowdown = true; + if (t.currentTime > t.duration / 2) { + t.oldCurrentTime = t.currentTime; + t.timestamp = Date.now(); + var delta = t.oldCurrentTime, + delta_wallclock = (t.timestamp - t.startTimestamp - t.bufferingTime) / 1000; + + t.mozPreservesPitch = false; + is(t.mozPreservesPitch, false, t.name + ": If we disable the pitch preservation, it should appear as such."); + + t.bufferingTime = 0; + + is(t.playbackRate, SLOW_RATE, t.name + ": The playback rate shoud be "+SLOW_RATE+"."); + ok(checkPlaybackRate(delta_wallclock, delta, SLOW_RATE, 0.25), t.name + ": We are effectively slowing down playback. (" + delta_wallclock + ", " + delta + ")"); + t.removeEventListener("timeupdate", ontimeupdate); + t.addEventListener("pause", onpaused); + t.playbackRate = NULL_RATE; + t.oldCurrentTime = t.currentTime; + setTimeout(function() { + afterNullPlaybackRate(e); + }, 100); + } +} + +function onpaused(e) { + var t = e.target; + t.pausedReceived = true; +} + +function afterNullPlaybackRate(e) { + var t = e.target; + + // skip if we have received 'ended' event or 'ended' event is pending. + if (t.gotEnded || t.ended) { + return; + } + + t.testedForNull = true; + + ok(t.currentTime == t.oldCurrentTime, t.name + ": Current time should not change when playbackRate is null (" + t.currentTime + " " + t.oldCurrentTime + ")."); + ok(!t.paused, t.name + ": The element should not be in paused state."); + t.removeEventListener("paused", onpaused); + is(t.pausedReceived, undefined, t.name + ": Paused event should not have been received."); + t.timestamp = Date.now(); + t.oldCurrentTime = t.currentTime; + t.playbackRate = VERY_FAST_RATE; + is(t.playbackRate, VERY_FAST_RATE, t.name + ": Playback rate should be clamped to " + VERY_FAST_RATE + "."); +} + +function onended(e) { + var t = e.target; + t.gotEnded = true; + + t.bufferingTime = 0; + // If we got "ended" too early, skip these tests. + if (t.testedForSlowdown && t.testedForNull) { + is(t.playbackRate, FAST_RATE, t.name + ": The playback rate should still be "+FAST_RATE+"."); + ok(!t.muted, t.name + ": The audio should be muted when playing at high speed, but should not appear as such."); + is(t.currentTime, t.duration, t.name + ": Current time should be equal to the duration (not change by playback rate)."); + } + finish_test(t); +} + +function onratechange(e) { + if (!e.target.ratechangecount) { + e.target.ratechangecount = 0; + } + e.target.ratechangecount++; +} + +function finish_test(element) { + removeNodeAndSource(element); + manager.finished(element.token); +} + +// These two functions handle the case when the playback pauses for buffering. It +// adjusts the timestamps to be accurate. Despite the fact that the web servers +// is supposed to be on the same machine, buffering pauses can occur (rarely, +// but still). +function onplaying(e) { + var t = e.target; + if (t.bufferingTimestamp != undefined) { + t.bufferingTime += (Date.now() - t.bufferingTimestamp); + t.bufferingTimestamp = undefined; + } +} + +function onwaiting(e) { + var t = e.target; + t.bufferingTimestamp = Date.now(); +} + +function onvolumechange(e) { + ok(false, e.target.name + ": We should not receive a volumechange event when changing the playback rate."); +} + +function startTest(test, token) { + let elemType = /^audio/.test(test.type) ? "audio" : "video"; + let element = document.createElement(elemType); + element.src = test.name; + element.name = test.name; + element.preload = "metadata"; + element.token = token; + element.controls = true; + element.bufferingTime = 0; + document.body.appendChild(element); + element.addEventListener("ratechange", onratechange); + element.addEventListener("timeupdate", ontimeupdate); + element.addEventListener("ended", onended); + element.addEventListener("waiting", onwaiting); + element.addEventListener("playing", onplaying); + element.addEventListener("volumechange", onvolumechange); + manager.started(token); + element.startTimestamp = Date.now(); + is(element.mozPreservesPitch, true, test.name + ": Pitch preservation should be enabled by default."); + element.addEventListener("loadedmetadata", function() { + is(element.playbackRate, 1.0, test.name + ": playbackRate should be initially 1.0"); + is(element.defaultPlaybackRate, 1.0, test.name + ": defaultPlaybackRate should be initially 1.0"); + element.playbackRate = VERY_SLOW_RATE; + is(element.playbackRate, VERY_SLOW_RATE, test.name + ": PlaybackRate should be " + VERY_SLOW_RATE + "."); + element.play(); + element.playbackRate = SLOW_RATE; + }); +} + +manager.runTests(gPlayedTests, startTest); + +</script> +</pre> +<div id="elements"> +</div> +</body> +</html> diff --git a/dom/media/test/test_playback_rate_playpause.html b/dom/media/test/test_playback_rate_playpause.html new file mode 100644 index 0000000000..2f3e6d8b9a --- /dev/null +++ b/dom/media/test/test_playback_rate_playpause.html @@ -0,0 +1,66 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that the playbackRate property is not reset when resuming the playback</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type='application/javascript'> + +let manager = new MediaTestManager; + +function ontimeupdate(e) { + var t = e.target; + if (t.currentTime != 0.0) { + dump(t.token + " t.currentTime != 0.0.\n"); + t.removeEventListener("timeupdate", ontimeupdate); + t.pause(); + is(t.playbackRate, 0.5, "PlaybackRate should not have changed after pause."); + } else { + dump(t.token + " t.currentTime == 0.0.\n"); + } +} + +function onpaused(e) { + var t = e.target; + dump(t.token + " onpaused.\n"); + t.play(); + is(t.playbackRate, 0.5, "PlaybackRate should not have changed after resuming playback."); + finish_test(t); +} + +function finish_test(element) { + dump(element.token + " finish_test.\n"); + removeNodeAndSource(element); + manager.finished(element.token); +} + +function startTest(test, token) { + let elemType = /^audio/.test(test.type) ? "audio" : "video"; + let element = document.createElement(elemType); + element.src = test.name; + element.token = token; + element.controls = true; + element.playbackRate = 0.5; + element.preload = "metadata"; + element.addEventListener("timeupdate", ontimeupdate); + element.addEventListener("pause", onpaused); + element.addEventListener("loadedmetadata", function() { + dump(element.token + " loadedmetadata\n"); + element.play(); + }); + document.body.appendChild(element); + manager.started(token); +} + +manager.runTests(gPlayedTests, startTest); + +</script> +</pre> +<div id="elements"> +</div> +</body> +</html> diff --git a/dom/media/test/test_playback_reactivate.html b/dom/media/test/test_playback_reactivate.html new file mode 100644 index 0000000000..dec5330ce5 --- /dev/null +++ b/dom/media/test/test_playback_reactivate.html @@ -0,0 +1,77 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test playback with dormant of media files that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +/* This testcase wants to test a video element's playback is not break + by dormant. + When the metadata is loaded, we remove the video element to trigger dormant. + Then set a timer to append the video element back and play it. + Test pass if the video plays to the end. +*/ + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.preload = "metadata"; + video.token = token; + + var handler = { + "ontimeout": function() { + Log(token, "timed out: ended=" + video.seenEnded + ", suspend=" + video.seenSuspend); + } + }; + manager.started(token, handler); + + video.src = test.name; + video.name = test.name; + + var check = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #1"); + Log(v.token, "removeChild: " + v.name); + document.body.removeChild(v); + var appendAndPlayElement = function() { + Log(v.token, "appendChild: " + v.name); + document.body.appendChild(v); + Log(v.token, "Element play: " + v.name); + v.play(); + } + setTimeout(appendAndPlayElement, 2000); + }}(test, video); + + var finish = function() { + video.finished = true; + removeNodeAndSource(video); + manager.finished(video.token); + } + + var checkEnded = function(t, v) { return function() { + is(t.name, v.name, t.name + ": Name should match #2"); + checkMetadata(t.name, v, t); + is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState"); + ok(v.ended, t.name + " checking playback has ended"); + + finish(); + }}(test, video); + + + video.addEventListener("loadedmetadata", check); + video.addEventListener("ended", checkEnded); + + document.body.appendChild(video); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_played.html b/dom/media/test/test_played.html new file mode 100644 index 0000000000..e8b4906c2c --- /dev/null +++ b/dom/media/test/test_played.html @@ -0,0 +1,245 @@ +<!DOCTYPE HTML> +<html> +<head> +<title>Test played member for media elements</title> +<script type="text/javascript" src="/MochiKit/MochiKit.js"></script> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +<script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id='test'> +<script class="testbody" type='application/javascript'> + +let manager = new MediaTestManager; + +function finish_test(element) { + removeNodeAndSource(element); + manager.finished(element.token); +} + +// Check that a file has been played in its entirety. +function check_full_file_played(element) { + element.addEventListener('ended', (function(e) { + let interval_count = e.target.played.length; + is(interval_count, 1, element.token + ": played.length must be 1"); + is(element.played.start(0), 0, element.token + ": start time shall be 0"); + is(element.played.end(0), e.target.duration, element.token + ": end time shall be duration"); + finish_test(e.target); + })); +} + +var tests = [ +// Without playing, check that player.played.length == 0. +{ + setup(element) { + element.addEventListener("loadedmetadata", function() { + is(element.played.length, 0, element.token + ": initial played.length equals zero"); + finish_test(element); + }); + }, + name: "test1" +}, +// Play the file, test the range we have. +{ + setup(element) { + check_full_file_played(element); + element.play(); + }, + name: "test2" +}, + +// Play the second half of the file, pause, play +// an check we have only one range. +{ + setup (element) { + element.onended = function (e) { + var t = e.target; + t.onended = null; + check_full_file_played(t); + t.pause(); + t.currentTime = 0; + t.play(); + }; + element.addEventListener("loadedmetadata", function() { + element.currentTime = element.duration / 2; + element.play(); + }); + }, + name: "test3" +}, + +// Play the first half of the file, seek back, while +// continuing to play. We shall have only one range. +{ + setup (element) { + let onTimeUpdate = function() { + if (element.currentTime > element.duration / 2) { + info(element.token + ": currentTime=" + element.currentTime + ", duration=" + element.duration); + element.removeEventListener("timeupdate", onTimeUpdate); + element.pause(); + var oldEndRange = element.played.end(0); + element.currentTime = element.duration / 4; + is(element.played.end(0), oldEndRange, + element.token + ": When seeking back, |played| should not be changed"); + element.play(); + } + } + element.addEventListener("timeupdate", onTimeUpdate); + check_full_file_played(element); + element.play(); + }, + name: "test4" +}, + +// Play and seek to have two ranges, and check that, as well a +// boundaries. +{ + setup (element) { + let seekTarget = 0; + let onTimeUpdate = function() { + if (element.currentTime > element.duration / 2) { + info(element.token + ": currentTime=" + element.currentTime + ", duration=" + element.duration); + element.removeEventListener("timeupdate", onTimeUpdate); + element.pause(); + // Remember seek target for later comparison since duration may change + // during playback. + seekTarget = element.currentTime = element.duration / 10; + element.currentTime = seekTarget; + element.play(); + } + } + + element.addEventListener("loadedmetadata", function() { + element.addEventListener("timeupdate", onTimeUpdate); + }); + + + element.addEventListener("ended", (function() { + if(element.played.length > 1) { + is(element.played.length, 2, element.token + ": element.played.length == 2"); + is(element.played.start(1), seekTarget, element.token + ": we should have seeked forward by one tenth of the duration"); + is(element.played.end(1), element.duration, element.token + ": end of second range shall be the total duration"); + } + is(element.played.start(0), 0, element.token + ": start of first range shall be 0"); + finish_test(element); + })); + + element.play(); + }, + name: "test5" +}, + +// Play to create two ranges, in the reverse order. check that they are sorted. +{ + setup (element) { + function end() { + element.pause(); + let p = element.played; + ok(p.length >= 1, element.token + ": There should be at least one range=" + p.length); + is(p.start(0), seekTarget, element.token + ": Start of first range should be the sixth of the duration"); + ok(p.end(p.length - 1) > 5 * element.duration / 6, element.token + ": End of last range should be greater that five times the sixth of the duration"); + finish_test(element); + } + + let seekTarget = 0; + function pauseseekrestart() { + element.pause(); + // Remember seek target for later comparison since duration may change + // during playback. + seekTarget = element.duration / 6; + element.currentTime = seekTarget; + element.play(); + } + + function onTimeUpdate_pauseseekrestart() { + if (element.currentTime > 5 * element.duration / 6) { + element.removeEventListener("timeupdate", onTimeUpdate_pauseseekrestart); + pauseseekrestart(); + element.addEventListener("timeupdate", onTimeUpdate_end); + } + } + + function onTimeUpdate_end() { + if (element.currentTime > 3 * element.duration / 6) { + element.removeEventListener("timeupdate", onTimeUpdate_end); + end(); + } + } + + element.addEventListener("timeupdate", onTimeUpdate_pauseseekrestart); + + element.addEventListener('loadedmetadata', function() { + element.currentTime = 4 * element.duration / 6; + element.play(); + }); + }, + name: "test6" +}, +// Seek repeatedly without playing. No range should appear. +{ + setup(element) { + let index = 1; + + element.addEventListener('seeked', function() { + index++; + element.currentTime = index * element.duration / 5; + is(element.played.length, 0, element.token + ": played.length should be 0"); + if (index == 5) { + finish_test(element); + } + }); + + element.addEventListener('loadedmetadata', function() { + element.currentTime = element.duration / 5; + }); + }, + name: "test7" +} +]; + +function createTestArray() { + var A = []; + for (var i=0; i<tests.length; i++) { + for (var k=0; k<gPlayedTests.length; k++) { + var t = {}; + t.setup = tests[i].setup; + t.name = tests[i].name + "-" + gPlayedTests[k].name; + t.type = gPlayedTests[k].type; + t.src = gPlayedTests[k].name; + A.push(t); + } + } + return A; +} + +function startTest(test, token) { + var elemType = getMajorMimeType(test.type); + var element = document.createElement(elemType); + element.src = test.src; + element.token = token; + element.preload = "metadata"; + test.setup(element); + manager.started(token); + + // Log events for debugging. + var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(e) { + var v = e.target; + info(v.token + ": got " + e.type); + } + events.forEach(function(e) { + element.addEventListener(e, logEvent); + }); + +} + + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_preload_actions.html b/dom/media/test/test_preload_actions.html new file mode 100644 index 0000000000..a9de54988c --- /dev/null +++ b/dom/media/test/test_preload_actions.html @@ -0,0 +1,582 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=548523 +--> +<head> + <title>Test for Bug 548523</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=548523">Mozilla Bug 548523</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<!-- <button onClick="SimpleTest.finish();">Finish</button> --> +<div>Tests complete: <span id="log" style="font-size: small;"></span></div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 548523 **/ + +SimpleTest.requestCompleteLog(); +var manager = new MediaTestManager; + +manager.onFinished = function() { + is(gotLoadEvent, true, "Should not have delayed the load event indefinitely"); +}; + +var test = getPlayableVideo(gSeekTests); +var baseName = test.name; +var gTest = test; +var bogusSrc = "bogus.duh"; +var bogusType = "video/bogus"; +var gotLoadEvent = false; +var finished = false; + +addLoadEvent(function() {gotLoadEvent=true;}); + +function log(m) { + var l = document.getElementById("log"); + // eslint-disable-next-line no-unsanitized/property + l.innerHTML += m; +} + +function maybeFinish(v, n) { + if (v._finished) { + return; + } + v._finished = true; + log(n + ","); + removeNodeAndSource(v); + manager.finished(v.token); +} + +function filename(uri) { + return uri.substr(uri.lastIndexOf("/")+1); +} + +// Every test must have a setup(v) function, and must call maybeFinish() when test is complete. +var tests = [ + { + // 1. Add preload:none video with src to document. Load should halt at NETWORK_IDLE and HAVE_NOTHING, + // after receiving a suspend event. Should not receive loaded events until after we call load(). + // Note the suspend event is explictly sent by our "stop the load" code, but other tests can't rely + // on it for the preload:metadata case, as there can be multiple suspend events when loading metadata. + suspend(e) { + var v = e.target; + is(v._gotLoadStart, true, "(1) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(1) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(1) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(1) NetworkState must be NETWORK_IDLE"); + maybeFinish(v, 1); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + v.src = test.name; + document.body.appendChild(v); // Causes implicit load, which will be halted due to preload:none. + }, + + name: "test1", + }, + { + // 2. Add preload:metadata video with src to document. Should halt with NETWORK_IDLE, HAVE_CURRENT_DATA + // after suspend event and after loadedmetadata. + loadeddata(e) { + var v = e.target; + is(v._gotLoadStart, true, "(2) Must get loadstart."); + is(v._gotLoadedMetaData, true, "(2) Must get loadedmetadata."); + ok(v.readyState >= v.HAVE_CURRENT_DATA, "(2) ReadyState must be >= HAVE_CURRENT_DATA"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(2) NetworkState must be NETWORK_IDLE"); + maybeFinish(v, 2); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "metadata"; + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadeddata", this.loadeddata); + v.src = test.name; + document.body.appendChild(v); // Causes implicit load, which will be halted after + // metadata due to preload:metadata. + }, + + name: "test2", + }, + { + // 3. Add preload:auto to document. Should receive canplaythrough eventually. + canplaythrough(e) { + var v = e.target; + is(v._gotLoadStart, true, "(3) Must get loadstart."); + is(v._gotLoadedMetaData, true, "(3) Must get loadedmetadata."); + maybeFinish(v, 3); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "auto"; + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("canplaythrough", this.canplaythrough); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + + name: "test3", + }, + { + // 4. Add preload:none video to document. Call play(), should load then play through. + suspend(e) { + var v = e.target; + if (v._gotSuspend) { + return; // We can receive multiple suspend events, like the one after download completes. + } + v._gotSuspend = true; + is(v._gotLoadStart, true, "(4) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(4) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(4) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(4) NetworkState must be NETWORK_IDLE"); + v.play(); // Should load and play through. + }, + + ended(e) { + ok(true, "(4) Got playback ended"); + maybeFinish(e.target, 4); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v._gotSuspend = false; + v.preload = "none"; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + v.addEventListener("ended", this.ended); + v.src = test.name; + document.body.appendChild(v); + }, + + name: "test4", + }, + { + // 5. preload:none video without resource, add to document, will implicitly start a + // preload:none load. Add a src, it shouldn't load. + suspend(e) { + var v = e.target; + is(v._gotLoadStart, true, "(5) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(5) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(5) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(5) NetworkState must be NETWORK_IDLE"); + maybeFinish(v, 5); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + document.body.appendChild(v); // Causes implicit load, which will be halted due to no resource. + v.src = test.name; // Load should start, and halt at preload:none. + }, + + name: "test5", + }, + { + // 6. preload:none video without resource, add to document, will implicitly start a + // preload:none load. Add a source, it shouldn't load. + suspend(e) { + var v = e.target; + is(v._gotLoadStart, true, "(6) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(6) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(6) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(6) NetworkState must be NETWORK_IDLE"); + maybeFinish(v, 6); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + document.body.appendChild(v); // Causes implicit load, which will be halted due to no resource. + var s = document.createElement("source"); + s.src = test.name; + s.type = test.type; + v.appendChild(s); // Load should start, and halt at preload:none. + }, + + name: "test6", + }, + { + // 7. create a preload:none document with multiple sources, the first of which is invalid. + // Add to document, then play. It should load and play through the second source. + suspend(e) { + var v = e.target; + if (v._gotSuspend) + return; // We can receive multiple suspend events, like the one after download completes. + v._gotSuspend = true; + is(v._gotLoadStart, true, "(7) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(7) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(7) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(7) NetworkState must be NETWORK_IDLE"); + v.play(); // Should load and play through. + }, + + ended(e) { + ok(true, "(7) Got playback ended"); + var v = e.target; + is(v._gotErrorEvent, true, "(7) Should get error event from first source load failure"); + maybeFinish(v, 7); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "none"; + v._gotErrorEvent = false; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + v.addEventListener("ended", this.ended); + var s1 = document.createElement("source"); + s1.src = "not-a-real-file.404" + s1.type = test.type; + s1.addEventListener("error", function(e){v._gotErrorEvent = true;}); + v.appendChild(s1); + var s2 = document.createElement("source"); + s2.src = test.name; + s2.type = test.type; + v.appendChild(s2); + document.body.appendChild(v); // Causes implicit load, which will be halt at preload:none on the second resource. + }, + + name: "test7", + }, + { + // 8. Change preload value from none to metadata should cause metadata to be loaded. + loadeddata(e) { + var v = e.target; + is(v._gotLoadedMetaData, true, "(8) Must get loadedmetadata."); + ok(v.readyState >= v.HAVE_CURRENT_DATA, "(8) ReadyState must be >= HAVE_CURRENT_DATA on suspend."); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(8) NetworkState must be NETWORK_IDLE when load is halted"); + maybeFinish(v, 8); + }, + + setup(v) { + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadstart", function(e){v.preload = "metadata";}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadeddata", this.loadeddata); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + + name: "test8", + }, + /*{ + // 9. Change preload value from metadata to auto should cause entire media to be loaded. + // For some reason we don't always receive the canplaythrough event, particuarly on this test. + // We've disabled this test until bug 568402 is fixed. + canplaythrough: + function(e) { + var v = e.target; + is(v._gotLoadStart, true, "(9) Must get loadstart."); + is(v._gotLoadedMetaData, true, "(9) Must get loadedmetadata."); + maybeFinish(v, 9); + }, + + setup: + function(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "metadata"; + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}, false); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}, false); + v.addEventListener("loadeddata", function(){v.preload = "auto"}, false); + v.addEventListener("canplaythrough", this.canplaythrough, false); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + },*/ + { + // 10. Change preload value from none to auto should cause entire media to be loaded. + canplaythrough(e) { + var v = e.target; + is(v._gotLoadedMetaData, true, "(10) Must get loadedmetadata."); + maybeFinish(v, 10); + }, + + setup(v) { + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadstart", function(e){v.preload = "auto";}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("canplaythrough", this.canplaythrough); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + + name: "test10", + }, + { + // 11. Change preload value from none to metadata should cause metadata to load. + loadeddata(e) { + var v = e.target; + is(v._gotLoadedMetaData, true, "(11) Must get loadedmetadata."); + ok(v.readyState >= v.HAVE_CURRENT_DATA, "(11) ReadyState must be >= HAVE_CURRENT_DATA."); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(11) NetworkState must be NETWORK_IDLE."); + maybeFinish(v, 11); + }, + + setup(v) { + v._gotLoadedMetaData = false; + v.preload = "none"; + v.addEventListener("loadstart", function(e){v.preload = "metadata";}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadeddata", this.loadeddata); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + + name: "test11", + }, + { + // 13. Change preload value from auto to none after specifying a src + // should load according to preload none, no buffering should have taken place + suspend(e) { + var v = e.target; + is(v._gotLoadStart, true, "(13) Must get loadstart."); + is(v._gotLoadedMetaData, false, "(13) Must not get loadedmetadata."); + is(v.readyState, v.HAVE_NOTHING, "(13) ReadyState must be HAVE_NOTHING"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(13) NetworkState must be NETWORK_IDLE"); + maybeFinish(v, 13); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "auto"; + v.src = test.name; + v.preload = "none"; + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("suspend", this.suspend); + document.body.appendChild(v); // Causes implicit load, should load according to preload none + document.createElement("source"); + }, + + name: "test13", + }, + { + // 14. Add preload:metadata video with src to document. Play(), should play through. + loadeddata(e) { + var v = e.target; + is(v._gotLoadStart, true, "(14) Must get loadstart."); + is(v._gotLoadedMetaData, true, "(14) Must get loadedmetadata."); + ok(v.readyState >= v.HAVE_CURRENT_DATA, "(14) ReadyState must be >= HAVE_CURRENT_DATA"); + // bug 962949 + // is(v.networkState, v.NETWORK_IDLE, "(14) NetworkState must be NETWORK_IDLE"); + v.play(); + }, + + ended(e) { + ok(true, "(14) Got playback ended"); + var v = e.target; + maybeFinish(v, 14); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "metadata"; + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("ended", this.ended); + v.addEventListener("loadeddata", this.loadeddata); + v.src = test.name; + document.body.appendChild(v); // Causes implicit load, which will be halted after + // metadata due to preload:metadata. + }, + + name: "test14", + }, + { + // 15. Autoplay should override preload:none. + ended(e) { + ok(true, "(15) Got playback ended."); + var v = e.target; + maybeFinish(v, 15); + }, + + setup(v) { + v._gotLoadStart = false; + v._gotLoadedMetaData = false; + v.preload = "none"; + v.autoplay = true; + v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}); + v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}); + v.addEventListener("ended", this.ended); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + + // Log events for debugging. + var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(e) { + info(e.target.token + ": got " + e.type); + } + events.forEach(function(e) { + v.addEventListener(e, logEvent); + }); + }, + + name: "test15", + }, + { + // 16. Autoplay should override preload:metadata. + ended(e) { + ok(true, "(16) Got playback ended."); + var v = e.target; + maybeFinish(v, 16); + }, + + setup(v) { + v.preload = "metadata"; + v.autoplay = true; + v.addEventListener("ended", this.ended); + v.src = test.name; // Causes implicit load. + document.body.appendChild(v); + }, + + name: "test16", + }, + { + // 17. On a preload:none video, adding autoplay should disable preload none, i.e. don't break autoplay! + ended(e) { + ok(true, "(17) Got playback ended."); + var v = e.target; + maybeFinish(v, 17); + }, + + setup(v) { + v.addEventListener("ended", this.ended); + v.preload = "none"; + document.body.appendChild(v); // Causes implicit load, which will be halted due to preload:none. + v.autoplay = true; + v.src = test.name; + }, + + name: "test17", + }, + { + // 18. On a preload='none' video, call play() before load algorithms's sync + // has run, the play() call should override preload='none'. + ended(e) { + ok(true, "(18) Got playback ended."); + var v = e.target; + maybeFinish(v, 18); + }, + + setup(v) { + v.addEventListener("ended", this.ended); + v.preload = "none"; + v.src = test.name; // Schedules async section to continue load algorithm. + document.body.appendChild(v); + v.play(); // Should cause preload:none to be overridden. + }, + + name: "test18", + }, + { + // 19. Set preload='auto' on first video source then switching preload='none' and swapping the video source to another. + // The second video should not start playing as it's preload state has been changed to 'none' from 'auto' + setup(v) { + v.preload = "auto"; + v.src = test.name; + // add a listener for when the video has loaded, so we know preload auto has worked + v.onloadedmetadata = function() { + is(v.preload, "auto", "(19) preload is initially auto"); + // set preload state to none and switch video sources + v.preload="none"; + v.src = test.name + "?asdf"; + + v.onloadedmetadata = function() { + ok(false, "(19) 'loadedmetadata' shouldn't fire when preload is none"); + } + + var ontimeout = function() { + v.removeEventListener("suspend", onsuspend); + ok(false, "(19) 'suspend' should've fired"); + maybeFinish(v, 19); + } + var cancel = setTimeout(ontimeout, 10000); + + var onsuspend = function() { + v.removeEventListener("suspend", onsuspend); + clearTimeout(cancel); + is(v.readyState, 0, "(19) no buffering has taken place"); + maybeFinish(v, 19); + } + v.addEventListener("suspend", onsuspend); + } + document.body.appendChild(v); + }, + + name: "test19", + } +]; + +var iterationCount = 0; +function startTest(t, token) { + if (t == tests[0]) { + ++iterationCount; + info("iterationCount=" + iterationCount); + } + if (iterationCount == 2) { + // Do this series of tests on logically different resources + t.name = baseName + "?" + Math.floor(Math.random()*100000); + } + var v = document.createElement("video"); + v.token = token; + t.setup(v); + manager.started(token); +} + +var twiceTests = tests.concat(tests); +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest); +function beginTest() { + manager.runTests(twiceTests, startTest); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_preload_attribute.html b/dom/media/test/test_preload_attribute.html new file mode 100644 index 0000000000..1e415035c5 --- /dev/null +++ b/dom/media/test/test_preload_attribute.html @@ -0,0 +1,44 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=479863 +--> +<head> + <title>Test for Bug 479863</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479863">Mozilla Bug 479863</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> + +<video id='v1'></video><audio id='a1'></audio> +<video id='v2' preload="auto"></video><audio id='a2' preload="auto"></audio> + +<pre id="test"> +<script type="application/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var v2 = document.getElementById('v2'); +var a2 = document.getElementById('a2'); +is(v1.getAttribute("preload"), null, "video preload via getAttribute should be null by default"); +is(a1.getAttribute("preload"), null, "video preload via getAttribute should be null by default"); +is(v1.preload, "", "v1.preload should be empty by default"); +is(a1.preload, "", "a1.preload should be empty by default"); +is(v2.preload, "auto", "v2.preload should be auto"); +is(a2.preload, "auto", "a2.preload should be auto"); + +v1.preload = "auto"; +a1.preload = "auto"; +is(v1.preload, "auto", "video.preload should be auto"); +is(a1.preload, "auto", "audio.preload should be auto"); +is(v1.getAttribute("preload"), "auto", "video preload attribute should be auto"); +is(a1.getAttribute("preload"), "auto", "video preload attribute should be auto"); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_preload_suspend.html b/dom/media/test/test_preload_suspend.html new file mode 100644 index 0000000000..7f1146360f --- /dev/null +++ b/dom/media/test/test_preload_suspend.html @@ -0,0 +1,112 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for Bug 479863</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +function checkSuspendCount(evt) { + var v = evt.target; + ++v.suspendCount; + is(v.networkState, v.NETWORK_IDLE, v.name + " got suspended, count=" + v.suspendCount); + if (v.suspendCount == v.expectedSuspendCount) { + removeNodeAndSource(v); + manager.finished(v.name); + } + if (v.suspendCount > v.expectedSuspendCount) { + ok(false, v.name + " got too many suspend events"); + } +} + +var tests = [ + { + name: 'v1', + preload: 'none', + expectedSuspendCount: 2, + onsuspend(evt) { + checkSuspendCount(evt); + if (evt.target.suspendCount == 1) { + evt.target.preload = 'auto'; + } + } + }, + { + name: 'v2', + preload: 'auto', + expectedSuspendCount: 1, + onsuspend: checkSuspendCount + }, + { + name: 'v3', + preload: 'none', + autoplay: true, + expectedSuspendCount: 1, + onsuspend: checkSuspendCount + }, + { + name: 'v4', + preload: 'none', + expectedSuspendCount: 2, + onsuspend(evt) { + checkSuspendCount(evt); + if (evt.target.suspendCount == 1) { + evt.target.play(); + } + } + }, + // disable v5 since media element doesn't support 'load' event anymore. + /*{ + name: 'v5', + preload: 'none', + expectedSuspendCount: 2, + onsuspend: function(evt) { + checkSuspendCount(evt); + if (evt.target.suspendCount == 1) { + evt.target.currentTime = 0.1; + } + } + },*/ + { + name: 'v6', + preload: 'none', + expectedSuspendCount: 2, + onsuspend(evt) { + checkSuspendCount(evt); + if (evt.target.suspendCount == 1) { + evt.target.autoplay = true; + } + } + } +]; + +function startTest(test, token) { + var v = document.createElement("video"); + v.name = test.name; + var key = Math.random(); + v.src = "seek.ogv?key=" + key + "&id=" + v.name; + v.preload = test.preload; + v.suspendCount = 0; + v.expectedSuspendCount = test.expectedSuspendCount; + if (test.autoplay) { + v.autoplay = true; + } + v.onsuspend = test.onsuspend; + document.body.appendChild(v); + manager.started(v.name); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, function() { + manager.runTests(tests, startTest); +}); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_preserve_playbackrate_after_ui_play.html b/dom/media/test/test_preserve_playbackrate_after_ui_play.html new file mode 100644 index 0000000000..98bdd66675 --- /dev/null +++ b/dom/media/test/test_preserve_playbackrate_after_ui_play.html @@ -0,0 +1,60 @@ +<!DOCTYPE HTML> +<html> +<head> + <title> Bug 1013933 - preserve playbackRate after clicking play button </title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/SimpleTest/EventUtils.js"></script> + <script type="application/javascript" src="browserElementTestHelpers.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> + +<div id="content"> + <video width="320" height="240" id="video" controls mozNoDynamicControls preload="auto"></video> +</div> + +<script type="text/javascript"> +/* + * Positions of the UI elements, relative to the upper-left corner of the + * <video> box. + */ +const videoHeight = 240; +const playButtonWidth = 28; +const playButtonHeight = 28; +const playButtonCenterX = 0 + Math.round(playButtonWidth / 2); +const playButtonCenterY = videoHeight - Math.round(playButtonHeight / 2); + +var expectedPlaybackRate = 0.5 + +function runTest() { + var video = document.getElementById("video"); + video.src = "audio.wav"; + video.loop = true; + video.playbackRate = expectedPlaybackRate; + + video.oncanplaythrough = function() { + video.oncanplaythrough = null; + is(video.paused, true, "video is not playing yet."); + is(video.playbackRate, expectedPlaybackRate, + "playbackRate is correct before clicking play button."); + + // Click the play button + synthesizeMouse(video, playButtonCenterX, playButtonCenterY, { }); + }; + + video.onplay = function() { + video.onplay = null; + is(video.paused, false, "video starts playing."); + is(video.playbackRate, expectedPlaybackRate, + "playbackRate is correct after clicking play button."); + video.pause(); + SimpleTest.finish(); + }; +} + +window.addEventListener("load", runTest); +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_progress.html b/dom/media/test/test_progress.html new file mode 100644 index 0000000000..b958b2b335 --- /dev/null +++ b/dom/media/test/test_progress.html @@ -0,0 +1,52 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: progress events</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function do_progress(e) { + var v = e.target; + ok(!v._finished, "Check no progress events after completed for " + v._name); +} + +function do_ended(e) { + var v = e.target; + ok(!v._finished, "Only one ended event for " + v._name); + v._finished = true; + v.removeEventListener("ended", do_ended); + v.removeEventListener("progress", do_progress); + removeNodeAndSource(v); + manager.finished(v.token); +} + +function startTest(test, token) { + var type = /^video/.test(test.type) ? "video" : "audio"; + var v = document.createElement(type); + v.token = token; + manager.started(token); + v.src = test.name; + v.autoplay = true; + v._name = test.name; + v._finished = false; + v.addEventListener("ended", do_ended); + v.addEventListener("progress", do_progress); + document.body.appendChild(v); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest); +function beginTest() { + manager.runTests(gProgressTests, startTest); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_reactivate.html b/dom/media/test/test_reactivate.html new file mode 100644 index 0000000000..43675f7771 --- /dev/null +++ b/dom/media/test/test_reactivate.html @@ -0,0 +1,64 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test reactivation of a media element from a dead document</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> + +<iframe id="frame" src="reactivate_helper.html"></iframe> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var elements; + +function playElement(e) { + // All elements played out, finish the test case. + if (!e) { + SimpleTest.finish(); + return; + } + + e.play(); + info("Element play: " + e._name); + var reviveElement = function() { + document.body.appendChild(e); + e.onended = function() { + info("Element ended: " + e._name); + removeNodeAndSource(e); + // Play next element. + playElement(elements.pop()); + } + } + setTimeout(reviveElement, 2000); +} + +function loadedAll(elementList) { + elements = elementList; + + // Log events for debugging. + var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(e) { + info(e.target._name + ": got " + e.type); + } + elementList.forEach(function(element) { + events.forEach(function(evt) { + element.addEventListener(evt, logEvent); + }); + }); + + // Blow away the subframe + document.body.removeChild(document.getElementById("frame")); + + // Play elements one by one to avoid hitting bug 847903 on MacOSX. + playElement(elements.pop()); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_readyState.html b/dom/media/test/test_readyState.html new file mode 100644 index 0000000000..17c363c503 --- /dev/null +++ b/dom/media/test/test_readyState.html @@ -0,0 +1,51 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: readyState</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<video id='v1'></video><audio id='a1'></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +"use strict"; +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var passed = "truthy"; + +is(v1.readyState, 0); +is(a1.readyState, 0); + +try { + v1.readyState = 0; +} catch (e) { + passed = !passed; +} +try { + a1.readyState = 0; +} catch (e) { + passed = !passed; +} +ok(passed === true, + "Setting readyState throws in strict mode (readonly attribute)"); +</script> + +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById('v1'); +var a1 = document.getElementById('a1'); +var passed = false; + +is(v1.readyState, 0); +is(a1.readyState, 0); + +try { + v1.readyState = 1; + a1.readyState = 1; + passed = v1.readyState === 0 && a1.readyState === 0; +} catch(e) { } +ok(passed, "Should not be able to set readyState (readonly attribute)"); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_referer.html b/dom/media/test/test_referer.html new file mode 100644 index 0000000000..2561e72b6a --- /dev/null +++ b/dom/media/test/test_referer.html @@ -0,0 +1,88 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=584480 +--> +<head> + <title>Test for Bug 584480</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=584480">Mozilla Bug </a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> +var media = []; + +function checkComplete() { + for (var i=0; i<media.length; ++i) { + if (!media[i]._complete) { + return; + } + } + + SimpleTest.finish(); +} + +function removeNode(v) { + v.removeEventListener("error", loadError); + v.removeEventListener("loadedmetadata", loadedMetadata); + v.remove(); + v.src = ""; +} + +function loadError(evt) { + // If no referer is sent then the sjs returns an error + ok(false, "check referer is sent with media request"); + evt.target._complete = true; + checkComplete(); + removeNode(evt.target); +} + +function loadedMetadata(evt) { + // If a referer is sent then the sjs returns a valid media + ok(true, "check referer is sent with media request"); + evt.target._complete = true; + checkComplete(); + removeNode(evt.target); +} + +// Create all media objects. +for (var i=0; i<gSmallTests.length; ++i) { + var test = gSmallTests[i]; + var type; + if (/^video/.test(test.type)) { + type = "video" + } else { + type = "audio"; + } + var v = document.createElement(type); + if (!v.canPlayType(test.type)) { + continue; + } + // ensure metadata is loaded for default preload is none on b2g + v.preload = "metadata"; + v.autoplay = "true"; + v._complete = false; + v.addEventListener("error", loadError); + v.addEventListener("loadedmetadata", loadedMetadata); + v.src = 'referer.sjs?name=' + test.name + '&type=' + test.type; + document.body.appendChild(v); // Will start load. + media.push(v); +} + +if (media.length == 0) { + todo(false, "No types supported"); +} else { + SimpleTest.waitForExplicitFinish(); +} +</script> +</pre> + +</body> +</html> diff --git a/dom/media/test/test_replay_metadata.html b/dom/media/test/test_replay_metadata.html new file mode 100644 index 0000000000..dab022885f --- /dev/null +++ b/dom/media/test/test_replay_metadata.html @@ -0,0 +1,120 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=467972 +--> +<head> + <title>Test for Bug 467972</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=467972">Mozilla Bug 467972</a> + +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +// Test for Bug 467972. Tests that when we play to end, seek to 0, and play again, that we don't +// send/receive multiple loadeddata and loadedmetadata events. + +var manager = new MediaTestManager; + +function seekStarted(evt) { + var v = evt.target; + v._gotSeekStarted = true; +} + +function seekEnded(evt) { + var v = evt.target; + v._gotSeekEnded = true; + v.play(); +} + +function loadedData(evt) { + var v = evt.target; + v._loadedDataCount++; + ok(v._loadedDataCount <= 1, "No more than 1 onloadeddata event for " + v._name); +} + +function loadedMetaData(evt) { + var v = evt.target; + v._loadedMetaDataCount++; + ok(v._loadedMetaDataCount <= 1, "No more than 1 onloadedmetadata event for " + v._name); + checkMetadata(v._name, v, v._test); + v.play(); +} + +function playing(evt) { + evt.target._playingCount++; +} + +function removeNodeAndListener(n) { + n.removeEventListener("loadedmetadata", loadedMetaData); + n.removeEventListener("ended", playbackEnded); + n.removeEventListener("playing", playing); + n.removeEventListener("loadeddata", loadedData); + n.removeEventListener("seeking", seekStarted); + n.removeEventListener("seeked", seekEnded); + removeNodeAndSource(n); +} + +function playbackEnded(evt) { + var v = evt.target; + v._endCount++; + ok(v.currentTime >= v.duration-0.1 && v.currentTime <= v.duration + 0.1, + "CurrentTime (" + v.currentTime + ") should be around " + v.duration + + " for " + v._name); + if (!v._playedOnce) { + v.currentTime = 0; + ok(v.seeking, "seeking should be true for " + v._name); + ok(!v.ended, "ended shouldn't be true once seeking has begun for " + v._name); + v._playedOnce = true; + } else { + ok(v._gotSeekEnded, "Should have received seekended for " + v._name); + ok(v._gotSeekStarted, "Should have received seekstarted for " + v._name); + is(v._loadedDataCount, 1, "Should have 1 onloadeddata event for " + v._name); + is(v._loadedMetaDataCount, 1, "Should have 1 onloadedmetadata event for " + v._name); + is(v._endCount, 2, "Should have received two ended events for " + v._name); + ok(v._playingCount > 0, "Should have at least one playing event for " + v._name); + v._finished = true; + removeNodeAndListener(v); + manager.finished(v.token); + } +} + +function startTest(test, token) { + var v = document.createElement('video'); + v.preload = "auto"; + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name; + v._playedOnce = false; + v._gotSeekEnded = false; + v._gotSeekStarted = false; + v._loadedDataCount = 0; + v._loadedMetaDataCount = 0; + v._playingCount = 0; + v._endCount = 0; + v._test = test; + v._finished = false; + v.addEventListener("loadedmetadata", loadedMetaData); + v.addEventListener("ended", playbackEnded); + v.addEventListener("playing", playing); + v.addEventListener("loadeddata", loadedData); + v.addEventListener("seeking", seekStarted); + v.addEventListener("seeked", seekEnded); + document.body.appendChild(v); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_reset_events_async.html b/dom/media/test/test_reset_events_async.html new file mode 100644 index 0000000000..482ec55986 --- /dev/null +++ b/dom/media/test/test_reset_events_async.html @@ -0,0 +1,58 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=975270 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug </title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="application/javascript" src="manifest.js"></script> + <script type="application/javascript"> + + /** Test for Bug 975270 **/ + // Test that 'emptied' and 'abort' events are fired asynchronously when re-starting + // media load. + SimpleTest.waitForExplicitFinish(); + + var a = document.createElement("audio"); + a._abort = 0; + a._emptied = 0; + a.preload = "metadata"; // On B2G we default to preload:none. + + is(a.networkState, HTMLMediaElement.NETWORK_EMPTY, "Shouldn't be loading"); + + a.addEventListener("abort", function(e) { a._abort++; }); + a.addEventListener("emptied", function(e) { a._emptied++; }); + a.addEventListener("loadedmetadata", + function(e) { + is(a._abort, 0, "Should not have received 'abort' before 'loadedmetadata"); + is(a._emptied, 0, "Should not have received 'emptied' before 'loadedmetadata"); + + a.addEventListener("loadstart", + function() { + is(a._abort, 1, "Should have received 'abort' before 'loadstart"); + is(a._emptied, 1, "Should have received 'emptied' before 'loadstart"); + SimpleTest.finish(); + }); + + a.src = ""; + is(a._abort, 0, "Should not have received 'abort' during setting a.src=''"); + is(a._emptied, 0, "Should not have received 'emptied' during setting a.src=''"); + }); + + a.src = getPlayableAudio(gSmallTests).name; + + </script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=">Mozilla Bug </a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> +</body> +</html> diff --git a/dom/media/test/test_reset_src.html b/dom/media/test/test_reset_src.html new file mode 100644 index 0000000000..1912369dac --- /dev/null +++ b/dom/media/test/test_reset_src.html @@ -0,0 +1,99 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=804875 +--> + +<head> + <title>Test for bug 804875</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="text/javascript" src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<a target="_blank" +href="https://bugzilla.mozilla.org/show_bug.cgi?id=804875">Mozilla Bug 804875</a> + +<canvas></canvas> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function finish(v) { + removeNodeAndSource(v); + manager.finished(v.token); +} + +function onLoadedData_Audio(e) { + var t = e.target; + is(t.videoHeight, 0, t.name + ": videoHeight should be zero when there is no video."); + is(t.videoWidth, 0, t.name + ": videoWidth should be zero when there is no video."); + is(t.mozPaintedFrames, 0, t.name + ": mozPaintedFrames should be zero when there is no video."); + is(t.mozFrameDelay, 0, t.name + ": mozFrameDelay should be zero when there is no video."); + var c = document.getElementsByTagName("canvas")[0].getContext("2d"); + try { + c.drawImage(t, 0, 0, t.videoHeight, t.videoWidth); + } catch (ex) { + ok(true, t.name + ": Trying to draw to a canvas should throw, since we don't have a frame anymore"); + finish(t); + return; + } + ok(false, t.name + ": We should not succeed to draw a video frame on the canvas."); + finish(t); +} + +function onTimeUpdate_Video(e) { + var t = e.target; + if (t.currentTime < t.duration / 4) { + return; + } + t.removeEventListener("timeupdate", onTimeUpdate_Video); + // There's no guarantee that a video frame composite notification reaches + // us before timeupdate fires. + ok(t.mozPaintedFrames >= 0, t.name + ": mozPaintedFrames should be positive or zero, is " + t.mozPaintedFrames + "."); + ok(t.mozFrameDelay >= 0, t.name + ": mozFrameDelay should be positive or zero, is " + t.mozFrameDelay + "."); + + if (t._firstTime) { + // eslint-disable-next-line no-self-assign + t.src = t.src; + t._firstTime = false; + } else { + var source = getPlayableAudio(gPlayTests); + if (!source) { + todo("No audio file available.") + finish(t); + } else { + t.removeEventListener("loadeddata", onLoadedData_Video); + t.addEventListener("loadeddata", onLoadedData_Audio); + t.src = source.name; + } + } +} + +function onLoadedData_Video(e) { + var t = e.target; + isnot(t.videoHeight, 0, t.name + ": We should have a videoHeight."); + isnot(t.videoWidth, 0, t.name + ": We should have a videoWidth."); + t.addEventListener("timeupdate", onTimeUpdate_Video); + t.play(); +} + +function startTest(test, token) { + var v = document.createElement('video'); + document.body.appendChild(v); + v._firstTime = true; + v.addEventListener("loadeddata", onLoadedData_Video); + v.src = test.name; + v.token = token; + v.name = test.name; + v.play(); + manager.started(token); +} + +manager.runTests(getPlayableVideos(gSmallTests.concat(gSeekTests)), startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_resolution_change.html b/dom/media/test/test_resolution_change.html new file mode 100644 index 0000000000..403f9b505d --- /dev/null +++ b/dom/media/test/test_resolution_change.html @@ -0,0 +1,52 @@ +<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of files with resolution changes</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function loadedData(e) {
+ var v = e.target;
+ v.addEventListener("resize", resize);
+ v.play();
+}
+
+function resize(e) {
+ var v = e.target;
+ v.seenResolutionChange = true;
+}
+
+function ended(e) {
+ var v = e.target;
+ ok(v.seenResolutionChange, v.token + ": A resolution change should have ocurred by the end of playback");
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ v.src = test.name;
+ v.seenResolutionChange = false;
+
+ v.addEventListener("loadeddata", loadedData)
+ v.addEventListener("ended", ended);
+
+ manager.started(token);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gResolutionChangeTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_resume.html b/dom/media/test/test_resume.html new file mode 100644 index 0000000000..0e22894be1 --- /dev/null +++ b/dom/media/test/test_resume.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: Test resume of server-dropped connections</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<audio preload="auto" id="a"></audio> +<iframe id="f"></iframe> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var key = Math.round(Math.random()*1000000000); +var a = document.getElementById("a"); +var f = document.getElementById("f"); + +function didEnd() { + ok(a.currentTime > 2.26, "Reached correct end time (got " + a.currentTime + ", expected > 2.26"); + SimpleTest.finish(); +} + +function didSendCancel() { + a.addEventListener("ended", didEnd); + a.play(); +} + +function didSuspend() { + a.removeEventListener("suspend", didSuspend); + + // Cache must have filled up, or something. Tell the Web server to drop + // our connection. + f.addEventListener("load", didSendCancel); + f.src = "cancellable_request.sjs?cancelkey=" + key; +} + +if (!a.canPlayType("audio/wave")) { + todo(false, "Test requires support for audio/wave"); +} else { + a.addEventListener("suspend", didSuspend); + a.src = "cancellable_request.sjs?key=" + key; + SimpleTest.waitForExplicitFinish(); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seamless_looping.html b/dom/media/test/test_seamless_looping.html new file mode 100644 index 0000000000..c9e3253a5d --- /dev/null +++ b/dom/media/test/test_seamless_looping.html @@ -0,0 +1,185 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for seamless loop of HTMLAudioElements</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<canvas id="canvas" width="300" height="300"></canvas> +<script type="application/javascript"> +/** + * This test is used to ensure every time we loop audio, the audio can loop + * seamlessly which means there won't have any silenece or noise between the + * end and the start. + */ + +SimpleTest.waitForExplicitFinish(); + +// Set DEBUG to true to add a canvas with a little drawing of what is going +// on, and actually outputs the audio to the speakers. +var DEBUG = true; +var LOOPING_COUNT = 0; +var MAX_LOOPING_COUNT = 10; +var TONE_FREQUENCY = 440; + +(async function testSeamlesslooping() { + info(`- create looping audio element -`); + let audio = createAudioElement(); + + info(`- start audio and analyze audio wave data to ensure looping audio without any silence or noise -`); + await playAudioAndStartAnalyzingWaveData(audio); + + info(`- test seamless looping multiples times -`); + for (LOOPING_COUNT = 0; LOOPING_COUNT < MAX_LOOPING_COUNT; LOOPING_COUNT++) { + await once(audio, "seeked"); + info(`- the round ${LOOPING_COUNT} of the seamless looping succeeds -`); + } + + info(`- end of seamless looping test -`); + SimpleTest.finish(); +})(); + +/** + * Test utility functions + */ +function createSrcBuffer() { + // Generate the sine in floats, then convert, for simplicity. + let channels = 1; + let sampleRate = 44100; + let buffer = new Float32Array(sampleRate * channels); + let phase = 0; + const TAU = 2 * Math.PI; + for (let i = 0; i < buffer.length; i++) { + // Adjust the gain a little so we're sure it's not going to clip. This is + // important because we're converting to 16bit integer right after, and + // clipping will clearly introduce a discontinuity that will be + // mischaracterized as a looping click. + buffer[i] = Math.sin(phase) * 0.99; + phase += TAU * TONE_FREQUENCY / 44100; + if (phase > 2 * TAU) { + phase -= TAU; + } + } + + // Make a RIFF header, it's 23 bytes + let buf = new Int16Array(buffer.length + 23); + buf[0] = 0x4952; + buf[1] = 0x4646; + buf[2] = (2 * buffer.length + 15) & 0x0000ffff; + buf[3] = ((2 * buffer.length + 15) & 0xffff0000) >> 16; + buf[4] = 0x4157; + buf[5] = 0x4556; + buf[6] = 0x6d66; + buf[7] = 0x2074; + buf[8] = 0x0012; + buf[9] = 0x0000; + buf[10] = 0x0001; + buf[11] = 1; + buf[12] = 44100 & 0x0000ffff; + buf[13] = (44100 & 0xffff0000) >> 16; + buf[14] = (2 * channels * sampleRate) & 0x0000ffff; + buf[15] = ((2 * channels * sampleRate) & 0xffff0000) >> 16; + buf[16] = 0x0004; + buf[17] = 0x0010; + buf[18] = 0x0000; + buf[19] = 0x6164; + buf[20] = 0x6174; + buf[21] = (2 * buffer.length) & 0x0000ffff; + buf[22] = ((2 * buffer.length) & 0xffff0000) >> 16; + + // convert to int16 and copy. + for (let i = 0; i < buffer.length; i++) { + buf[i + 23] = Math.round(buffer[i] * (1 << 15)); + } + return buf; +} + +function createAudioElement() { + /* global audio */ + window.audio = document.createElement("audio"); + audio.src = URL.createObjectURL(new Blob([createSrcBuffer()], + { type: 'audio/wav' })); + audio.controls = true; + audio.loop = true; + document.body.appendChild(audio); + return audio; +} + +async function playAudioAndStartAnalyzingWaveData(audio) { + createAudioWaveAnalyser(audio); + ok(await once(audio, "canplay").then(() => true, () => false), + `audio can start playing.`) + ok(await audio.play().then(() => true, () => false), + `audio started playing successfully.`); +} + +function createAudioWaveAnalyser(source) { + /* global ac, analyser */ + window.ac = new AudioContext(); + window.analyser = ac.createAnalyser(); + analyser.frequencyBuf = new Float32Array(analyser.frequencyBinCount); + analyser.smoothingTimeConstant = 0; + analyser.fftSize = 2048; // 1024 bins + + let sourceNode = ac.createMediaElementSource(source); + sourceNode.connect(analyser); + + if (DEBUG) { + analyser.connect(ac.destination); + analyser.timeDomainBuf = new Float32Array(analyser.frequencyBinCount); + let cvs = document.querySelector("canvas"); + analyser.c = cvs.getContext("2d"); + analyser.w = cvs.width; + analyser.h = cvs.height; + } + + analyser.notifyAnalysis = () => { + if (LOOPING_COUNT >= MAX_LOOPING_COUNT) { + return; + } + let {frequencyBuf} = analyser; + analyser.getFloatFrequencyData(frequencyBuf); + // Let things stabilize at the beginning. See bug 1441509. + if (LOOPING_COUNT > 1) { + analyser.doAnalysis(frequencyBuf, ac.sampleRate); + } + + if (DEBUG) { + let {c, w, h, timeDomainBuf} = analyser; + c.clearRect(0, 0, w, h); + analyser.getFloatTimeDomainData(timeDomainBuf); + for (let i = 0; i < frequencyBuf.length; i++) { + c.fillRect(i, h, 1, -frequencyBuf[i] + analyser.minDecibels); + } + + for (let i = 0; i < timeDomainBuf.length; i++) { + c.fillRect(i, h / 2, 1, -timeDomainBuf[i] * h / 2); + } + } + + requestAnimationFrame(analyser.notifyAnalysis); + } + + analyser.doAnalysis = (buf, ctxSampleRate) => { + // The size of an FFT is twice the number of bins in its output. + let fftSize = 2 * buf.length; + // first find a peak where we expect one. + let binIndexTone = 1 + Math.round(TONE_FREQUENCY * fftSize / ctxSampleRate); + ok(buf[binIndexTone] > -25, + `Could not find a peak: ${buf[binIndexTone]} db at ${TONE_FREQUENCY}Hz`); + + // check that the energy some octaves higher is very low. + let binIndexOutsidePeak = 1 + Math.round(TONE_FREQUENCY * 4 * buf.length / ctxSampleRate); + ok(buf[binIndexOutsidePeak] < -110, + `Found unexpected high frequency content: ${buf[binIndexOutsidePeak]}db at ${TONE_FREQUENCY * 4}Hz`); + } + + analyser.notifyAnalysis(); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-1.html b/dom/media/test/test_seek-1.html new file mode 100644 index 0000000000..8690a475b6 --- /dev/null +++ b/dom/media/test/test_seek-1.html @@ -0,0 +1,84 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 1; + +function test_seek1(v, seekTime, is, ok, finish) { + +var startPassed = false; +var endPassed = false; +var seekFlagStart = false; +var seekFlagEnd = false; +var readonly = true; +var completed = false; + +function startTest() { + ok(!completed, "Should not be completed yet"); + ok(!v.seeking, "seeking should default to false"); + try { + v.seeking = true; + readonly = v.seeking === false; + } + catch(e) { + readonly = "threw exception: " + e; + } + is(readonly, true, "seeking should be readonly"); + + v.currentTime = seekTime; + seekFlagStart = v.seeking; +} + +function seekStarted() { + ok(!completed, "should not be completed yet"); + ok(Math.abs(v.currentTime - seekTime) < 0.1, + "Video currentTime should be around " + seekTime + ": " + v.currentTime + " (seeking)"); + startPassed = true; +} + +function seekEnded() { + ok(!completed, "shuld not be completed yet"); + ok(Math.abs(v.currentTime - seekTime) < 0.1, + "Video currentTime should be around " + seekTime + ": " + v.currentTime + " (seeked)"); + endPassed = true; + seekFlagEnd = v.seeking; + v.play(); +} + +function playbackEnded() { + ok(!completed, "should not be completed yet"); + + completed = true; + ok(startPassed, "seeking event"); + ok(endPassed, "seeked event"); + ok(seekFlagStart, "seeking flag on start should be true"); + ok(!seekFlagEnd, "seeking flag on end should be false"); + finish(); +} + +once(v, "ended", playbackEnded); +once(v, "loadedmetadata", startTest); +once(v, "seeking", seekStarted); +once(v, "seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-10.html b/dom/media/test/test_seek-10.html new file mode 100644 index 0000000000..6c02384ed4 --- /dev/null +++ b/dom/media/test/test_seek-10.html @@ -0,0 +1,56 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 10; + +function test_seek10(v, seekTime, is, ok, finish) { + +// Test bug 523335 - ensure that if we close a stream while seeking, we +// don't hang during shutdown. This test won't "fail" per se if it's regressed, +// it will instead start to cause random hangs in the mochitest harness on +// shutdown. + +function startTest() { + // Must be duration*0.9 rather than seekTime, else we don't hit that problem. + // This is probably due to the seek bisection finishing too quickly, before + // we can close the stream. + v.currentTime = v.duration * 0.9; +} + +function done(evt) { + ok(true, "We don't acutally test anything..."); + finish(); +} + +function seeking() { + ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime); + v.onerror = done; + v.src = "not a valid video file."; + v.load(); // Cause the existing stream to close. +} + +v.addEventListener("loadeddata", startTest); +v.addEventListener("seeking", seeking); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-11.html b/dom/media/test/test_seek-11.html new file mode 100644 index 0000000000..2207f684d7 --- /dev/null +++ b/dom/media/test/test_seek-11.html @@ -0,0 +1,76 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +PARALLEL_TESTS = 1; +const SEEK_TEST_NUMBER = 11; + +function test_seek11(v, seekTime, is, ok, finish) { + +// Test for bug 476973, multiple seeks to the same position shouldn't cause problems. + +var seekedNonZero = false; +var completed = false; +var target = 0; + +function startTest() { + if (completed) + return; + target = v.duration / 2; + v.currentTime = target; + v.currentTime = target; + v._seekTarget = target; +} + +function startSeeking() { + ok(v.currentTime >= v._seekTarget - 0.1, + "Video currentTime should be around " + v._seekTarget + ": " + v.currentTime); + if (!seekedNonZero) { + v.currentTime = target; + v._seekTarget = target; + seekedNonZero = true; + } +} + +function seekEnded() { + if (completed) + return; + + if (v.currentTime > 0) { + ok(v.currentTime > target - 0.1 && v.currentTime < target + 0.1, + "Seek to wrong destination " + v.currentTime); + v.currentTime = 0.0; + v._seekTarget = 0.0; + } else { + ok(seekedNonZero, "Successfully seeked to nonzero"); + ok(true, "Seek back to zero was successful"); + completed = true; + finish(); + } +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", startSeeking); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-12.html b/dom/media/test/test_seek-12.html new file mode 100644 index 0000000000..28decabadc --- /dev/null +++ b/dom/media/test/test_seek-12.html @@ -0,0 +1,59 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 12; + +function test_seek12(v, seekTime, is, ok, finish) { +var completed = false; + +function startTest() { + if (completed) + return; + ok(!v.seeking, "seeking should default to false"); + v.currentTime = seekTime; + is(v.currentTime, seekTime, "currentTime must report seek target immediately"); + is(v.seeking, true, "seeking flag on start should be true"); +} + +function seekStarted() { + if (completed) + return; + //is(v.currentTime, seekTime, "seeking: currentTime must be seekTime"); + ok(Math.abs(v.currentTime - seekTime) < 0.01, "seeking: currentTime must be seekTime"); +} + +function seekEnded() { + if (completed) + return; + completed = true; + //is(v.currentTime, seekTime, "seeked: currentTime must be seekTime"); + ok(Math.abs(v.currentTime - seekTime) < 0.01, "seeked: currentTime must be seekTime"); + is(v.seeking, false, "seeking flag on end should be false"); + finish(); +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-13.html b/dom/media/test/test_seek-13.html new file mode 100644 index 0000000000..81eda9b658 --- /dev/null +++ b/dom/media/test/test_seek-13.html @@ -0,0 +1,72 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 13; + +function test_seek13(v, seekTime, is, ok, finish) { +var completed = false; + +function startTest() { + if (completed) + return; + ok(!v.seeking, "seeking should default to false"); + v.currentTime = v.duration; + is(v.currentTime, v.duration, "currentTime must report seek target immediately"); + is(v.seeking, true, "seeking flag on start should be true"); +} + +function seekStarted() { + if (completed) + return; + //is(v.currentTime, v.duration, "seeking: currentTime must be duration"); + ok(Math.abs(v.currentTime - v.duration) < 0.01, + "seeking: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")"); +} + +function seekEnded() { + if (completed) + return; + //is(v.currentTime, v.duration, "seeked: currentTime must be duration"); + ok(Math.abs(v.currentTime - v.duration) < 0.01, + "seeked: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")"); + is(v.seeking, false, "seeking flag on end should be false"); +} + +function playbackEnded() { + if (completed) + return; + completed = true; + //is(v.currentTime, v.duration, "ended: currentTime must be duration"); + ok(Math.abs(v.currentTime - v.duration) < 0.01, + "ended: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")"); + is(v.seeking, false, "seeking flag on end should be false"); + is(v.ended, true, "ended must be true"); + finish(); +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); +v.addEventListener("ended", playbackEnded); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-14.html b/dom/media/test/test_seek-14.html new file mode 100644 index 0000000000..7e19fe3bd3 --- /dev/null +++ b/dom/media/test/test_seek-14.html @@ -0,0 +1,43 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +const SEEK_TEST_NUMBER = 14; + +function test_seek14(v, seekTime, is, ok, finish) { + var completed = false; + + function startTest() { + v.play(); + v.currentTime = v.duration; + } + + function playbackEnded() { + if (completed) { + ok(false, "'ended' should only fire once."); + return; + } + completed = true; + // Finish the test after 700ms. We should receive only one 'ended' event. + setTimeout(finish, 700); + } + + v.addEventListener("loadedmetadata", startTest); + v.addEventListener("ended", playbackEnded); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-2.html b/dom/media/test/test_seek-2.html new file mode 100644 index 0000000000..7b666df961 --- /dev/null +++ b/dom/media/test/test_seek-2.html @@ -0,0 +1,76 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +PARALLEL_TESTS = 1; +const SEEK_TEST_NUMBER = 2; + +function test_seek2(v, seekTime, is, ok, finish) { + +// Test seeking works if current time is set before video is +// playing. +var startPassed = false; +var endPassed = false; +var completed = false; + +function startTest() { + if (completed) + return; + + v.currentTime=seekTime; + v.play(); +} + +function seekStarted() { + if (completed) + return; + + ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime); + startPassed = true; +} + +function seekEnded() { + if (completed) + return; + + endPassed = true; +} + +function playbackEnded() { + if (completed) + return; + + completed = true; + ok(startPassed, "send seeking event"); + ok(endPassed, "send seeked event"); + ok(v.ended, "Checking playback has ended"); + ok(Math.abs(v.currentTime - v.duration) <= 0.1, "Checking currentTime at end: " + v.currentTime); + finish(); +} + +v.addEventListener("ended", playbackEnded); +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-3.html b/dom/media/test/test_seek-3.html new file mode 100644 index 0000000000..c030f03d20 --- /dev/null +++ b/dom/media/test/test_seek-3.html @@ -0,0 +1,68 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 3; + +function test_seek3(v, seekTime, is, ok, finish) { + +// Test seeking works if current time is set but video is not played. +var completed = false; +var gotTimeupdate = false; + +function startTest() { + if (completed) + return; + + v.currentTime=seekTime; +} + +function timeupdate() { + gotTimeupdate = true; + v.removeEventListener("timeupdate", timeupdate); +} + +function seekStarted() { + if (completed) + return; + + ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime); + v.addEventListener("timeupdate", timeupdate); +} + +function seekEnded() { + if (completed) + return; + + var t = v.currentTime; + ok(Math.abs(t - seekTime) <= 0.1, "Video currentTime should be around " + seekTime + ": " + t); + ok(gotTimeupdate, "Should have got timeupdate between seeking and seekended"); + completed = true; + finish(); +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-4.html b/dom/media/test/test_seek-4.html new file mode 100644 index 0000000000..4e5c1fee59 --- /dev/null +++ b/dom/media/test/test_seek-4.html @@ -0,0 +1,70 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 4; + +function test_seek4(v, seekTime, is, ok, finish) { + +// Test for a seek, followed by another seek before the first is complete. +var seekCount = 0; +var completed = false; + +function startTest() { + if (completed) + return; + + v.currentTime=seekTime; + v._seekTarget=seekTime; +} + +function seekStarted() { + if (completed) + return; + + seekCount += 1; + + ok(v.currentTime >= v._seekTarget - 0.1, + "Video currentTime should be around " + v._seekTarget + ": " + v.currentTime); + if (seekCount == 1) { + v.currentTime=seekTime/2; + v._seekTarget=seekTime/2; + } +} + +function seekEnded() { + if (completed) + return; + + if (seekCount == 2) { + ok(Math.abs(v.currentTime - seekTime/2) <= 0.1, "seek on target: " + v.currentTime); + completed = true; + finish(); + } +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-5.html b/dom/media/test/test_seek-5.html new file mode 100644 index 0000000000..a86477f77c --- /dev/null +++ b/dom/media/test/test_seek-5.html @@ -0,0 +1,69 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 5; + +function test_seek5(v, seekTime, is, ok, finish) { + +// Test for a seek, followed by a play before the seek completes, ensure we play at the end of the seek. +var startPassed = false; +var endPassed = false; +var completed = false; + +function startTest() { + if (completed) + return; + + v.currentTime=seekTime; +} + +function seekStarted() { + if (completed) + return; + ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime); + startPassed = true; + v.play(); +} + +function seekEnded() { + if (completed) + return; + endPassed = true; +} + +function playbackEnded() { + if (completed) + return; + ok(startPassed, "Got seeking event"); + ok(endPassed, "Got seeked event"); + completed = true; + finish(); +} + +v.addEventListener("ended", playbackEnded); +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeking", seekStarted); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-6.html b/dom/media/test/test_seek-6.html new file mode 100644 index 0000000000..32554c4c2f --- /dev/null +++ b/dom/media/test/test_seek-6.html @@ -0,0 +1,64 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 6; + +function test_seek6(v, seekTime, is, ok, finish) { + +// Test for bug identified by Chris Pearce in comment 40 on +// bug 449159. +var seekCount = 0; +var completed = false; +var interval; + +function poll() { + v.currentTime; +} + +function startTest() { + if (completed) + return; + interval = setInterval(poll, 10); + v.currentTime = Math.random() * v.duration; +} + +function seekEnded() { + if (completed) + return; + + seekCount++; + ok(true, "Seek " + seekCount); + if (seekCount == 3) { + clearInterval(interval); + completed = true; + finish(); + } else { + v.currentTime = Math.random() * v.duration; + } +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-7.html b/dom/media/test/test_seek-7.html new file mode 100644 index 0000000000..96139d6f83 --- /dev/null +++ b/dom/media/test/test_seek-7.html @@ -0,0 +1,59 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 7; + +function test_seek7(v, seekTime, is, ok, finish) { + +// If a NaN is passed to currentTime, make sure this is caught +// otherwise an infinite loop in the Ogg backend occurs. +var completed = false; +var thrown1 = false; +var thrown3 = false; + +function startTest() { + if (completed) + return; + + try { + v.currentTime = NaN; + } catch(e) { + thrown1 = true; + } + + try { + v.currentTime = Math.random; + } catch(e) { + thrown3 = true; + } + + completed = true; + ok(thrown1, "Setting currentTime to invalid value of NaN"); + ok(thrown3, "Setting currentTime to invalid value of a function"); + finish(); +} + +v.addEventListener("loadedmetadata", startTest); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-8.html b/dom/media/test/test_seek-8.html new file mode 100644 index 0000000000..2f6e390eef --- /dev/null +++ b/dom/media/test/test_seek-8.html @@ -0,0 +1,42 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 8; + +function test_seek8(v, seekTime, is, ok, finish) { + +function startTest() { + v.currentTime = 1000; +} + +function seekEnded() { + ok(Math.abs(v.currentTime - v.duration) < 0.2, + "currentTime " + v.currentTime + " close to " + v.duration); + finish(); +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek-9.html b/dom/media/test/test_seek-9.html new file mode 100644 index 0000000000..c03f6a75e3 --- /dev/null +++ b/dom/media/test/test_seek-9.html @@ -0,0 +1,41 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +// The data being used in these tests is specified in manifest.js. +// The functions to build the test array and to run a specific test are in +// seek_support.js. + +const SEEK_TEST_NUMBER = 9; + +function test_seek9(v, seekTime, is, ok, finish) { + +function startTest() { + v.currentTime = -1000; +} + +function seekEnded() { + is(v.currentTime, 0, "currentTime clamped to 0"); + finish(); +} + +v.addEventListener("loadedmetadata", startTest); +v.addEventListener("seeked", seekEnded); + +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seekLies.html b/dom/media/test/test_seekLies.html new file mode 100644 index 0000000000..f3141eaabd --- /dev/null +++ b/dom/media/test/test_seekLies.html @@ -0,0 +1,28 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: server lies about range requests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="text/javascript" src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body onunload="mediaTestCleanup();"> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +function on_metadataloaded() { + var v = document.getElementById('v'); + var d = Math.round(v.duration*1000); + ok(d == 4000, "Checking duration: " + d); + SimpleTest.finish(); +} + +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +<video id='v' + preload="metadata" + src='seekLies.sjs' + onloadedmetadata='on_metadataloaded();'></video> +</body> +</html> diff --git a/dom/media/test/test_seekToNextFrame.html b/dom/media/test/test_seekToNextFrame.html new file mode 100644 index 0000000000..755d06e622 --- /dev/null +++ b/dom/media/test/test_seekToNextFrame.html @@ -0,0 +1,95 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test seekToNextFrame of media files that should play OK</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var video = document.createElement('video'); + video.preload = "metadata"; + video.token = token; + video.seenSeeking = false; + video.seenEnded = false; + + var handler = { + "ontimeout": function() { + Log(token, "timed out: ended=" + video.seenEnded); + } + }; + manager.started(token, handler); + + video.src = test.name; + video.name = test.name; + + function callSeekToNextFrame() { + video.seekToNextFrame().then( + () => { + if (!video.seenSeeking) { + ok(false, video.token + ": Should have already received seeking event."); + } + video.seenSeeking = false; + if (!video.ended) { + callSeekToNextFrame(); + } + }, + () => { + ok(false, video.token + ": seekToNextFrame() failed."); + } + ); + } + + var onLoadedmetadata = function(t, v) { return function() { + callSeekToNextFrame(); + }}(test, video); + + var finish = function() { + video.finished = true; + video.removeEventListener("loadedmetadata", onLoadedmetadata); + video.removeEventListener("seeking", onSeeking); + removeNodeAndSource(video); + manager.finished(video.token); + } + + var onEnded = function(t, v) { return function() { + v.seenEnded = true; + finish(); + }}(test, video); + + var onSeeking = function(t, v) { return function() { + if (v.seenSeeking) { + ok(false, v.token + ": Should yet receive seeking event."); + } + v.seenSeeking = true; + }}(test, video); + + video.addEventListener("loadedmetadata", onLoadedmetadata); + video.addEventListener("seeking", onSeeking); + video.addEventListener("ended", onEnded); + + document.body.appendChild(video); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.seekToNextFrame.enabled", true ], + ["media.dormant-on-pause-timeout-ms", -1] + ] + }, + function() { + manager.runTests(gSeekToNextFrameTests, startTest); + }); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek_duration.html b/dom/media/test/test_seek_duration.html new file mode 100644 index 0000000000..11ee4cb534 --- /dev/null +++ b/dom/media/test/test_seek_duration.html @@ -0,0 +1,55 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="seek_support.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +/** + * This test is used to make sure video's duration won't be changed when it + * reachs to the end after seeking to position where the time is very close to + * video's end time. + */ + +SimpleTest.waitForExplicitFinish(); + +(async function startTest() +{ + const video = document.createElement('video'); + video.src = "bunny.webm"; + document.body.appendChild(video); + + const loadedMetadata = once(video, "loadedmetadata"); + const canplay = once(video, "canplay"); + const end = once(video, "ended"); + + info(`- wait for video loading metadata -`); + await loadedMetadata; + const originalDuration = video.duration; + + info(`- seek video to the position which is close to end time -`); + // video's duration is 2.1 and the last key frame is in 2.0, we want to seek + // to that keyframe. + video.currentTime = originalDuration - 0.1; + + info(`- play video until it ends -`); + await canplay; + await video.play(); + await end; + + ok(video.duration === originalDuration, `Duration shouldn't change`); + removeNodeAndSource(video); + + SimpleTest.finish(); +})(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek_negative.html b/dom/media/test/test_seek_negative.html new file mode 100644 index 0000000000..98f0b9c910 --- /dev/null +++ b/dom/media/test/test_seek_negative.html @@ -0,0 +1,77 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seeking to a negative time with readyState HAVE_NOTHING</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function startTest(test, token) { + var type = getMajorMimeType(test.type); + var v = document.createElement(type); + v.token = token; + manager.started(token); + + // Seek to negative start time. + v.currentTime = -123; + is(v.readyState, v.HAVE_NOTHING, "readyState is HAVE_NOTHING"); + ok(!v.seeking, "can't be seeking prior src defined"); + is(v.currentTime, -123, "currentTime is original seek time"); + + v.src = test.name; + + // Initialize running variables. + v._name = test.name; + v._seekStarted = false; + v._seekCompleted = false; + v._metadata = false; + + var events = [ "suspend", "play", "canplay", "canplaythrough", "loadstart", + "loadedmetadata", "loadeddata", "playing", "ended", "error", + "stalled", "emptied", "abort", "waiting", "pause" ]; + function logEvent(e) { + var video = e.target; + Log(e.target.token, "got " + e.type + " with currentTime = " + video.currentTime); + } + events.forEach(function(e) { + v.addEventListener(e, logEvent); + }); + + once(v, "seeking", function() { + v._seekStarted = true; + ok(v.currentTime >= 0, "currentTime should be positive"); + }); + once(v, "seeked", function() { + v._seekCompleted = true; + ok(v.currentTime >= 0, "currentTime should be positive"); + }); + once(v, "loadedmetadata", function() { + v._metadata = true; + ok(v.seeking, "element is seeking once readyState is HAVE_METADATA"); + ok(v.currentTime >= 0, "currentTime should be positive"); + }); + once(v, "ended", function() { + ok(v._seekStarted, "seek should have started"); + ok(v._seekCompleted, "seek should have completed"); + ok(v._metadata, "loadedmetadata fired"); + ok(v.currentTime >= 0, "currentTime should be positive"); + removeNodeAndSource(v); + manager.finished(v.token); + }); + + v.play(); +} + + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek_nosrc.html b/dom/media/test/test_seek_nosrc.html new file mode 100644 index 0000000000..56461007bd --- /dev/null +++ b/dom/media/test/test_seek_nosrc.html @@ -0,0 +1,58 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seek tests</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +var SEEK_TIME = 3.5; +var seekStarted = false; +var seekCompleted = false; +var metadata = false; + +var v = document.createElement('video'); +document.body.appendChild(v); +SimpleTest.registerCleanupFunction(function () { + v.remove(); +}); + +try { + v.currentTime = SEEK_TIME; +} catch (e) { + ok(false, "should not fire '" + e + "' event"); +} +is(v.readyState, v.HAVE_NOTHING, "readyState is HAVE_NOTHING"); +ok(!v.seeking, "can't be seeking prior src defined"); +is(v.currentTime, SEEK_TIME, "currentTime is default playback start position"); +once(v, "seeking", function() { + seekStarted = true; +}); +once(v, "seeked", function() { + seekCompleted = true; +}); +once(v, "loadedmetadata", function() { + metadata = true; + ok(v.seeking, "element is seeking once readyState is HAVE_METADATA"); +}); +once(v, "ended", function() { + ok(seekStarted, "seek should have started"); + ok(seekCompleted, "seek should have completed"); + ok(metadata, "loadedmetadata fired"); + ok(v.currentTime >= SEEK_TIME, "currentTime should be after seek time"); + SimpleTest.finish(); +}); + +v.src = "seek.webm"; +v.play(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek_out_of_range.html b/dom/media/test/test_seek_out_of_range.html new file mode 100644 index 0000000000..a477ebdd19 --- /dev/null +++ b/dom/media/test/test_seek_out_of_range.html @@ -0,0 +1,48 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: seeking off the end of a file</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +// Test if the ended event works correctly. + +async function initTest(test, token) { + var type = getMajorMimeType(test.type); + var v = document.createElement(type); + v.preload = "auto"; + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name; + document.body.appendChild(v); + + await once(v, "loadedmetadata"); + info(`${v._name}: seeking to the end of the media.`); + v.currentTime = 3.0 * v.duration; + // Wait for 'seeked' and 'ended' to be fired. + await Promise.all([once(v, "seeked"), once(v, "ended")]); + // Check currentTime is near the end of the media. + ok(Math.abs(v.duration - v.currentTime) < 0.1, + "Should be at end of media for " + v._name + " t=" + v.currentTime + " d=" + v.duration); + // Call play() to start playback from the beginning. + v.play(); + await once(v, "ended"); + ok(v.ended, "Checking ended set after seeking to EOF and playing for " + v._name); + removeNodeAndSource(v); + manager.finished(v.token); +} + +manager.runTests(gSmallTests, initTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_seek_promise_bug1344357.html b/dom/media/test/test_seek_promise_bug1344357.html new file mode 100644 index 0000000000..6c441bc1cf --- /dev/null +++ b/dom/media/test/test_seek_promise_bug1344357.html @@ -0,0 +1,35 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: bug 1344357</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> + +<script> + +// This test always succeeds at runtime but should not cause any leaks at the end of mochitests. + +let manager = new MediaTestManager; + +function initTest(test, token) { + manager.started(token); + + var win = window.open(); + var video = win.document.createElement("video"); + video.autoplay = true; + video.src = "http://example.com/tests/dom/media/test/" + test.name; + win.document.body.appendChild(video); + video.currentTime = test.duration / 2; + video.addEventListener("seeking", () => { + win.close(); + manager.finished(token); + }, true); +} + +manager.runTests(gSmallTests, initTest); + +</script>
\ No newline at end of file diff --git a/dom/media/test/test_seekable1.html b/dom/media/test/test_seekable1.html new file mode 100644 index 0000000000..45c84b35df --- /dev/null +++ b/dom/media/test/test_seekable1.html @@ -0,0 +1,66 @@ +<!DOCTYPE HTML> +<html> +<head> +<title>Test seekable member for media elements</title> +<script type="text/javascript" src="/MochiKit/MochiKit.js"></script> +<script src="/tests/SimpleTest/SimpleTest.js"></script> +<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +<script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id='test'> +<script class="testbody" type='application/javascript'> + +let manager = new MediaTestManager; + +function finish_test(element) { + if (element.parentNode) + element.remove(); + element.src=""; + manager.finished(element.token); +} + +var tests = [ +// Test using a finite media stream, and a server supporting range requests +{ +setup(element) { + is(element.seekable.length, 0, "seekable.length should be initialy 0."); + element.addEventListener("loadedmetadata", function() { + is(element.seekable.length, 1, "seekable.length should be 1 for a server supporting range requests."); + + is(element.seekable.start(0), 0.0, "The start of the first range should be the initialTime."); + is(element.seekable.end(0), element.duration, "The end of the first range should be the duration.") + finish_test(element); + }); + } +} +]; + +function createTestArray() { + var A = []; + for (var k=0; k < gProgressTests.length; k++) { + var t = {}; + t.setup = tests[0].setup; + t.name = gProgressTests[k].name; + t.type = gProgressTests[k].type; + A.push(t); + } + return A; +} + +function startTest(test, token) { + var elemType = getMajorMimeType(test.type); + var element = document.createElement(elemType); + element.preload = "auto"; + element.src = test.name; + element.token = token; + test.setup(element); + manager.started(token); +} + +manager.runTests(createTestArray(), startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_source.html b/dom/media/test/test_source.html new file mode 100644 index 0000000000..34cc9e7f8a --- /dev/null +++ b/dom/media/test/test_source.html @@ -0,0 +1,91 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: append source child</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<video id="v1"></video> +<audio id="a1"></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var v1 = document.getElementById("v1"); +var a1 = document.getElementById("a1"); +v1.preload = "auto"; +a1.preload = "auto"; + +is(v1.src, "", "src should be null"); +is(a1.src, "", "src should be null"); +is(v1.currentSrc, "", "currentSrc should be null"); +is(a1.currentSrc, "", "currentSrc should be null"); +is(v1.childNodes.length, 0, "should have no children"); +is(a1.childNodes.length, 0, "should have no children"); + +function newSource(filter) { + var candidates = gSmallTests.filter(function(x){return filter.test(x.type);}); + if (candidates.length > 0) { + var e = document.createElement("source"); + e.type = candidates[0].type; + e.src = candidates[0].name; + return e; + } + return null + +} + +var audioLoaded = false; +var videoLoaded = false; + +function loaded(e) { + var media = e.target; + ok(media.networkState > 0, "networkState should be > 0"); + is(media.childNodes.length, 1, "should have 1 child"); + var sourceFile = media.currentSrc.substring(media.currentSrc.lastIndexOf('/')+1); + var resource = media.firstChild.src.substring(media.firstChild.src.lastIndexOf('/')+1); + is(sourceFile, resource, "loaded wrong resource!"); + if (media == a1) + audioLoaded = true; + else if (media == v1) + videoLoaded = true; + if (audioLoaded && videoLoaded) { + SimpleTest.finish(); + } +} + +v1.addEventListener('loadeddata', loaded); +a1.addEventListener('loadeddata', loaded); + +var videoSource = newSource(/^video/); +if (videoSource) { + v1.appendChild(videoSource); + v1.load(); +} else { + // No video backends? Don't test anything. + videoLoaded = true; +} + +var audioSource = newSource(/^audio/); +if (audioSource) { + a1.appendChild(audioSource); + a1.load(); +} else { + audioLoaded = true; +} + +if (!audioLoaded && !videoLoaded) { + SimpleTest.waitForExplicitFinish(); +} else { + if (audioLoaded) { + todo(false, "No audio types supported"); + } + if (videoLoaded) { + todo(false, "No video types supported"); + } +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_source_null.html b/dom/media/test/test_source_null.html new file mode 100644 index 0000000000..485edafd2d --- /dev/null +++ b/dom/media/test/test_source_null.html @@ -0,0 +1,33 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=752087 +--> +<head> + <title>Test for Bug 752087</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=752087">Mozilla Bug 752087</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> + +<video id="v"> +<script> +var v = document.getElementById('v'); +v.src = null; // crashes on NULL access if not handled + +ok(true, "setting video.src to null didn't crash!"); +var srcPath = v.src.split('/'); +ok(srcPath.length >= 3, "src should be within dom/media/test"); +is(srcPath[srcPath.length - 1], "null", "Setting src to null is handled like 'null' string"); +</script> +</video> +</body> +</html> diff --git a/dom/media/test/test_source_write.html b/dom/media/test/test_source_write.html new file mode 100644 index 0000000000..29fffebd10 --- /dev/null +++ b/dom/media/test/test_source_write.html @@ -0,0 +1,40 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=462455 +--> +<head> + <title>Test for Bug 462455</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=462455">Mozilla Bug 462455</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> + +<video id="v"> +<script> +var loadStarted = false; +document.write('Pause parsing!'); +var v = document.getElementById('v'); + +var resource = getPlayableVideo(gSmallTests); +if (resource != null) { + var s = document.createElement("source"); + s.src = resource.name; + s.type = resource.type; + v.appendChild(s); +} + +ok(v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE, + "We shouldn't start network load until the current task returns."); +</script> +</video> +</body> +</html> diff --git a/dom/media/test/test_standalone.html b/dom/media/test/test_standalone.html new file mode 100644 index 0000000000..094052847f --- /dev/null +++ b/dom/media/test/test_standalone.html @@ -0,0 +1,61 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: standalone video documents</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="doTest()"> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var iframes = []; + +for (let i=0; i<gSmallTests.length; ++i) { + var test = gSmallTests[i]; + + // We can't play WAV files in stand alone documents, so just don't + // run the test on non-video content types. + var tag = getMajorMimeType(test.type); + if (tag != "video" || !document.createElement("video").canPlayType(test.type)) + continue; + + let f = document.createElement("iframe"); + f.src = test.name; + f._test = test; + f.id = "frame" + i; + iframes.push(f); + document.body.appendChild(f); +} + + +function filename(uri) { + return uri.substr(uri.lastIndexOf("/")+1); +} + +function doTest() +{ + for (let i=0; i<iframes.length; ++i) { + let f = document.getElementById(iframes[i].id); + var v = f.contentDocument.body.firstChild; + is(v.tagName.toLowerCase(), "video", "Is video element"); + var src = filename(v.currentSrc); + is(src, iframes[i]._test.name, "Name ("+src+") should match ("+iframes[i]._test.name+")"); + is(v.controls, true, "Controls set (" + src + ")"); + is(v.autoplay, true, "Autoplay set (" + src + ")"); + } + SimpleTest.finish(); +} + +if (iframes.length == 0) { + todo(false, "No types supported"); +} else { + SimpleTest.waitForExplicitFinish(); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_capture_origin.html b/dom/media/test/test_streams_capture_origin.html new file mode 100644 index 0000000000..13d5589c0d --- /dev/null +++ b/dom/media/test/test_streams_capture_origin.html @@ -0,0 +1,90 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for Bug 1189506</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1189506">Mozilla Bug 1189506</a> +<p id="display"></p> +<video id="vin"></video> +<video id="vout"></video> +<video id="vout_cors" crossorigin></video> +<canvas id="cin" width="40" height="30"></canvas> +<canvas id="cout" width="40" height="30"></canvas> +<canvas id="cout_cors" width="40" height="30"></canvas> +<div id="content" style="display: none"> +</div> +<pre id="test"> +<script type="application/javascript"> +/* global vin, vout, vout_cors, cin, cout, cout_cors */ + +/** Test for Bug 1189506 **/ + +SimpleTest.waitForExplicitFinish(); + +async function start() { + const resource = getPlayableVideo(gSmallTests).name; + vin.src = "http://example.org:8000/tests/dom/media/test/" + resource; + vin.preload = "metadata"; + + await new Promise(r => vin.onloadedmetadata = r); + vout.srcObject = vin.mozCaptureStreamUntilEnded(); + vout_cors.srcObject = vin.mozCaptureStreamUntilEnded(); + vin.play(); + vout.play(); + vout_cors.play(); + + await new Promise(r => vout.onended = r); + is(vin.ended, vout.ended, "Source media element ends first"); + + const ctxin = SpecialPowers.wrap(cin.getContext("2d")); + ctxin.drawImage(vin, 0, 0); + + { + info("Testing that the last frame is rendered"); + const powerCtx = SpecialPowers.wrap(cout.getContext("2d")); + powerCtx.drawImage(vout, 0, 0); + const datain = ctxin.getImageData(0, 0, cin.width, cin.height); + const dataout = powerCtx.getImageData(0, 0, cout.width, cout.height); + for (let i = 0; i < datain.data.length; i += 4) { + const pixelin = datain.data.slice(i, i + 4).join(','); + const pixelout = dataout.data.slice(i, i + 4).join(','); + if (pixelin != pixelout) { + is(pixelout, pixelin, `Pixel #${i/4} is rendered as expected`); + break; + } + } + is(datain.data.length / 4, cin.width * cin.height, + "Checked expected number of pixels"); + } + + { + info("Testing that the principal is set"); + const ctx = cout.getContext("2d"); + ctx.drawImage(vout, 0, 0); + SimpleTest.doesThrow(() => ctx.getImageData(0, 0, cout.width, cout.height), + "SecurityError"); + } + + { + info("Testing that the crossorigin attribute is ignored for MediaStreams"); + const ctx = cout_cors.getContext("2d"); + ctx.drawImage(vout_cors, 0, 0); + SimpleTest.doesThrow( + () => ctx.getImageData(0, 0, cout_cors.width, cout_cors.height), + "SecurityError"); + } +} + +(async () => { + try { await start(); } + catch(e) { ok(false, `Rejected with ${e}`); } + finally { SimpleTest.finish(); } +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_element_capture.html b/dom/media/test/test_streams_element_capture.html new file mode 100644 index 0000000000..2aa6deb587 --- /dev/null +++ b/dom/media/test/test_streams_element_capture.html @@ -0,0 +1,120 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that a MediaStream captured from one element plays back in another</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +const manager = new MediaTestManager(); + +function checkDrawImage(vout, msg) { + const canvas = document.createElement("canvas"); + const ctx = canvas.getContext("2d"); + ctx.drawImage(vout, 0, 0); + const imgData = ctx.getImageData(0, 0, 1, 1); + is(imgData.data[3], 255, msg); +} + +function isGreaterThanOrEqualEps(a, b, msg) { + ok(a >= b, `Got ${a}, expected at least ${b}; ${msg}`); +} + +async function startTest(test, token) { + manager.started(token); + const v = document.createElement('video'); + const vout = document.createElement('video'); + + v.token = token; + v.id = "MediaDecoder"; + vout.id = "MediaStream"; + + document.body.appendChild(vout); + + // Log events for debugging. + const events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(e) { + Log(token, `${e.target.id} got ${e.type}`); + } + for (const e of events) { + v.addEventListener(e, logEvent); + vout.addEventListener(e, logEvent); + }; + + v.src = test.name; + v.preload = 'metadata'; + await new Promise(r => v.onloadedmetadata = r); + + const stream = v.mozCaptureStreamUntilEnded(); + vout.srcObject = stream; + is(vout.srcObject, stream, + `${token} set output element .srcObject correctly`); + // Wait for the resource fetch algorithm to have run, so that the media + // element is hooked up to the MediaStream and ready to go. If we don't do + // this, we're not guaranteed to render the very first video frame, which + // can make this test fail the drawImage test when a video resource only + // contains one frame. + await new Promise(r => vout.onloadstart = r); + v.play(); + vout.play(); + + await Promise.race([ + Promise.all([ + new Promise(r => vout.onended = r), + new Promise(r => v.onended = r), + ]), + new Promise((res, rej) => vout.onerror = () => rej(new Error(vout.error.message))), + new Promise((res, rej) => v.onerror = () => rej(new Error(v.error.message))), + ]); + + let duration = test.duration; + if (typeof(test.contentDuration) == "number") { + duration = test.contentDuration; + } + if (duration) { + isGreaterThanOrEqualEps(vout.currentTime, duration, + `${token} current time at end`); + } + is(vout.readyState, vout.HAVE_CURRENT_DATA, + `${token} checking readyState`); + ok(vout.ended, `${token} checking playback has ended`); + isnot(stream.getTracks().length, 0, `${token} results in some tracks`); + if (stream.getVideoTracks().length > 0) { + ok(test.type.match(/^video/), `${token} is a video resource`); + checkDrawImage(vout, `${token} checking video frame pixel has been drawn`); + } + vout.remove(); + removeNodeAndSource(v); +} + +(async () => { + SimpleTest.requestCompleteLog(); + SimpleTest.waitForExplicitFinish(); + await SpecialPowers.pushPrefEnv( + { "set": [ + ["privacy.reduceTimerPrecision", false], + // This test exhibits bug 1543980 with RDD enabled. + ["media.rdd-process.enabled", false], + ]}); + let tests = gPlayTests; + // Filter out bug1377278.webm due to bug 1541401. + tests = tests.filter(t => !t.name.includes("1377278")); + + manager.runTests(tests, async (test, token) => { + try { + await startTest(test, token); + } catch(e) { + ok(false, `Caught exception for ${token}: ${e}`); + } + manager.finished(token); + }); +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_element_capture_mediatrack.html b/dom/media/test/test_streams_element_capture_mediatrack.html new file mode 100644 index 0000000000..6b4332d8af --- /dev/null +++ b/dom/media/test/test_streams_element_capture_mediatrack.html @@ -0,0 +1,100 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that a media element captureStream works when disabling MediaTracks</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +const manager = new MediaTestManager(); + +async function startTest(test, token) { + manager.started(token); + const v = document.createElement('video'); + + document.body.appendChild(v); + v.token = token; + v.id = "MediaDecoder"; + + // Log events for debugging. + const events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + function logEvent(e) { + Log(token, `${token}: ${e.target.id} got ${e.type}`); + } + for (const e of events) { + v.addEventListener(e, logEvent); + }; + + v.src = test.name; + v.preload = 'metadata'; + await new Promise(r => v.onloadedmetadata = r); + + const stream = v.mozCaptureStream(); + + is(stream.getAudioTracks().length, Math.min(1, v.audioTracks.length), + `${token}: Expected number of audio tracks`); + is(stream.getVideoTracks().length, Math.min(1, v.videoTracks.length), + `${token}: Expected number of video tracks`); + + if (v.audioTracks.length) { + v.audioTracks[0].enabled = false; + const track = stream.getAudioTracks()[0]; + await new Promise(r => track.onended = r); + is(track.readyState, "ended", `${token}: Audio track has ended on removal`); + await new Promise(r => stream.onremovetrack = r); + is(stream.getAudioTracks().length, 0, + `${token}: Audio track was removed on removetrack`); + } + + if (v.videoTracks.length) { + v.videoTracks[0].selected = false; + const track = stream.getVideoTracks()[0]; + await new Promise(r => track.onended = r); + is(track.readyState, "ended", `${token}: Video track has ended on removal`); + await new Promise(r => stream.onremovetrack = r); + is(stream.getVideoTracks().length, 0, + `${token}: Video track was removed on removetrack`); + } + + stream.onaddtrack = () => ok(false, "Unexpected addtrack event"); + + v.play(); + + await Promise.race([ + new Promise(r => v.ontimeupdate = r), + new Promise((res, rej) => v.onerror = () => rej(new Error(v.error.message))), + ]); + + is(stream.getTracks().length, 0, `${token}: no tracks appeared during playback`); + removeNodeAndSource(v); +} + +(async () => { + SimpleTest.requestCompleteLog(); + SimpleTest.waitForExplicitFinish(); + await SpecialPowers.pushPrefEnv( + { "set": [ + ["media.track.enabled", true], + ]}); + let tests = gPlayTests; + // Filter out bug1377278.webm due to bug 1541401. + tests = tests.filter(t => !t.name.includes("1377278")); + + manager.runTests(tests, async (test, token) => { + try { + await startTest(test, token); + } catch(e) { + ok(false, `Caught exception for ${token}: ${e}`); + } + manager.finished(token); + }); +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_element_capture_playback.html b/dom/media/test/test_streams_element_capture_playback.html new file mode 100644 index 0000000000..ce083069ef --- /dev/null +++ b/dom/media/test/test_streams_element_capture_playback.html @@ -0,0 +1,47 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that capturing a stream doesn't stop the underlying element from firing events</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<audio id="a"></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +var a = document.getElementById('a'); +var validTimeUpdate = false; + +function startTest() { + a.src = "big.wav"; + var context = new AudioContext(); + var node = context.createMediaElementSource(a); + node.connect(context.destination); + a.addEventListener("timeupdate", function() { + if (a.currentTime > 0.0 && a.currentTime < 5.0 && !validTimeUpdate) { + validTimeUpdate = true; + ok(true, "Received reasonable currentTime in a timeupdate"); + SimpleTest.finish(); + } + }); + a.addEventListener("ended", function() { + if (!validTimeUpdate) { + ok(false, "Received reasonable currentTime in a timeupdate"); + SimpleTest.finish(); + } + }); + a.play(); +} + +if (a.canPlayType("audio/wave")) { + startTest(); +} else { + todo(false, "No playable audio"); +} +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_element_capture_reset.html b/dom/media/test/test_streams_element_capture_reset.html new file mode 100644 index 0000000000..625b0fe23f --- /dev/null +++ b/dom/media/test/test_streams_element_capture_reset.html @@ -0,0 +1,174 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that reloading, pausing and seeking in a media element that's being captured behaves as expected</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script src="manifest.js"></script> +</head> +<body> +<video id="v"></video> +<video id="vout"></video> +<video id="vout_untilended"></video> +<pre id="test"> +<script> +const v = document.getElementById('v'); +const vout = document.getElementById('vout'); +const vout_untilended = document.getElementById('vout_untilended'); + +function dumpEvent(event) { + const video = event.target; + info( + `${video.name}:${video.id} GOT EVENT ${event.type} ` + + `currentTime=${video.currentTime} paused=${video.paused} ` + + `ended=${video.ended} readyState=${video.readyState}` + ); +} + +function unexpected(event) { + ok(false, `${event.type} event received on ${event.target.id} unexpectedly`); +}; + +const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"]; +for (const e of events) { + v.addEventListener(e, dumpEvent); + vout.addEventListener(e, dumpEvent); + vout_untilended.addEventListener(e, dumpEvent); +} + +function isWithinEps(a, b, msg) { + ok(Math.abs(a - b) < 0.01, + "Got " + a + ", expected " + b + "; " + msg); +} + +function isGreaterThanOrEqualEps(a, b, msg) { + ok(a >= b - 0.01, + "Got " + a + ", expected at least " + b + "; " + msg); +} + +async function startTest(test) { + const seekTime = test.duration/2; + const contentDuration = test.contentDuration ?? test.duration; + + v.src = test.name; + v.name = test.name; + vout.name = test.name; + vout_untilended.name = test.name; + v.preload = "metadata"; + await new Promise(r => v.onloadedmetadata = r); + + vout.srcObject = v.mozCaptureStream(); + vout.play(); + + vout_untilended.srcObject = v.mozCaptureStreamUntilEnded(); + vout_untilended.play(); + + for (const track of [ + ...vout.srcObject.getTracks(), + ...vout_untilended.srcObject.getTracks(), + ]) { + ok(track.muted, `${track.kind} track ${track.id} should be muted`); + } + + v.play(); + + await Promise.all([ + ...vout.srcObject.getTracks(), + ...vout_untilended.srcObject.getTracks() + ].map(t => new Promise(r => t.onunmute = r))); + + await new Promise(r => v.onended = r); + isGreaterThanOrEqualEps(v.currentTime, test.duration, + "checking v.currentTime at first 'ended' event"); + + await Promise.all([ + new Promise(r => vout.onended = r), + new Promise(r => vout_untilended.onended = r), + ]); + + isGreaterThanOrEqualEps(vout.currentTime, contentDuration, + "checking vout.currentTime at first 'ended' event"); + ok(vout.ended, "checking vout has actually ended"); + ok(vout_untilended.ended, "checking vout_untilended has actually ended"); + + vout_untilended.srcObject.onaddtrack = unexpected; + vout_untilended.onplaying = unexpected; + vout_untilended.onended = unexpected; + + const voutPreSeekCurrentTime = vout.currentTime; + v.currentTime = seekTime; + await new Promise(r => v.onseeked = r); + + is(v.currentTime, seekTime, "Finished seeking"); + is(vout.currentTime, voutPreSeekCurrentTime, + "checking vout.currentTime has not changed after seeking"); + + v.play(); + vout.play(); + + await new Promise(r => v.onended = r); + isGreaterThanOrEqualEps(v.currentTime, test.duration, + "checking v.currentTime at second 'ended' event"); + + await new Promise(r => vout.onended = r); + isGreaterThanOrEqualEps(vout.currentTime, + (test.duration - seekTime) + contentDuration, + "checking vout.currentTime after seeking and playing through again"); + + v.src = test.name + "?1"; + vout.play(); + await v.play(); + + isnot(vout.srcObject.getTracks().length, 0, "There are some output tracks"); + + vout.onended = unexpected; + vout.srcObject.onremovetrack = unexpected; + + v.pause(); + await Promise.all( + vout.srcObject.getTracks().map(t => new Promise(r => t.onmute = r)) + ); + + for (const track of vout.srcObject.getTracks()) { + track.onunmute = unexpected; + } + + v.currentTime = 0; + await new Promise(r => v.onseeked = r); + + v.play(); + await Promise.all( + vout.srcObject.getTracks().map(t => new Promise(r => t.onunmute = r)) + ); + + vout.srcObject.onremovetrack = null; + + await new Promise(r => v.onended = r); + isGreaterThanOrEqualEps(v.currentTime, test.duration, + "checking v.currentTime at third 'ended' event"); + + await new Promise(r => vout.onended = r); + isGreaterThanOrEqualEps(vout.currentTime, + (test.duration - seekTime) + contentDuration*2, + "checking vout.currentTime after seeking, playing through and reloading"); +} + +(async () => { + SimpleTest.waitForExplicitFinish(); + try { + const testVideo = getPlayableVideo(gSmallTests); + if (testVideo) { + await startTest(testVideo); + } else { + todo(false, "No playable video"); + } + } catch(e) { + ok(false, `Error: ${e}`); + } finally { + SimpleTest.finish(); + } +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_element_capture_twice.html b/dom/media/test/test_streams_element_capture_twice.html new file mode 100644 index 0000000000..0e30be1801 --- /dev/null +++ b/dom/media/test/test_streams_element_capture_twice.html @@ -0,0 +1,79 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that capturing a media element, then reloading and capturing again, works</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<video id="v"></video> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +const v = document.getElementById('v'); + +function dumpEvent(event) { + const video = event.target; + info(video.name + " GOT EVENT " + event.type + + " currentTime=" + video.currentTime + + " paused=" + video.paused + + " ended=" + video.ended + + " readyState=" + video.readyState); +} + +const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"]; +for (let i = 0; i < events.length; ++i) { + v.addEventListener(events[i], dumpEvent); +} + +async function startTest(src) { + v.preload = "metadata"; + v.src = src; + await new Promise(r => v.onloadedmetadata = r); + const s1 = v.mozCaptureStream(); + const tracks = s1.getTracks(); + is(tracks.length, 2, "Expected total tracks, s1, capture 1"); + is(s1.getAudioTracks().length, 1, "Expected audio tracks, s1, capture 1"); + is(s1.getVideoTracks().length, 1, "Expected video tracks, s1, capture 1"); + is(s1.getAudioTracks()[0].readyState, "live", "Live audio, s1, capture 1"); + is(s1.getVideoTracks()[0].readyState, "live", "Live video, s1, capture 1"); + + v.src = null; + for (let i = 0; i < tracks.length; ++i) { + await Promise.race(tracks.map(t => new Promise(r => t.onended = r))); + await new Promise(r => s1.onremovetrack = r); + } + is(s1.getTracks().length, 0, "Expected total tracks, s1, metadata 2"); + + v.src = src; + await new Promise(r => v.onloadedmetadata = r); + is(s1.getTracks().length, 2, "Expected total tracks, s1, metadata 2"); + is(s1.getAudioTracks().length, 1, "Expected audio tracks, s1, metadata 2"); + is(s1.getVideoTracks().length, 1, "Expected video tracks, s1, metadata 2"); + is(s1.getAudioTracks()[0].readyState, "live", "Live audio, s1, metadata 2"); + is(s1.getVideoTracks()[0].readyState, "live", "Live video, s1, metadata 2"); + + const s2 = v.mozCaptureStream(); + is(s1.getTracks().length, 2, "Expected total tracks remains, s1, capture 2"); + is(s2.getTracks().length, 2, "Expected total tracks, s2, capture 2"); + is(s2.getAudioTracks().length, 1, "Expected audio tracks, s2, capture 2"); + is(s2.getVideoTracks().length, 1, "Expected video tracks, s2, capture 2"); + is(s2.getAudioTracks()[0].readyState, "live", "Live audio, s2, capture 2"); + is(s2.getVideoTracks()[0].readyState, "live", "Live video, s2, capture 2"); +} + +(async function() { + try { + await startTest("short-video.ogv"); + } catch(e) { + ok(false, `Caught error: ${e}${e.stack ? '\n' + e.stack : ''}`); + } finally { + SimpleTest.finish(); + } +})(); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_firstframe.html b/dom/media/test/test_streams_firstframe.html new file mode 100644 index 0000000000..ea1dbd35c1 --- /dev/null +++ b/dom/media/test/test_streams_firstframe.html @@ -0,0 +1,67 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that a non-autoplaying, non-playing element with a MediaStream source triggers canplay and shows a first frame</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="manifest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +async function runTest() { + const canvas = document.createElement("canvas"); + canvas.getContext("2d"); + const helper = new CaptureStreamTestHelper2D(100, 100); + + const video = document.createElement("video"); + document.body.appendChild(video); + + video.srcObject = canvas.captureStream(); + helper.drawColor(canvas, helper.red); + + await Promise.race([ + new Promise(r => video.oncanplay = r), + new Promise(r => setTimeout(r, 30000)) + .then(() => Promise.reject(new Error("Canplay timeout"))), + ]); + + ok(true, "Got \"canplay\""); + is(video.readyState, video.HAVE_ENOUGH_DATA, "Expected readyState"); + ok(helper.isPixel(helper.getPixel(video), helper.red), + "First frame is rendered before playing"); + + helper.drawColor(canvas, helper.green); + await helper.pixelMustNotBecome(video, helper.green, { + time: 1000, + infoString: "Rendered first frame doesn't change on new frame from source" + }); + ok(helper.isPixel(helper.getPixel(video), helper.red), + "First frame is still rendered"); + + video.play(); + helper.drawColor(canvas, helper.blue); + await helper.pixelMustBecome(video, helper.blue, { + infoString: "New frame gets rendered when playing" + }); + + video.srcObject.getTracks().forEach(t => t.stop()); +} + +(async function() { + SimpleTest.waitForExplicitFinish(); + SimpleTest.requestFlakyTimeout("Explicit timeout reasons"); + try { + await runTest(); + } catch(e) { + ok(false, e); + } + SimpleTest.finish(); +})(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_gc.html b/dom/media/test/test_streams_gc.html new file mode 100644 index 0000000000..d2ba23042b --- /dev/null +++ b/dom/media/test/test_streams_gc.html @@ -0,0 +1,44 @@ +<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test garbage collection of captured stream (bug 806754)</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="doTest()">
+<audio id="a" preload="metadata"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+var a = document.getElementById('a');
+a.src = getPlayableAudio(gSmallTests).name;
+
+function forceGC() {
+ SpecialPowers.gc();
+ SpecialPowers.forceGC();
+ SpecialPowers.forceCC();
+}
+
+function doTest() {
+ a.mozCaptureStreamUntilEnded();
+
+ a.addEventListener("seeked", function() {
+ a.play();
+
+ a.addEventListener("play", function() {
+ a.addEventListener("ended", function() {
+ ok(true, "GC completed OK");
+ SimpleTest.finish();
+ });
+ });
+ });
+
+ a.currentTime = a.duration;
+ setTimeout(forceGC, 0);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_individual_pause.html b/dom/media/test/test_streams_individual_pause.html new file mode 100644 index 0000000000..c49b563d76 --- /dev/null +++ b/dom/media/test/test_streams_individual_pause.html @@ -0,0 +1,77 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test for bug 1073406. Pausing a video element should not pause another playing the same stream.</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> + <script type="text/javascript" src="gUM_support.js"></script> +</head> +<body> +<video id="video1" autoplay></video> +<video id="video2" autoplay></video> +<script class="testbody" type="text/javascript"> +function getVideoImagePixelData(v) { + const canvas = document.createElement("canvas"); + const ctx = canvas.getContext("2d"); + ctx.drawImage(v, 0, 0); + const imgData = ctx.getImageData(canvas.width/2, canvas.height/2, 1, 1).data; + return "r" + imgData[0] + + "g" + imgData[1] + + "b" + imgData[2] + + "a" + imgData[3]; +} + +async function startTest() { + // This test expects fake devices so that the video color will change + // over time, explicitly request fakes. + await pushGetUserMediaTestPrefs({fakeAudio: true, fakeVideo: true}); + const stream = await navigator.mediaDevices.getUserMedia({video: true}); + const video1 = document.getElementById('video1'); + const video2 = document.getElementById('video2'); + + video1.srcObject = stream; + video2.srcObject = stream; + + await new Promise(r => video1.onplaying = r); + video1.pause(); + await new Promise(r => video1.onpause = r); + + const v1PausedImageData = getVideoImagePixelData(video1); + const v2PausedImageData = getVideoImagePixelData(video2); + + while (getVideoImagePixelData(video2) == v2PausedImageData) { + info("video2 has not progressed. Waiting."); + await new Promise(r => video2.ontimeupdate = r); + } + + // Wait for a while to be sure video1 would have gotten a frame + // if it is playing. + for (let i = 3; i != 0; i--) { + await new Promise(r => video2.ontimeupdate = r); + } + info("video2 progressed OK"); + + isnot(video1.currentTime, video2.currentTime, + "v1 and v2 should not be at the same currentTime"); + is(getVideoImagePixelData(video1), v1PausedImageData, + "video1 video frame should not have updated since video1 paused"); + + for (const t of stream.getTracks()) { + t.stop(); + } +} + +SimpleTest.waitForExplicitFinish(); +(async function() { + try { + await startTest(); + } catch(error) { + ok(false, "getUserMedia should not fail, got " + error.name); + } finally { + SimpleTest.finish(); + } +})(); +</script> +</body> +</html> diff --git a/dom/media/test/test_streams_srcObject.html b/dom/media/test/test_streams_srcObject.html new file mode 100644 index 0000000000..8ea5797d14 --- /dev/null +++ b/dom/media/test/test_streams_srcObject.html @@ -0,0 +1,60 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test interactions of src and srcObject</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body onload="doTests()"> +<audio id="a1"></audio> +<audio id="a2"></audio> +<pre id="test"> +<script class="testbody" type="text/javascript"> +SimpleTest.waitForExplicitFinish(); + +var doTest = srcObject => new Promise(resolve => { + var a = document.getElementById('a1'); + a.src = getPlayableAudio(gSmallTests).name; + var b = new Audio(); + + var newSrc = a.src + "?2"; + b.src = newSrc; + is(b[srcObject], null, "Initial srcObject is null"); + var stream = a.mozCaptureStream(); + b[srcObject] = stream; + is(b[srcObject], stream, "Stream set correctly"); + try { + b[srcObject] = "invalid"; + ok(false, "Setting srcObject to an invalid value should throw."); + } catch (e) { + ok(e instanceof TypeError, "Exception should be a TypeError"); + } + is(b[srcObject], stream, "Stream not set to invalid value"); + is(b.src, newSrc, "src attribute not affected by setting srcObject"); + var step = 0; + b.addEventListener("loadedmetadata", function() { + if (step == 0) { + is(b.currentSrc, "", "currentSrc set to empty string while playing srcObject"); + b[srcObject] = null; + is(b[srcObject], null, "Stream set to null"); + // The resource selection algorithm will run again and choose b.src + } else if (step == 1) { + is(b.currentSrc, b.src, "currentSrc set to src now that srcObject is null"); + resolve(); + } + ++step; + }); + a.play(); + b.play(); +}); + +var doTests = () => doTest("srcObject") + .catch(e => ok(false, "Unexpected error: " + e)) + .then(() => SimpleTest.finish()) + .catch(e => ok(false, "Coding error: " + e)); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_streams_tracks.html b/dom/media/test/test_streams_tracks.html new file mode 100644 index 0000000000..845bc36ca3 --- /dev/null +++ b/dom/media/test/test_streams_tracks.html @@ -0,0 +1,66 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaStreamTrack interfaces</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +const manager = new MediaTestManager; + +function testTracks(tracks, hasTrack, kind, src) { + is(tracks.length, hasTrack ? 1 : 0, `Correct ${kind} track count for ${src}`); + for (const track of tracks) { + is(track.readyState, "live", `Track ${track.id} should still be live`); + is(track.kind, kind, `Correct track kind for track ${track.id} of ${src}`); + ok(/\{........-....-....-....-............\}/.test(track.id), + `id ${track.id} for ${track.kind} track of ${src} has correct form`); + } +} + +async function startTest(test, token) { + try { + info(`Starting test of ${test.name}`); + const element = document.createElement("video"); + + element.token = token; + manager.started(token); + + element.src = test.name; + element.test = test; + const stream = element.mozCaptureStreamUntilEnded(); + + element.play(); + + await new Promise(r => element.onloadedmetadata = r); + + testTracks(stream.getAudioTracks(), test.hasAudio, "audio", test.name); + testTracks(stream.getVideoTracks(), test.hasVideo, "video", test.name); + const tracks = stream.getTracks(); + + await new Promise(r => element.onended = r); + + for (let i = 0; i < tracks.length; ++i) { + await Promise.race( + tracks.map(t => new Promise(r => t.onended = r)) + ); + await new Promise(r => stream.onremovetrack = r); + } + + testTracks(stream.getAudioTracks(), false, "audio", test.name); + testTracks(stream.getVideoTracks(), false, "video", test.name); + } catch(e) { + ok(false, `Caught error: ${e}`); + } finally { + manager.finished(token); + } +} + +manager.runTests(gTrackTests, startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_suspend_media_by_inactive_docshell.html b/dom/media/test/test_suspend_media_by_inactive_docshell.html new file mode 100644 index 0000000000..7f819ca33b --- /dev/null +++ b/dom/media/test/test_suspend_media_by_inactive_docshell.html @@ -0,0 +1,67 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test suspending media by inactive docShell</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<video id="testVideo" src="gizmo.mp4" loop></video> +<script class="testbody" type="text/javascript"> +/** + * When calling `browser.suspendMediaWhenInactive`, it can set the docShell's + * corresponding flag that is used to suspend media when the docShell is + * inactive. This test is used to check if we can suspend/resume the media + * correctly when changing docShell's active state. + */ +async function startTest() { + const video = document.getElementById("testVideo"); + + info(`start video`); + await video.play(); + + info(`set docShell inactive which would suspend media`); + await setDocShellActive(false); + + info(`set docShell active which would resume media`); + await setDocShellActive(true); + + SimpleTest.finish(); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + {"set": [["media.testing-only-events", true]]}, startTest); + +/** + * The following are test helper functions. + */ +function mediaSuspendedStateShouldEqualTo(expected) { + const video = document.getElementById("testVideo"); + const result = SpecialPowers.wrap(video).isSuspendedByInactiveDocOrDocShell; + is(result, expected, `media's suspended state is correct`); +} + +function setDocShellActive(isActive) { + const win = SpecialPowers.wrap(window); + const docShell = win.docShell; + const browsingContext = win.browsingContext; + // This flag is used to prevent media from playing when docShell is inactive. + browsingContext.top.suspendMediaWhenInactive = true; + browsingContext.isActive = isActive; + // After updating `docshell.isActive`, it would suspend/resume media and we + // wait suspending/resuming finishing by listening to `MozMediaSuspendChanged` + return new Promise(r => { + docShell.chromeEventHandler.addEventListener("MozMediaSuspendChanged", + () => { + mediaSuspendedStateShouldEqualTo(!isActive); + r(); + }, {once : true} + ); + }); +} + +</script> +</body> +</html> diff --git a/dom/media/test/test_temporary_file_blob_video_plays.html b/dom/media/test/test_temporary_file_blob_video_plays.html new file mode 100644 index 0000000000..87f6b3c4e6 --- /dev/null +++ b/dom/media/test/test_temporary_file_blob_video_plays.html @@ -0,0 +1,75 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test MediaRecorder Recording canvas stream</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script src="/tests/dom/canvas/test/captureStream_common.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<div id="content"> +</div> +<script class="testbody" type="text/javascript"> + +function startTest() { + var canvas = document.createElement("canvas"); + canvas.width = canvas.height = 100; + document.getElementById("content").appendChild(canvas); + + var helper = new CaptureStreamTestHelper2D(100, 100); + helper.drawColor(canvas, helper.red); + + var stream = canvas.captureStream(0); + + var blob; + + let mediaRecorder = new MediaRecorder(stream); + is(mediaRecorder.stream, stream, + "Media recorder stream = canvas stream at the start of recording"); + + mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired"); + + mediaRecorder.onerror = () => ok(false, "Recording failed"); + + mediaRecorder.ondataavailable = ev => { + is(blob, undefined, "Should only get one dataavailable event"); + blob = ev.data; + }; + + mediaRecorder.onstart = () => { + info("Got 'start' event"); + // We just want one frame encoded, to see that the recorder produces something readable. + mediaRecorder.stop(); + }; + + mediaRecorder.onstop = () => { + info("Got 'stop' event"); + ok(blob, "Should have gotten a data blob"); + + var video = document.createElement("video"); + video.id = "recorded-video"; + video.src = URL.createObjectURL(blob); + video.play(); + video.onerror = err => { + ok(false, "Should be able to play the recording. Got error. code=" + video.error.code); + SimpleTest.finish(); + }; + document.getElementById("content").appendChild(video); + helper.pixelMustBecome(video, helper.red, { + threshold: 128, + infoString: "Should become red", + }).then(SimpleTest.finish); + }; + + mediaRecorder.start(); + is(mediaRecorder.state, "recording", "Media recorder should be recording"); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({set:[["media.recorder.max_memory", 1]]}, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_timeupdate_small_files.html b/dom/media/test/test_timeupdate_small_files.html new file mode 100644 index 0000000000..fca35e5b71 --- /dev/null +++ b/dom/media/test/test_timeupdate_small_files.html @@ -0,0 +1,88 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=495319 +--> + +<head> + <title>Bug 495319 - playing back small audio files should fire timeupdate</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495319">Mozilla Bug 495319</a> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function ended(e) { + var v = e.target; + ++v.counter.ended; + is(v.counter.ended, 1, v._name + " should see ended only once"); + ok(v.counter.timeupdate > 0, v._name + " should see at least one timeupdate: " + v.currentTime); + + // Rest event counters for we don't allow events after ended. + eventsToLog.forEach(function(event) { + v.counter[event] = 0; + }); + + // Finish the test after 500ms. We shouldn't receive any timeupdate events + // after the ended event, so this gives time for any pending timeupdate events + // to fire so we can ensure we don't regress behaviour. + setTimeout( + function() { + // Remove the event listeners before removing the video from the document. + // We should receive a timeupdate and pause event when we remove the element + // from the document (as the element is specified to behave as if pause() was + // invoked when it's removed from a document), and we don't want those + // confusing the test results. + v.removeEventListener("ended", ended); + eventsToLog.forEach(function(event) { + v.removeEventListener(event, logEvent); + }); + removeNodeAndSource(v); + manager.finished(v.token); + }, + 500); +} + +var eventsToLog = ["play", "canplay", "canplaythrough", "loadstart", "loadedmetadata", + "loadeddata", "playing", "timeupdate", "error", "stalled", "emptied", "abort", + "waiting", "pause"]; + +function logEvent(event) { + var v = event.target; + ++v.counter[event.type]; + if (v.counter.ended > 0) { + is(v.counter[event.type], 0, v._name + " got unexpected " + event.type + " after ended"); + } +} + +function startTest(test, token) { + var type = getMajorMimeType(test.type); + var v = document.createElement(type); + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name; + + // Keep how many events received for each event type. + v.counter = {}; + eventsToLog.forEach(function(e) { + v.addEventListener(e, logEvent); + v.counter[e] = 0; + }); + v.addEventListener("ended", ended); + v.counter.ended = 0; + document.body.appendChild(v); + v.play(); +} + +manager.runTests(gSmallTests, startTest); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_unseekable.html b/dom/media/test/test_unseekable.html new file mode 100644 index 0000000000..e6d69f09f4 --- /dev/null +++ b/dom/media/test/test_unseekable.html @@ -0,0 +1,101 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: unseekable</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +/* + +Test that unseekable media can't be seeked. We load a media that shouldn't +be seekable, and play through once. While playing through we repeatedly try +to seek and check that nothing happens when we do. We also verify that the +seekable ranges are empty. + +*/ + +var manager = new MediaTestManager; + +var onseeking = function(event) { + var v = event.target; + v.actuallySeeked = true; +}; + +var onseeked = function(event) { + var v = event.target; + v.actuallySeeked = true; +}; + +var ontimeupdate = function(event) { + var v = event.target; + + // Check that when we seek nothing happens. + var t = v.currentTime; + v.currentTime = v.currentTime /= 2; + ok(Math.abs(t - v.currentTime) < 0.01, "Current time shouldn't change when seeking in unseekable media: " + v.name); + + // Check that the seekable ranges are empty. + is(v.seekable.length, 0, "Should have no seekable ranges in unseekable media: " + v.name); +}; + +var onended = function(event) { + var v = event.target; + + // Remove the event listeners so that they can't run if there are any pending + // events. + v.removeEventListener("seeking", onseeking); + v.removeEventListener("seeked", onseeked); + v.removeEventListener("timeupdate", ontimeupdate); + v.removeEventListener("ended", onended); + + v.src = ""; + if (v.parentNode) { + v.remove(); + } + + // Verify that none of the seeks we did in timeupdate actually seeked. + ok(!v.actuallySeeked, "Should not be able to seek in unseekable media: " + v.name); + + manager.finished(v.token); +} + +function startTest(test, token) { + var v = document.createElement('video'); + manager.started(token); + v.name = test.name; + v.src = test.name; + v.token = token; + v.autoplay = "true"; + + v.actuallySeeked = false; + + v.addEventListener("seeking", onseeking); + v.addEventListener("seeked", onseeked); + v.addEventListener("timeupdate", ontimeupdate); + v.addEventListener("ended", onended); + + document.body.appendChild(v); +} + +function canPlay(candidates) { + var v = document.createElement("video"); + var resources = candidates.filter(function(x){return v.canPlayType(x.type);}); + return (resources.length > 0); +} + +if (canPlay(gUnseekableTests)) { + manager.runTests(gUnseekableTests, startTest); +} else { + todo(false, "No files of supported format to test"); +} + + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_videoDocumentTitle.html b/dom/media/test/test_videoDocumentTitle.html new file mode 100644 index 0000000000..dd52dba26c --- /dev/null +++ b/dom/media/test/test_videoDocumentTitle.html @@ -0,0 +1,57 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=463830 +--> +<head> + <title>Test for Bug 463830</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463830">Mozilla Bug 463830</a> +<p id="display"></p> +<iframe id="i"></iframe> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +<script type="application/javascript"> + +/** Test for Bug 463830 **/ + +var gTests = [ + { file: "320x240.ogv", title: "320x240.ogv" }, + { file: "bug461281.ogg", title: "bug461281.ogg" }, +]; + +var gTestNum = 0; + +addLoadEvent(runTest); + +var title; +var i = document.getElementById("i"); + +function runTest() { + if (gTestNum == gTests.length) { + SimpleTest.finish(); + return; + } + if (gTestNum == 0) { + i.addEventListener("load", function() { + is(i.contentDocument.title, title, "Doc title incorrect"); + setTimeout(runTest, 0); + }); + } + + title = gTests[gTestNum].title; + i.src = gTests[gTestNum].file; + gTestNum++; +} + +SimpleTest.waitForExplicitFinish(); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_videoPlaybackQuality_totalFrames.html b/dom/media/test/test_videoPlaybackQuality_totalFrames.html new file mode 100644 index 0000000000..1b69a3b64f --- /dev/null +++ b/dom/media/test/test_videoPlaybackQuality_totalFrames.html @@ -0,0 +1,46 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Count the tatol frames of a video</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +var startTest = function(test, token) { + manager.started(token); + var v = document.createElement('video'); + v.token = token; + document.body.appendChild(v); + v.src = test.name; + + function ended(event) { + var video = event.target; + is(video.getVideoPlaybackQuality().totalVideoFrames, test.totalFrameCount,test.name+ " totalFrames should match!"); + removeNodeAndSource(video); + manager.finished(video.token); + } + v.addEventListener("ended", ended); + v.play(); +}; + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv( + { + "set": [ + ["media.decoder.skip-to-next-key-frame.enabled", false], + ["media.av1.use-dav1d", true] + ] + }, + function() { + manager.runTests(getPlayableVideos(gFrameCountTests), startTest); + }); + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_video_dimensions.html b/dom/media/test/test_video_dimensions.html new file mode 100644 index 0000000000..4d9c2a185f --- /dev/null +++ b/dom/media/test/test_video_dimensions.html @@ -0,0 +1,96 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that a video element has set video dimensions on loadedmetadata</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +var manager = new MediaTestManager; + +var startTest = function(test, token) { + manager.started(token); + var v1 = document.createElement('video'); + var v2 = document.createElement('video'); + var vout = document.createElement('video'); + + // Avoid a race for hardware resources between v1 and v2 on platforms with + // a hardware decoder, like B2G. + v1.preload = 'none'; + v2.preload = 'none'; + + var numVideoElementsFinished = 0; + + var ondurationchange = function(ev) { + var v = ev.target; + info(v.testName + " got durationchange"); + v.durationchange = true; + }; + var onresize = function(ev) { + var v = ev.target; + info(v.testName + " got resize"); + ok(!v.resize, v.testName + " should only fire resize once for same size"); + v.resize = true; + ok(v.durationchange, v.testName + + " durationchange event should have been emitted before resize"); + is(v.videoWidth, test.width, v.testName + " width should be set on resize"); + is(v.videoHeight, test.height, v.testName + " height should be set on resize"); + }; + var onloadedmetadata = function(ev) { + var v = ev.target; + info(v.testName + " got loadedmetadata"); + ok(!v.loadedmetadata, v.testName + " should only fire loadedmetadata once"); + v.loadedmetadata = true; + ok(v.resize, v.testName + + " resize event should have been emitted before loadedmetadata"); + + numVideoElementsFinished += 1; + if (v === v1) { + removeNodeAndSource(v1); + v2.load(); + } + + if (v === v2) { + vout.srcObject = v2.mozCaptureStreamUntilEnded(); + v2.play(); + vout.play(); + } + + if (numVideoElementsFinished === 3) { + removeNodeAndSource(v2); + removeNodeAndSource(vout); + manager.finished(token); + } + }; + var setupElement = function(v, id) { + v.durationchange = false; + v.ondurationchange = ondurationchange; + v.resize = false; + v.onresize = onresize; + v.loadedmetadata = false; + v.onloadedmetadata = onloadedmetadata; + document.body.appendChild(v); + }; + + v1.testName = test.name; + v2.testName = test.name + " (Captured)"; + vout.testName = test.name + " (Stream)"; + + v1.src = test.name; + v2.src = test.name; + + setupElement(v1, "v1"); + setupElement(v2, "v2"); + setupElement(vout, "vout"); + + v1.play(); +}; + +manager.runTests(getPlayableVideos(gSmallTests), startTest); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_video_gzip_encoding.html b/dom/media/test/test_video_gzip_encoding.html new file mode 100644 index 0000000000..355a245713 --- /dev/null +++ b/dom/media/test/test_video_gzip_encoding.html @@ -0,0 +1,25 @@ +<!DOCTYPE HTML>
+<html>
+ <head>
+ <title>Bug 1370177 gzipped mp4 with Content-Length</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ </head>
+ <body>
+ <!--
+ Tests that an MP4 file served over a "Content-Encoding: gzip"
+ HTTP channel with a "Content-Length" header set to the length
+ of the compressed file works.
+ -->
+ <video id='v' src="http://mochi.test:8888/tests/dom/media/test/gzipped_mp4.sjs" controls autoplay></video>
+ <script>
+ SimpleTest.waitForExplicitFinish();
+ var v = document.getElementById('v');
+ v.addEventListener("ended", ()=>{
+ SimpleTest.finish();
+ mediaTestCleanup();
+ });
+ </script>
+ </body>
+</html>
diff --git a/dom/media/test/test_video_in_audio_element.html b/dom/media/test/test_video_in_audio_element.html new file mode 100644 index 0000000000..a53adf9414 --- /dev/null +++ b/dom/media/test/test_video_in_audio_element.html @@ -0,0 +1,68 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=1060896 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug 1060896</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> + <script type="text/javascript" src="manifest.js"></script> + <script type="application/javascript"> + + /** + * Test for Bug 1060896; tests that loading a video inside an audio element works. + **/ + + var manager = new MediaTestManager; + + function error(event) { + var a = event.target; + ok(!a.mozHasAudio, "Media must've had no active tracks to play"); + a.removeEventListener("error", error); + a.removeEventListener("ended", ended); + removeNodeAndSource(a); + manager.finished(a.token); + } + + function ended(event) { + var a = event.target; + a.removeEventListener("error", error); + a.removeEventListener("ended", ended); + removeNodeAndSource(a); + manager.finished(a.token); + } + + function initTest(test, token) { + var a = document.createElement('audio'); + a.token = token; + manager.started(token); + a.autoplay = true; + + a.addEventListener("error", error); + a.addEventListener("ended", ended); + + a.src = test.name; + } + + var videos = getPlayableVideos(gSmallTests); + // Bug 1216012, skip the test on emulator-kk. + if (getAndroidVersion() == 19) { + todo(false, "Test disabled on emulator-kk."); + } else { + manager.runTests(videos, initTest); + } + + </script> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1060896">Mozilla Bug 1060896</a> +<p id="display"></p> +<div id="content" style="display: none"> + +</div> +<pre id="test"> +</pre> +</body> +</html> diff --git a/dom/media/test/test_video_stats_resistfingerprinting.html b/dom/media/test/test_video_stats_resistfingerprinting.html new file mode 100644 index 0000000000..331e6a8101 --- /dev/null +++ b/dom/media/test/test_video_stats_resistfingerprinting.html @@ -0,0 +1,87 @@ +<!DOCTYPE HTML> +<html> +<!-- +Mozilla Bug: +https://bugzilla.mozilla.org/show_bug.cgi?id=1369309 +Tor Ticket: +https://trac.torproject.org/projects/tor/ticket/15757 +--> +<head> + <meta charset="utf-8"> + <title>Test for Bug 1369309</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <script type="application/javascript" src="manifest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/> +</head> +<body> +<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=682299">Mozilla Bug 1369309</a> +<a target="_blank" href="https://trac.torproject.org/projects/tor/ticket/15757">Tor Ticket 15757</a> + +<!-- The main testing script --> +<script class="testbody" type="text/javascript"> + var manager = new MediaTestManager; + const SPOOFED_FRAMES_PER_SECOND = 30; + const SPOOFED_DROPPED_RATIO = 0.05; + // Push the setting of 'privacy.resistFingerprinting' into gTestPrefs, which + // will be set during MediaTestManager.runTests(). + gTestPrefs.push( + ["privacy.resistFingerprinting", true], + ["privacy.resistFingerprinting.reduceTimerPrecision.microseconds", 100000], + ["privacy.resistFingerprinting.reduceTimerPrecision.jitter", false], + // We use 240p as the target resoultion since 480p is greater than every video + // source in our test suite, so we need to use 240p here for allowing us to + // test dropped rate here. + ["privacy.resistFingerprinting.target_video_res", 240] + ); + var testCases = [ + { name:"320x240.ogv", type:"video/ogg", width:320, height:240, duration:0.266, drop: false }, + { name:"seek.webm", type:"video/webm", width:320, height:240, duration:3.966, drop: false }, + { name:"gizmo.mp4", type:"video/mp4", width:560, height:320, duration:5.56, drop: true } + ]; + + function checkStats(v, shouldDrop) { + // Rounding the current time to 100ms. + let currentTime = Math.floor(v.currentTime * 10) / 10; + let dropRate = 0; + + if (shouldDrop) { + dropRate = SPOOFED_DROPPED_RATIO; + } + + is(v.mozParsedFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10), + "mozParsedFrames should be spoofed if fingerprinting resistance is enabled"); + is(v.mozDecodedFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10), + "mozDecodedFrames should be spoofed if fingerprinting resistance is enabled"); + is(v.mozPresentedFrames, + parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * (1 - dropRate), 10), + "mozPresentedFrames should be spoofed if fingerprinting resistance is enabled"); + is(v.mozPaintedFrames, + parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * (1 - dropRate), 10), + "mozPaintedFrames should be spoofed if fingerprinting resistance is enabled"); + is(v.mozFrameDelay, 0.0, + "mozFrameDelay should be 0.0 if fingerprinting resistance is enabled"); + let playbackQuality = v.getVideoPlaybackQuality(); + is(playbackQuality.totalVideoFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10), + "VideoPlaybackQuality.totalVideoFrames should be spoofed if fingerprinting resistance is enabled"); + is(playbackQuality.droppedVideoFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * dropRate, 10), + "VideoPlaybackQuality.droppedVideoFrames should be spoofed if fingerprinting resistance is enabled"); + } + + function startTest(test, token) { + let v = document.createElement("video"); + v.token = token; + v.src = test.name; + manager.started(token); + once(v, "ended", () => { + checkStats(v, test.drop); + removeNodeAndSource(v); + manager.finished(v.token); + }); + v.play(); + } + + manager.runTests(testCases, startTest); + +</script> +</body> +</html> diff --git a/dom/media/test/test_video_to_canvas.html b/dom/media/test/test_video_to_canvas.html new file mode 100644 index 0000000000..3267dc63f5 --- /dev/null +++ b/dom/media/test/test_video_to_canvas.html @@ -0,0 +1,68 @@ +<!DOCTYPE HTML> +<html> +<!-- +https://bugzilla.mozilla.org/show_bug.cgi?id=486646 +--> + +<head> + <title>Test for Bug 486646</title> + <script type="text/javascript" src="/MochiKit/MochiKit.js"></script> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> + <script type="text/javascript" src="manifest.js"></script> +</head> +<body> +<script class="testbody" type="text/javascript"> + +var manager = new MediaTestManager; + +function loaded(e) { + var v = e.target; + ok(v.readyState >= v.HAVE_CURRENT_DATA, + "readyState must be >= HAVE_CURRENT_DATA for " + v._name); + + var canvas = document.createElement("canvas"); + canvas.width = v.videoWidth; + canvas.height = v.videoHeight; + document.body.appendChild(canvas); + var ctx = canvas.getContext("2d"); + try { + ctx.drawImage(v, 0, 0); + ok(true, "Shouldn't throw exception while drawing to canvas from video for " + v._name); + } catch (ex) { + ok(false, "Shouldn't throw exception while drawing to canvas from video for " + v._name); + } + + v._finished = true; + v.remove(); + manager.finished(v.token); +} + +function startTest(test, token) { + var type = getMajorMimeType(test.type); + if (type != "video") + return; + + var v = document.createElement('video'); + v.token = token; + manager.started(token); + v.src = test.name; + v._name = test.name; + v._finished = false; + v.autoplay = true; + v.style.display = "none"; + v.addEventListener("ended", loaded); + document.body.appendChild(v); +} + +SimpleTest.waitForExplicitFinish(); +SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest); +function beginTest() { + manager.runTests(gSmallTests, startTest); +} + +</script> +</pre> + +</body> +</html> diff --git a/dom/media/test/test_volume.html b/dom/media/test/test_volume.html new file mode 100644 index 0000000000..3d9e5bd91a --- /dev/null +++ b/dom/media/test/test_volume.html @@ -0,0 +1,41 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Media test: volume attribute set</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> + +<video id='v1'></video><audio id='a1'></audio> + +<pre id="test"> +<script class="testbody" type="text/javascript"> + +function test(element, value, shouldThrow) { + var threw = null; + try { + element.volume = value; + } catch (ex) { + threw = ex.name; + } + is(shouldThrow, threw, "Case: " +element.id+ " setVolume=" + value); +} + + +var ids = [document.getElementById('v1'), document.getElementById('a1')]; + +for (let i=0; i<ids.length; i++) { + var element = ids[i]; + test(element, 0.0, null); + test(element, 1.0, null); + test(element, -0.1, "IndexSizeError"); + test(element, 1.1, "IndexSizeError"); + test(element, undefined, "TypeError"); + test(element, NaN, "TypeError"); +} + +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_vp9_superframes.html b/dom/media/test/test_vp9_superframes.html new file mode 100644 index 0000000000..c570561b31 --- /dev/null +++ b/dom/media/test/test_vp9_superframes.html @@ -0,0 +1,31 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Test that all VP9 frames are decoded (contains superframes)</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> + +SimpleTest.waitForExplicitFinish(); + +function test() { + var video = document.createElement("video"); + video.src = "vp9-superframes.webm"; + video.play(); + video.addEventListener("ended", function () { + let vpq = video.getVideoPlaybackQuality(); + is(vpq.totalVideoFrames, 120, "totalVideoFrames must contains 120 frames"); + SimpleTest.finish(); + }); +} + +addLoadEvent(function() { + test(); +}); +</script> +</pre> +</body> +</html> diff --git a/dom/media/test/test_wav_ended1.html b/dom/media/test/test_wav_ended1.html new file mode 100644 index 0000000000..99f6e38243 --- /dev/null +++ b/dom/media/test/test_wav_ended1.html @@ -0,0 +1,43 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Wave Media test: ended</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<pre id="test"> +<script class="testbody" type="text/javascript"> +// Test if the ended event works correctly. +var endPassed = false; +var completed = false; + +function startTest() { + if (completed) + return; + var v = document.getElementById('v'); + v.play(); +} + +function playbackEnded() { + if (completed) + return; + + var v = document.getElementById('v'); + completed = true; + ok(v.currentTime >= 0.9 && v.currentTime <= 1.1, + "Checking currentTime at end: " + v.currentTime); + ok(v.ended, "Checking playback has ended"); + SimpleTest.finish(); +} + +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +<audio id='v' + onloadedmetadata='return startTest();' + onended='return playbackEnded();'> + <source type='audio/x-wav' src='r11025_s16_c1.wav'> +</audio> +</body> +</html> diff --git a/dom/media/test/test_wav_ended2.html b/dom/media/test/test_wav_ended2.html new file mode 100644 index 0000000000..0c172139f0 --- /dev/null +++ b/dom/media/test/test_wav_ended2.html @@ -0,0 +1,62 @@ +<!DOCTYPE HTML> +<html> +<head> + <title>Wave Media test: ended and replaying</title> + <script src="/tests/SimpleTest/SimpleTest.js"></script> + <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" /> +</head> +<body> +<!-- try with autoplay and no v.play in starttest, also with both --> +<pre id="test"> +<script class="testbody" type="text/javascript"> +// Test if audio can be replayed after ended. +var completed = false; +var playingCount = 0; +var endCount = 0; + +function startTest() { + if (completed) + return; + + var v = document.getElementById('v'); + v.play(); +} + +function playbackStarted() { + if (completed) + return; + + playingCount++; +} + +function playbackEnded() { + if (completed) + return; + + endCount++; + var v = document.getElementById('v'); + ok(v.currentTime >= 0.9 && v.currentTime <= 1.1, + "Checking currentTime at end: " + v.currentTime); + ok(v.ended, "Checking playback has ended"); + ok(playingCount > 0, "Expect at least one playing event"); + playingCount = 0; + if (endCount < 2) { + v.play(); + } else { + ok(endCount == 2, "Check playback after ended event"); + completed = true; + SimpleTest.finish(); + } +} + +SimpleTest.waitForExplicitFinish(); +</script> +</pre> +<audio id='v' + onloadedmetadata='return startTest();' + onplaying='return playbackStarted();' + onended='return playbackEnded();'> + <source type='audio/x-wav' src='r11025_s16_c1.wav'> +</audio> +</body> +</html> diff --git a/dom/media/test/variable-channel.ogg b/dom/media/test/variable-channel.ogg Binary files differnew file mode 100644 index 0000000000..77e116889c --- /dev/null +++ b/dom/media/test/variable-channel.ogg diff --git a/dom/media/test/variable-channel.ogg^headers^ b/dom/media/test/variable-channel.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/variable-channel.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/variable-channel.opus b/dom/media/test/variable-channel.opus Binary files differnew file mode 100644 index 0000000000..76b8991167 --- /dev/null +++ b/dom/media/test/variable-channel.opus diff --git a/dom/media/test/variable-channel.opus^headers^ b/dom/media/test/variable-channel.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/variable-channel.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/variable-preskip.opus b/dom/media/test/variable-preskip.opus Binary files differnew file mode 100644 index 0000000000..a82e831ce6 --- /dev/null +++ b/dom/media/test/variable-preskip.opus diff --git a/dom/media/test/variable-preskip.opus^headers^ b/dom/media/test/variable-preskip.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/variable-preskip.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/variable-samplerate.ogg b/dom/media/test/variable-samplerate.ogg Binary files differnew file mode 100644 index 0000000000..0c2726e835 --- /dev/null +++ b/dom/media/test/variable-samplerate.ogg diff --git a/dom/media/test/variable-samplerate.ogg^headers^ b/dom/media/test/variable-samplerate.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/variable-samplerate.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/variable-samplerate.opus b/dom/media/test/variable-samplerate.opus Binary files differnew file mode 100644 index 0000000000..1d15170798 --- /dev/null +++ b/dom/media/test/variable-samplerate.opus diff --git a/dom/media/test/variable-samplerate.opus^headers^ b/dom/media/test/variable-samplerate.opus^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/variable-samplerate.opus^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vbr-head.mp3 b/dom/media/test/vbr-head.mp3 Binary files differnew file mode 100644 index 0000000000..35f4105491 --- /dev/null +++ b/dom/media/test/vbr-head.mp3 diff --git a/dom/media/test/vbr-head.mp3^headers^ b/dom/media/test/vbr-head.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vbr-head.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vbr.mp3 b/dom/media/test/vbr.mp3 Binary files differnew file mode 100644 index 0000000000..38eb376a97 --- /dev/null +++ b/dom/media/test/vbr.mp3 diff --git a/dom/media/test/vbr.mp3^headers^ b/dom/media/test/vbr.mp3^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vbr.mp3^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/very-short.mp3 b/dom/media/test/very-short.mp3 Binary files differnew file mode 100644 index 0000000000..1d15dcad59 --- /dev/null +++ b/dom/media/test/very-short.mp3 diff --git a/dom/media/test/video-overhang.ogg b/dom/media/test/video-overhang.ogg Binary files differnew file mode 100644 index 0000000000..e11b28fb5b --- /dev/null +++ b/dom/media/test/video-overhang.ogg diff --git a/dom/media/test/video-overhang.ogg^headers^ b/dom/media/test/video-overhang.ogg^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/video-overhang.ogg^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vp9-short.webm b/dom/media/test/vp9-short.webm Binary files differnew file mode 100644 index 0000000000..16d32abee3 --- /dev/null +++ b/dom/media/test/vp9-short.webm diff --git a/dom/media/test/vp9-short.webm^headers^ b/dom/media/test/vp9-short.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vp9-short.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vp9-superframes.webm b/dom/media/test/vp9-superframes.webm Binary files differnew file mode 100644 index 0000000000..d695e42357 --- /dev/null +++ b/dom/media/test/vp9-superframes.webm diff --git a/dom/media/test/vp9-superframes.webm^headers^ b/dom/media/test/vp9-superframes.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vp9-superframes.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vp9.webm b/dom/media/test/vp9.webm Binary files differnew file mode 100644 index 0000000000..221877e303 --- /dev/null +++ b/dom/media/test/vp9.webm diff --git a/dom/media/test/vp9.webm^headers^ b/dom/media/test/vp9.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vp9.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vp9cake-short.webm b/dom/media/test/vp9cake-short.webm Binary files differnew file mode 100644 index 0000000000..2d353d98a7 --- /dev/null +++ b/dom/media/test/vp9cake-short.webm diff --git a/dom/media/test/vp9cake-short.webm^headers^ b/dom/media/test/vp9cake-short.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vp9cake-short.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/vp9cake.webm b/dom/media/test/vp9cake.webm Binary files differnew file mode 100644 index 0000000000..4ea70ed302 --- /dev/null +++ b/dom/media/test/vp9cake.webm diff --git a/dom/media/test/vp9cake.webm^headers^ b/dom/media/test/vp9cake.webm^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/vp9cake.webm^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata.wav b/dom/media/test/wave_metadata.wav Binary files differnew file mode 100644 index 0000000000..5e17547c30 --- /dev/null +++ b/dom/media/test/wave_metadata.wav diff --git a/dom/media/test/wave_metadata.wav^headers^ b/dom/media/test/wave_metadata.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata_bad_len.wav b/dom/media/test/wave_metadata_bad_len.wav Binary files differnew file mode 100644 index 0000000000..b89c4818be --- /dev/null +++ b/dom/media/test/wave_metadata_bad_len.wav diff --git a/dom/media/test/wave_metadata_bad_len.wav^headers^ b/dom/media/test/wave_metadata_bad_len.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata_bad_len.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata_bad_no_null.wav b/dom/media/test/wave_metadata_bad_no_null.wav Binary files differnew file mode 100644 index 0000000000..18063048c3 --- /dev/null +++ b/dom/media/test/wave_metadata_bad_no_null.wav diff --git a/dom/media/test/wave_metadata_bad_no_null.wav^headers^ b/dom/media/test/wave_metadata_bad_no_null.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata_bad_no_null.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata_bad_utf8.wav b/dom/media/test/wave_metadata_bad_utf8.wav Binary files differnew file mode 100644 index 0000000000..b6f2a675b2 --- /dev/null +++ b/dom/media/test/wave_metadata_bad_utf8.wav diff --git a/dom/media/test/wave_metadata_bad_utf8.wav^headers^ b/dom/media/test/wave_metadata_bad_utf8.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata_bad_utf8.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata_unknown_tag.wav b/dom/media/test/wave_metadata_unknown_tag.wav Binary files differnew file mode 100644 index 0000000000..b19fb5170f --- /dev/null +++ b/dom/media/test/wave_metadata_unknown_tag.wav diff --git a/dom/media/test/wave_metadata_unknown_tag.wav^headers^ b/dom/media/test/wave_metadata_unknown_tag.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata_unknown_tag.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wave_metadata_utf8.wav b/dom/media/test/wave_metadata_utf8.wav Binary files differnew file mode 100644 index 0000000000..352db285bb --- /dev/null +++ b/dom/media/test/wave_metadata_utf8.wav diff --git a/dom/media/test/wave_metadata_utf8.wav^headers^ b/dom/media/test/wave_metadata_utf8.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wave_metadata_utf8.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_alaw.wav b/dom/media/test/wavedata_alaw.wav Binary files differnew file mode 100644 index 0000000000..ef090d16e0 --- /dev/null +++ b/dom/media/test/wavedata_alaw.wav diff --git a/dom/media/test/wavedata_alaw.wav^headers^ b/dom/media/test/wavedata_alaw.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_alaw.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_float.wav b/dom/media/test/wavedata_float.wav Binary files differnew file mode 100644 index 0000000000..155f891d51 --- /dev/null +++ b/dom/media/test/wavedata_float.wav diff --git a/dom/media/test/wavedata_float.wav^headers^ b/dom/media/test/wavedata_float.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_float.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_s16.wav b/dom/media/test/wavedata_s16.wav Binary files differnew file mode 100644 index 0000000000..6a69cd78f6 --- /dev/null +++ b/dom/media/test/wavedata_s16.wav diff --git a/dom/media/test/wavedata_s16.wav^headers^ b/dom/media/test/wavedata_s16.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_s16.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_s24.wav b/dom/media/test/wavedata_s24.wav Binary files differnew file mode 100644 index 0000000000..dbdb6aac1e --- /dev/null +++ b/dom/media/test/wavedata_s24.wav diff --git a/dom/media/test/wavedata_s24.wav^headers^ b/dom/media/test/wavedata_s24.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_s24.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_u8.wav b/dom/media/test/wavedata_u8.wav Binary files differnew file mode 100644 index 0000000000..1d895c2ce0 --- /dev/null +++ b/dom/media/test/wavedata_u8.wav diff --git a/dom/media/test/wavedata_u8.wav^headers^ b/dom/media/test/wavedata_u8.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_u8.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store diff --git a/dom/media/test/wavedata_ulaw.wav b/dom/media/test/wavedata_ulaw.wav Binary files differnew file mode 100644 index 0000000000..0874face21 --- /dev/null +++ b/dom/media/test/wavedata_ulaw.wav diff --git a/dom/media/test/wavedata_ulaw.wav^headers^ b/dom/media/test/wavedata_ulaw.wav^headers^ new file mode 100644 index 0000000000..4030ea1d3d --- /dev/null +++ b/dom/media/test/wavedata_ulaw.wav^headers^ @@ -0,0 +1 @@ +Cache-Control: no-store |