2019/functional/functionalTest.js (1,167 lines of code) (raw):
/**
* @license
* Copyright 2018 Google Inc. All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
'use strict';
/**
* Functional Test Suite.
* @class
*/
var FunctionalTest = function() {
const URL_PATH = 'lib/functional/src/';
const MEDIA_PATH = 'test-materials/media/';
const MAX_URL_LENGTH = 2047;
const TEST_MEDIA_SRC = {
Jpg: MEDIA_PATH + 'qual-e/pass.jpg',
Png: MEDIA_PATH + 'qual-e/pass.png',
Webp: MEDIA_PATH + 'qual-e/pass.webp',
SslTestVideoMp4: MEDIA_PATH + 'qual-e/sslTestVideo.mp4',
SslTestVideoWebm: MEDIA_PATH + 'qual-e/sslTestVideo.webm',
};
const MS_TO_WAIT_FOR_FONTS = 3000;
const COBALT_REQ_URL =
'https://cobalt.googlesource.com/cobalt/+/19.lts.stable/' +
'src/cobalt/build/build.id?format=TEXT';
var functionalVersion = 'Current Editor\'s Draft';
var webkitPrefix = MediaSource.prototype.version.indexOf('webkit') >= 0;
var tests = [];
var testArea = document.getElementById('testArea');
var info = 'No Media Support!';
if (window.MediaSource) {
info = 'Spec Version: ' + functionalVersion;
info += ' | webkit prefix: ' + webkitPrefix.toString();
}
info += ' | Default Timeout: ' + TestBase.timeout + 'ms';
var fields = ['passes', 'failures', 'timeouts'];
var createFunctionalTest = function(name, category, mandatory) {
var t = createTest(name);
t.prototype.index = tests.length;
t.prototype.passes = 0;
t.prototype.failures = 0;
t.prototype.timeouts = 0;
t.prototype.category = category || 'Functional';
if (typeof mandatory === 'boolean') {
t.prototype.mandatory = mandatory;
}
tests.push(t);
return t;
};
// Create and check if the element is supported by browser
var createElement = function(elementName) {
var element = document.createElement(elementName);
if (Object.prototype.toString.call(element)
=== '[object HTMLUnknownElement]') {
throw Error(elementName + ' not supported!');
}
return element;
};
/**
* Validate the existence of AudioContext and its properties.
*/
var createAudioContextPropertyTest = function(value, evaluate) {
var name = value ? 'AudioContext.' + value : 'AudioContext';
var test = createFunctionalTest(name,'AudioContext');
test.prototype.title = 'Test if' + name + ' exists';
test.prototype.start = function(runner) {
try {
var audioCtx = new (window.AudioContext || window.webkitAudioContext)();
if (!evaluate(audioCtx)) {
throw 'No ' + name + '.';
}
} catch (e) {
return runner.fail(e);
}
runner.succeed();
}
};
createAudioContextPropertyTest(null, function(audioCtx) {
return !!(audioCtx);
});
createAudioContextPropertyTest('destination', function(audioCtx) {
return !!(audioCtx.destination);
});
createAudioContextPropertyTest('sampleRate', function(audioCtx) {
return typeof audioCtx.sampleRate === 'number';
});
createAudioContextPropertyTest('currentTime', function(audioCtx) {
return isNaN(audioCtx.currentTime) == false;
});
createAudioContextPropertyTest('decodeAudioData', function(audioCtx) {
return !!(audioCtx.decodeAudioData);
});
createAudioContextPropertyTest('createBufferSource', function(audioCtx) {
return (!!audioCtx.createBufferSource &&
audioCtx.createBufferSource() instanceof AudioBufferSourceNode);
});
/**
* Validate the existence of AudioBufferSourceNode and its properties.
*/
var createAudioBufferSourceNodeTest = function(value, evaluate, mandatory) {
var name = value ? 'AudioBufferSourceNode.' + value : 'AudioBufferSourceNode';
var test = createFunctionalTest(name, 'AudioBufferSourceNode', mandatory);
test.prototype.title = 'Test if ' + name + ' exists.';
test.prototype.start = function(runner) {
try {
var audioCtx = new (window.AudioContext || window.webkitAudioContext)();
var source = audioCtx.createBufferSource();
if (!evaluate(source)) {
throw 'No ' + name + '.';
}
} catch (e) {
return runner.fail(e);
}
runner.succeed();
}
};
createAudioBufferSourceNodeTest(null, function(source) {
return !!source;
});
createAudioBufferSourceNodeTest('buffer', function(source) {
return !!(typeof source.buffer);
});
createAudioBufferSourceNodeTest('onended', function(source) {
return !!(typeof source.onended);
});
createAudioBufferSourceNodeTest(
'playbackRate',
function(source) {
return (typeof source.playbackRate.defaultValue === 'number') &&
(typeof source.playbackRate.value === 'number');
},
false);
createAudioBufferSourceNodeTest('start', function(source) {
return !!(typeof source.start);
});
createAudioBufferSourceNodeTest('stop', function(source) {
return !!(typeof source.stop);
});
/**
* Validate the existence of AudioNode and its properties.
*/
var createAudioNodeTest = function(value, evaluate) {
var name = value ? 'AudioNode.' + value : 'AudioNode';
var test = createFunctionalTest(name, 'AudioNode');
test.prototype.title = 'Test if ' + name + ' exists.';
test.prototype.start = function(runner) {
try {
var audioCtx = new (window.AudioContext || window.webkitAudioContext)();
var node = audioCtx.destination;
if (!evaluate(node, audioCtx)) {
throw 'No ' + name + '.';
}
} catch (e) {
return runner.fail(e);
}
runner.succeed();
}
};
createAudioNodeTest(null, function(node) {
return !!(node);
});
createAudioNodeTest('context', function(node, audioCtx) {
return node.context === audioCtx;
});
createAudioNodeTest('numberOfInputs', function(node) {
return typeof node.numberOfInputs === 'number';
});
createAudioNodeTest('numberOfOutputs', function(node) {
return typeof node.numberOfOutputs === 'number';
});
createAudioNodeTest('channelCountMode', function(node) {
return typeof node.channelCountMode === 'string';
});
createAudioNodeTest('channelInterpretation', function(node) {
return typeof node.channelInterpretation === 'string';
});
createAudioNodeTest('connect', function(node, audioCtx) {
return (!!(audioCtx.createBufferSource().connect));
});
createAudioNodeTest('disconnect', function(node, audioCtx) {
return (!!(audioCtx.createBufferSource().disconnect));
});
// expand url into MAX length
var expandUrl = function(url, charset) {
return url + new Array(MAX_URL_LENGTH - url.length - 1).join(charset);
}
/**
* Ensure super long navigation URL is properly supported.
*/
var testNavigationURLLength = createFunctionalTest(
'NavigationURLLength', 'URL Length', false);
testNavigationURLLength.prototype.title = 'Test Navigation URL Length support.';
testNavigationURLLength.prototype.start = function(runner) {
try {
var navElement = createElement('iframe');
var randChar =
'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'
.charAt(Math.floor(Math.random() * 62));
var url = expandUrl(URL_PATH + 'navUrlTest.html?q=', randChar);
navElement.setAttribute('src', url);
navElement.setAttribute('id', 'urlTestFrame' + randChar);
testArea.appendChild(navElement);
var frame = document.getElementById('urlTestFrame' + randChar);
frame.onload = function() {
var doc = frame.contentDocument ||
frame.contentWindow.document || frame.document;
var result = doc.getElementsByTagName('title')[0].innerHTML;
this.log('Result: ' + result)
testArea.removeChild(navElement);
if (result == 'Success') {
return runner.succeed();
}
throw Error(result);
}.bind(this);
} catch (e) {
return runner.fail(e);
}
};
/**
* Ensure super long image src URL is properly supported.
*/
var testImageSrcURLLength = createFunctionalTest(
'ImageSrcURLLength', 'URL Length');
testImageSrcURLLength.prototype.title = 'Test Image Src URL Length support.';
testImageSrcURLLength.prototype.start = function(runner) {
try {
var imgElement = createElement('img');
var url = expandUrl(TEST_MEDIA_SRC.Jpg + '?q=', 'a');
imgElement.setAttribute('src', url);
testArea.appendChild(imgElement);
imgElement.onload = function() {
testArea.removeChild(imgElement);
imgElement = null;
return runner.succeed();
}.bind(this);
} catch (e) {
return runner.fail(e);
}
};
/**
* Ensure super long video src URL is properly supported.
*/
var createVideoSrcURLTest = function(format, src) {
var test = createFunctionalTest(
'VideoSrcURLLength - ' + format, 'URL Length');
test.prototype.title = 'Test ' + format + ' Video Src URL Length support.';
test.prototype.start = function(runner, video) {
try {
var randChar =
'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'
.charAt(Math.floor(Math.random() * 62));
var longUrl = expandUrl(src + '?q=', randChar);
video.setAttribute('src', longUrl);
video.onplaying = function onPlaying() {
video.removeEventListener('onplaying', onPlaying);
return runner.succeed();
}.bind(this);
video.onerror = function() {
var message = String().concat(this.name + ' - value returned: ', false);
throw Error(message);
}.bind(this);
video.play();
} catch (e) {
return runner.fail(e);
}
}
};
createVideoSrcURLTest('mp4', TEST_MEDIA_SRC.SslTestVideoMp4);
/**
* Ensure super long XHR request URL is properly supported.
*/
var testXHRURLLength = createFunctionalTest('XHRURLLength', 'URL Length');
testXHRURLLength.prototype.title = 'Test XHR Src URL Length support.';
testXHRURLLength.prototype.start = function(runner) {
try {
var xmlHttp = new XMLHttpRequest();
var url = expandUrl(TEST_MEDIA_SRC.Png + '?q=', 'a');
xmlHttp.open('GET', url, true);
xmlHttp.send(null);
xmlHttp.onreadystatechange = function() {
if (xmlHttp.readyState == 4) {
if (xmlHttp.status == 200) {
return runner.succeed();
} else {
runner.fail(Error('Xhr request status is not 200 OK'));
}
}
};
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate the existence of CORS through Modernizr.
*/
var testCORS = createFunctionalTest('CORS', 'Security');
testCORS.prototype.title = 'Test if CORS works.';
testCORS.prototype.start = function(runner) {
try {
if (window.Modernizr.cors) {
return runner.succeed();
}
throw Error('CORS is not supported.');
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate the existence of localStorage properties.
*/
var createLocalStorageTest = function(value, evaluate, cleanup) {
var name = value ? 'localStorage.' + value : 'localStorage';
var test = createFunctionalTest(name, 'localStorage');
test.prototype.title = 'Test if ' + name + ' works.';
test.prototype.start = function(runner) {
try {
if (evaluate()) {
return runner.succeed();
}
throw name + ' errors';
} catch (e) {
if (cleanup) {
cleanup();
}
return runner.fail(e);
}
}
};
createLocalStorageTest(null, function() {
return Modernizr.localstorage;
});
createLocalStorageTest('setItem', function() {
return Modernizr.prefixed('setItem', window.localStorage , false) != false;
});
createLocalStorageTest('getItem', function() {
return Modernizr.prefixed('getItem', window.localStorage , false) != false;
});
createLocalStorageTest('removeItem', function() {
return Modernizr.prefixed('removeItem', window.localStorage , false) != false;
});
createLocalStorageTest(
'1M Char Quota',
function() {
var testData = new Array(1000 * 1000).join('2');
localStorage.setItem('yt-quale', testData);
localStorage.removeItem('yt-quale');
return true;
},
function() {
localStorage.removeItem('yt-quale');
});
/**
* Test if HTTPS onload fires.
*/
var testHTTPS = createFunctionalTest('HTTPS', 'HTTP');
testHTTPS.prototype.title = 'Test if HTTPS onload fires.';
testHTTPS.prototype.start = function(runner) {
try {
var sslImg = createElement('img');
sslImg.setAttribute(
'src', 'https://www.google.com/images/srpr/logo3w.png');
testArea.appendChild(sslImg);
sslImg.onload = function() {
testArea.removeChild(sslImg);
sslImg = null;
return runner.succeed();
}.bind(this);
} catch (e) {
return runner.fail(e);
}
};
/**
* Test if XMLHTTPRequest works.
*/
var testXMLHTTPRequest = createFunctionalTest('XMLHTTPRequest', 'HTTP');
testXMLHTTPRequest.prototype.title = 'Test if XMLHTTPRequest works.';
testXMLHTTPRequest.prototype.start = function(runner) {
try {
if (new XMLHttpRequest()) {
return runner.succeed();
}
} catch (e) {
return runner.fail(e);
}
};
/**
* Test if XMLHTTPRequest Level 2 works.
*/
var testXMLHTTPRequestLevel2 = createFunctionalTest(
'XMLHTTPRequest Level 2', 'HTTP');
testXMLHTTPRequestLevel2.prototype.title =
'Test if XMLHTTPRequest Level 2 works.';
testXMLHTTPRequestLevel2.prototype.start = function(runner) {
try {
// test from https://github.com/NielsLeenheer/html5test/pull/100
if (window.XMLHttpRequest && ('upload' in new XMLHttpRequest())) {
return runner.succeed();
}
throw Error('Does not support XMLHTTPRequest Level 2');
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate Fetch API properties work as expect.
*/
var createFetchAPITest = function(value, evaluate) {
var name = 'Fetch API - ' + value;
var test = createFunctionalTest(name, 'Fetch API');
test.prototype.title = 'Test if ' + value + ' in Fetch API works.';
test.prototype.start = function(runner) {
try {
if (evaluate()) {
return runner.succeed();
}
throw Error(name + ' errors');
} catch (e) {
return runner.fail(e);
}
}
};
createFetchAPITest('fetch()', function() {
return !!(window.fetch);
});
createFetchAPITest('Headers', function() {
var httpHeaders = {
'Content-Type' : 'image/jpeg',
'Accept-Charset' : 'utf-8'
};
var testHeaders = new Headers(httpHeaders);
return testHeaders.get('Content-Type') == "image/jpeg";
});
/**
* Validate Fetch API Request work as expect.
*/
var testFetchAPIRequest= createFunctionalTest(
'Fetch API - Request', 'Fetch API');
testFetchAPIRequest.prototype.title = 'Test if Fetch API Request works.';
testFetchAPIRequest.prototype.start = function(runner) {
try {
var fetchRequest = new Request(
'https://qual-e.appspot.com/test',
{method: 'POST', body: '{"result":"pass"}'});
// This shouldn't work, as method is read-only
fetchRequest.method = 'GET';
} catch (e) {
if (fetchRequest.url == 'https://qual-e.appspot.com/test' &&
fetchRequest.method == 'POST' && fetchRequest.credentials != 'include')
return runner.succeed();
}
return runner.fail('Fetch API Request should be read-only.')
};
/**
* Validate Fetch API Response work as expect.
*/
var testFetchAPIResponse= createFunctionalTest(
'Fetch API - Response', 'Fetch API');
testFetchAPIResponse.prototype.title = 'Test if Fetch API Response works.';
testFetchAPIResponse.prototype.start = function(runner) {
try {
var init = {'status': 200, 'statusText': 'YouTube'};
var myResponse = new Response('test_body', init);
myResponse.statusText = 'Should not work since it is read-only';
} catch (e) {
if (myResponse.statusText == 'YouTube' && myResponse.ok) {
return runner.succeed();
}
}
return runner.fail('Fetch API does not work as expected');
};
/**
* Test fetch API's streaming feature.
*/
var testFetchStreamAPIRequest = createFunctionalTest(
'Fetch API - stream', 'Fetch API');
testFetchStreamAPIRequest.prototype.title =
'Test if Fetch API handles steaming. ';
testFetchStreamAPIRequest.prototype.start = function(runner){
try {
const FETCH_TIME_OUT = 3000;
const timeout = setTimeout(function(){
throw Error('Fetch took too long');
}, FETCH_TIME_OUT);
fetch('lib/functional/fetch-sample.txt').then(function(response) {
var reader = response.body.getReader();
var partialCell = '';
var returnNextCell = false;
var returnCellAfter = "YouTube";
function search() {
return reader.read().then(function(result) {
var chunk = result.value;
for (var i = 3; i < chunk.byteLength; i++) {
partialCell += String.fromCharCode(chunk[i]);
}
var cellBoundry = /(?:,|\r\n)/;
var completeCells = partialCell.split(cellBoundry);
if (!result.done) {
// last cell may not be complete
partialCell = completeCells[completeCells.length - 1];
completeCells = completeCells.slice(0, -1);
}
for (var cell of completeCells) {
cell = cell.trim();
if (returnNextCell) {
reader.cancel("No more reading needed.");
return cell;
}
if (cell === returnCellAfter) {
returnNextCell = true;
}
}
if (result.done) {
throw Error("Could not find value after " + returnCellAfter);
}
return search();
});
}
return search();
}).then(function(){
clearTimeout(timeout);
return runner.succeed();
});
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate Streams API - ReadableByteStream.
*/
var testStreamsAPI= createFunctionalTest(
'Streams API - ReadableByteStream', 'Assorted');
testStreamsAPI.prototype.title = 'Test Streams API - ReadableByteStream.';
testStreamsAPI.prototype.start = function(runner) {
try {
var init = {'status': 200, 'statusText' : 'YouTube'};
var myResponse = new Response('test_body', init);
if (!!window.ReadableStream && myResponse.ok) {
return runner.succeed();
}
throw Error('Response not ok');
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate browser supports strict mode in ECMA262-5.
*/
var testECMA262= createFunctionalTest('ECMA262-5 Strict Mode', 'Assorted');
testECMA262.prototype.title = 'Test browser supports strict mode in ECMA262-5.';
testECMA262.prototype.start = function(runner) {
try {
'use strict';
var a = 1;
return runner.succeed();
} catch (e) {
return runner.fail(e)
}
};
/**
* Validate the existence of window.requestAnimationFrame through Modernizr.
*/
var testAnimationFrame= createFunctionalTest(
'RequestAnimationFrame', 'Assorted');
testAnimationFrame.prototype.title = 'Test request Animation Frame.';
testAnimationFrame.prototype.start = function(runner) {
try {
if (Modernizr.prefixed('requestAnimationFrame', window, false)) {
return runner.succeed();
}
throw Error('window.requestAnimationFrame does not exist');
} catch (e) {
return runner.fail(e)
}
};
/**
* Test if js date object is correctly supported.
*/
var testJSDateObject= createFunctionalTest(
'JavaScript Date Object', 'Assorted');
testJSDateObject.prototype.title = 'Test if js date object is supported.';
testJSDateObject.prototype.start = function(runner) {
try {
var dateObj = new Date(Date.parse('2012-11-01T14:12:09.000Z'));
var dateObjFormatted = dateObj.getUTCDate() + '/' +
(dateObj.getUTCMonth() + 1) + '/' +
dateObj.getUTCFullYear().toString().substr(2,2);
if (dateObjFormatted == '1/11/12' || dateObjFormatted == '2012/11/2') {
return runner.succeed();
}
throw Error('date object is not supported correctly');
} catch (e) {
return runner.fail(e)
}
};
/**
* Validate support for specified image format.
*/
var createImgTest = function(suffix) {
var test = createFunctionalTest(suffix.toUpperCase(), 'Assorted');
test.prototype.title = 'Test image format ' + suffix;
test.prototype.start = function(runner) {
try {
var imgElement = createElement('img');
var url = TEST_MEDIA_SRC[suffix];
imgElement.setAttribute('src', url);
testArea.appendChild(imgElement);
imgElement.onload = function() {
testArea.removeChild(imgElement);
imgElement = null;
return runner.succeed();
}.bind(this);
} catch (e) {
return runner.fail(e);
}
}
};
createImgTest('Jpg');
createImgTest('Png');
/**
* Validate the window height and width are in right size.
*/
var testWindowSize = createFunctionalTest('Window Size', 'Assorted');
testWindowSize.prototype.title = 'Test window size in current browser.';
testWindowSize.prototype.start = function(runner) {
try {
var result = false;
switch (window.innerHeight) {
case 720:
if(window.innerWidth == 1280)
result = true;
break;
case 1080:
if(window.innerWidth == 1920)
result = true;
break;
case 1440:
if(window.innerWidth == 2560)
result = true;
break;
case 2160:
if(window.innerWidth == 3840)
result = true;
break;
}
if (!result) {
throw 'Failed test with window size in ' +
window.innerWidth + ' x ' + window.innerHeight;
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate existence of WebP properties.
*/
var createWebPTest = function(name, uri) {
var test = createFunctionalTest(name, 'WebP');
test.prototype.title = 'Test ' + name + ' support.';
test.prototype.start = function(runner) {
var img = new Image();
img.addEventListener('load', function() {
return runner.succeed();
}, false);
img.addEventListener('error', function(event) {
return runner.fail('Request to website with valid SSL certificate fails.');
}, false);
img.src = uri;
};
};
var webpTests = [{
'uri': '',
'name': 'WebP'
}, {
'uri': '',
'name': 'WebP Alpha'
}, {
'uri': '',
'name': 'WebP Animation'
}, {
'uri': '',
'name': 'WebP Lossles'
}];
for (var i = 0; i < webpTests.length; i++) {
createWebPTest(webpTests[i].name, webpTests[i].uri);
}
/**
* Validate specified video format could be played.
*/
var createMediaFormatTest = function(name, stream, codec, mandatory) {
var test = createFunctionalTest(name, 'Video / Audio', mandatory);
test.prototype.title = 'Test if browser can play ' + name;
test.prototype.start = function(runner, video) {
try {
if (!!(MediaSource.isTypeSupported &&
MediaSource.isTypeSupported(
stream + '; codecs="' + codec + '"'))) {
return runner.succeed();
}
throw name + ' not supported.';
} catch (e) {
return runner.fail(e);
}
};
};
createMediaFormatTest('MP4 + H.264', 'video/mp4', 'avc1.4d401e');
createMediaFormatTest('WebM + VP9', 'video/webm', 'vp9');
createMediaFormatTest('WebM + VP9 Profile 2', 'video/webm', 'vp9.2');
createMediaFormatTest('WebM + Opus', 'audio/webm', 'opus', false);
/**
* Ensure WebGL is correctly supported.
*/
var testWebGLSupport = createFunctionalTest(
'WebGL Support Detected', 'WebGL', harnessConfig.support_webgl);
testWebGLSupport.prototype.title = 'Test if WebGL is correctly supported';
testWebGLSupport.prototype.start = function(runner) {
try {
if (window.WebGLRenderingContext) {
var canvas = createElement('canvas');
var ctx = canvas.getContext('webgl');
if (!!(ctx)) {
return runner.succeed();
}
}
throw 'WebGL not correctly supported.'
} catch (e) {
return runner.fail(e);
}
};
/**
* Ensure WebGL is properly disabled if not supported.
*/
var testWebGLDisabled = createFunctionalTest(
'WebGL Properly Disabled', 'WebGL', !harnessConfig.support_webgl);
testWebGLDisabled.prototype.title = 'Test if WebGL is properly disabled';
testWebGLDisabled.prototype.start = function(runner) {
try {
if (window.WebGLRenderingContext) {
var canvas = createElement('canvas');
var ctx = canvas.getContext('webgl');
if (ctx) {
throw 'WebGL not correctly disbabled.'
}
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
}
};
/**
* Test WebSpeech API through Modernizr.
*/
var testWebSpeechAPI = createFunctionalTest(
'WebSpeech API', 'Assorted', harnessConfig.support_webspeech);
testWebSpeechAPI.prototype.title = 'Test WebSpeech API.';
testWebSpeechAPI.prototype.start = function(runner) {
try {
if (Modernizr.speechrecognition) {
return runner.succeed();
}
throw 'WebSpeech not supported.';
} catch (e) {
return runner.fail(e);
}
};
/**
* Ensure video/x-flv is unsupported.
*/
var testNoVideoXFlv = createFunctionalTest(
'video/x-flv is unsupported', 'Assorted');
testNoVideoXFlv.prototype.title = 'Make sure video/x-flv is unsupported.';
testNoVideoXFlv.prototype.start = function(runner) {
try {
if (!Modernizr.video.flv) {
return runner.succeed();
}
throw 'video/x-flv is still supported.'
} catch (e) {
return runner.fail(e);
}
};
/**
* Validate MediaSource properties through Modernizr.
*/
var createMediaSourceTest = function(arg) {
var test = createFunctionalTest('MediaSource.' + arg, 'MediaSource');
test.prototype.title = 'Test if MediaSource.' + arg + ' is supported.';
test.prototype.start = function(runner, video) {
try {
var media = new MediaSource();
window.attachMediaSource(video, media);
if (Modernizr.prefixed(arg, media, false) != false) {
return runner.succeed();
}
throw 'MediaSource.' + arg + ' support failed.';
} catch (e) {
return runner.fail(e);
}
};
}
createMediaSourceTest('addSourceBuffer');
createMediaSourceTest('sourceBuffers');
createMediaSourceTest('activeSourceBuffers');
createMediaSourceTest('duration');
createMediaSourceTest('removeSourceBuffer');
createMediaSourceTest('readyState');
createMediaSourceTest('endOfStream');
/**
* Test if specified type is supported through videoElement.isTypeSupported.
*/
var createEMETest = function(name, evaluate, mandatory) {
var test = createFunctionalTest(name, 'EME Basic', mandatory);
test.prototype.title = 'Test ' + name;
test.prototype.start = function(runner, video) {
try {
if (evaluate(video)) {
return runner.succeed();
}
throw name + ' failed.';
} catch (e) {
return runner.fail(e);
}
}
};
createEMETest(
'canPlayType.keySystem',
function(videoElement) {
return videoElement.canPlayType(
util.createVideoFormatStr('mp4', 'avc1.42E01E'),
'org.w3.clearkey') != undefined ||
videoElement.canPlayType(
util.createVideoFormatStr('webm', 'vp9'),
'org.w3.clearkey') != undefined;
});
var widevineTest = function(videoElement) {
return videoElement.canPlayType(
util.createVideoFormatStr('mp4', 'avc1.42E01E'),
'com.widevine.alpha') != undefined ||
videoElement.canPlayType(
util.createVideoFormatStr('webm', 'vp9'),
'com.widevine.alpha') != undefined;
};
createEMETest('Widevine', widevineTest);
var playreadyTest = function(videoElement) {
return videoElement.canPlayType(
util.createVideoFormatStr('mp4', 'avc1.42E01E'),
'com.youtube.playready') != undefined;
};
createEMETest('PlayReady', playreadyTest, false);
createEMETest('DRM', function(videoElement) {
return widevineTest || playreadyTest;
});
createEMETest(
'Video.generateKeyRequest',
function(videoElement) {
return Modernizr.prefixed(
'generateKeyRequest', videoElement, false) != false;
},
false);
/**
* Creates a MediaDevices test that validates whether the device has
* Audio input media device and it implements correctly.
*/
var createMediaDevicesTest = function() {
var name = 'EnumerateDevices';
var test = createFunctionalTest(name, name, harnessConfig.support_webspeech);
test.prototype.title = 'Test if ' + name + ' exists';
test.prototype.start = function(runner) {
try {
if (undefined == navigator.mediaDevices.enumerateDevices) {
throw Error('enumerateDevices does not exists!');
}
navigator.mediaDevices.enumerateDevices().then(function(listOfDevices) {
for (var device of listOfDevices) {
if (undefined == device.kind) {
throw Error('kind does not exists!');
}
if (undefined == device.label) {
throw Error('label does not exists!');
}
if (device.kind == "audioinput") {
return runner.succeed();
}
}
throw Error('No audioinput found!');
});
} catch (e) {
return runner.fail(e);
}
};
};
createMediaDevicesTest();
/**
* Create a UAString test that validates the format of current User Agent.
*/
var createUAStringTest = function() {
var name = 'User Agent';
var test = createFunctionalTest(name, name);
test.prototype.title = 'Test if ' + name + ' format is correct';
test.prototype.start = function(runner) {
try{
var re_2019 = new RegExp(
[
'([-.A-Za-z0-9\\\\\\/]*)_([-.A-Za-z0-9\\\\\\/]*)_([-.A-Za-z0-9\\',
'\\\\/]*)_2019 ?\\/ ?[-_.A-Za-z0-9\\\\]* \\(([-_.A',
'-Za-z0-9\\\\\\/ ]+), ?([^,]*), ?([WIREDLSwiredls\\\\\\/]*)\\)'
].join(''));
var useragentParsed = re_2019.exec(navigator.userAgent);
if (useragentParsed &&
useragentParsed[2].match(/^(BDP|GAME|OTT|STB|TV|ATV)$/)) {
return runner.succeed();
}
throw Error('invalid User agent format');
} catch (e) {
return runner.fail(e);
}
};
};
createUAStringTest();
/**
* Create a UAString test that validates version of certain items in user agent.
*/
var createCobaltVersionTest = function(regexp, version, name) {
var title = 'User Agent';
var test = createFunctionalTest(name + ' ' + title, title);
test.prototype.title = 'Test if ' + name + title + ' format is correct';
test.prototype.start = function(runner) {
try {
var useragentParsed = regexp.exec(navigator.userAgent);
if (name === 'Cobalt') {
var xhr = new XMLHttpRequest();
xhr.open('GET', COBALT_REQ_URL);
xhr.send();
xhr.onreadystatechange = function() {
if (xhr.readyState === 4) {
if (xhr.status === 200) {
try {
var buildVersion = Number(window.atob(xhr.responseText));
} catch (e) {
runner.fail(e);
}
if (useragentParsed && Number(useragentParsed[1].split('.')[1]) === buildVersion) {
return runner.succeed();
}
runner.fail(Error(name + ' version is not ' + buildVersion));
} else {
runner.fail(Error('Failed to request Cobalt build version'));
}
}
};
} else {
if (useragentParsed && Number(useragentParsed[1]) >= version) {
return runner.succeed();
}
throw Error(name + ' version is less than ' + version);
}
} catch (e) {
return runner.fail(e);
}
};
};
createCobaltVersionTest(new RegExp(/19\.lts\.([0-9]\.[0-9]{6})/), 0, 'Cobalt');
createCobaltVersionTest(new RegExp(/Starboard\/([0-9]+)/), 10, 'Starboard');
/**
* Create a CPU memory test that validates the size of memory allocation.
*/
var createMemoryTest = function(processor, size) {
var title = processor + ' Memory Allocation';
var test = createFunctionalTest(title, title);
test.prototype.title = 'Test if ' + title + ' is correct';
test.prototype.start = function(runner) {
try{
if (typeof h5vcc != 'undefined' && typeof h5vcc.cVal != 'undefined') {
var memFree = Number(h5vcc.cVal.getValue('Memory.' + processor +
'.Free'));
var memUsed = Number(h5vcc.cVal.getValue('Memory.' + processor +
'.Used'));
if (memFree + memUsed < size) {
throw Error(processor + ' memory allocated less than ' + size);
}
} else {
var a = new Array(size - 50 * 1024 * 1024).join('M');
var f = function(arr){};
f(a);
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
}
};
};
createMemoryTest('CPU', 150 * 1024 * 1024);
/**
* Ensure Same-origin Policy is enabled through accessing localStorage.
*/
var testSameOriginPolicy = createFunctionalTest(
'Same-origin Policy', 'Assorted');
testSameOriginPolicy.prototype.title = 'Test Same-origin Policy.';
testSameOriginPolicy.prototype.start = function(runner) {
var value = localStorage.getItem('yt.leanback::schema-version');
runner.log('[Same-origin Policy] - localStorage.getItem(' +
'"yt.leanback::schema-version"); returned: ' + value);
runner.checkEq(
value, null, 'localStorage.getItem("yt.leanback::schema-version")');
return runner.succeed();
};
var createFontDiv = function(format, content) {
var span = createElement('span');
span.className = 'dancing-script-' + format;
span.innerHTML = content;
return span;
};
/**
* Validate specified font format is supported by browser.
* Assume ttf is supported.
*/
var createFontTest = function(format) {
var test = createFunctionalTest('Fonts - ' + format, 'Fonts', false);
test.prototype.title = 'Test if font format ' + format + ' is supported.';
test.prototype.start = function(runner) {
try {
testArea.removeAttribute('style');
var fontDiv = createElement('div');
fontDiv.className = 'font-block';
testArea.appendChild(fontDiv);
var characters =
'ABCĆČDĐEFGHIJKLMNOPQRSŠTUVWXYZŽabcčćdđefghijklmnopqrsštuvwxyzž' +
'ĂÂÊÔƠƯăâêôơư1234567890‘?’“!”(%)[#]{@}/&\<-+÷×=>®©$€£¥¢:;,.*';
var ttfEle = createFontDiv('ttf', characters)
var testEle = createFontDiv(format, characters)
fontDiv.appendChild(ttfEle);
fontDiv.appendChild(testEle);
// Cobalt does not emit any event when fonts load finish.
// So use setTimeout as a workaround.
setTimeout(function(event) {
var ttfWidth = ttfEle.offsetWidth;
var testWidth = testEle.offsetWidth;
testArea.removeChild(fontDiv);
testArea.setAttribute('style', 'display: none');
if (ttfWidth !== testWidth) {
throw 'Widths don\'t match: ttf:' + ttfWidth + ' and ' +
format + ':' + testWidth;
}
return runner.succeed();
}, MS_TO_WAIT_FOR_FONTS);
} catch (e) {
return runner.fail(e);
}
};
};
createFontTest('woff');
createFontTest('woff2');
var setCookie = function(key, value, path, domain, expires, httponly) {
var payload = key + '=' + value;
payload += '; path=' + (path ? path : '/');
payload += '; domain=' + (domain ? domain : '.ytlr-cert.appspot.com');
if (expires) {
payload += '; expires=' + expires;
}
if (httponly) {
payload += '; httponly';
}
document.cookie = payload;
}
var setCookieObj = function(cookie) {
setCookie(
cookie['key'],
cookie['value'],
cookie['path'],
cookie['domain'],
cookie['expires'],
cookie['httponly']);
};
var getCookieObject = function() {
var arr = document.cookie.split('; ');
var result = {};
for (var i = 0; i < arr.length; i++) {
var kv = arr[i].split('=');
if (kv.length > 1) {
result[kv[0]] = kv[1];
}
}
return result;
};
// Clear cookie by marking it as expired.
var clearCookie = function(cookie) {
setCookie(
cookie['key'],
cookie['value'],
cookie['path'],
cookie['domain'],
'Thu, 01-Jan-1970 00:00:00 GMT',
cookie['httponly']);
};
var clearTestCookies = function() {
for (var i in valid_cookies) {
clearCookie(valid_cookies[i]);
}
for (var i in invalid_cookies) {
clearCookie(invalid_cookies[i]);
}
};
var getCookiePath = function() {
return (testVersion != 'Test-In-Progress') ? testVersion : 'tip';
};
// 'valid_path_cookie' and 'valid_domain_cookie' would fail if you download
// the source code and setup your own server
var valid_cookies = [
{key: 'valid_cookie', value: 'valid'},
{key: 'valid_path_cookie', value: 'valid', path: getCookiePath()},
{
key: 'valid_domain_cookie',
value: 'valid',
domain: '.ytlr-cert.appspot.com'
}
];
var invalid_cookies = [
{key: 'invalid_http_only', value: 'invalid', httponly: true},
{key: 'invalid_path_cookie', value: 'invalid', path: '/p/a/t/h'},
{
key: 'invalid_domain_cookie',
value: 'invalid',
domain: '.invalid-domain.com'
},
{
key: 'invalid_expired_cookie',
value: 'invalid',
expires: 'Thu, 01-Jan-1970 00:00:00 GMT'
}
];
/**
* Ensure browser could set cookie and save properly.
*/
var testSetCookie = createFunctionalTest('Set Cookie', 'Cookie');
testSetCookie.prototype.title = 'Test if browser could save cookie.'
testSetCookie.prototype.start = function(runner) {
try {
for (var i in valid_cookies) {
setCookieObj(valid_cookies[i]);
}
for (var i in invalid_cookies) {
setCookieObj(invalid_cookies[i]);
}
var keys = Object.keys(getCookieObject());
for (var i in valid_cookies) {
var k = valid_cookies[i].key;
if (!keys.includes(k)) {
throw 'Key ' + k + ' should exist in document.cookie but it does not.';
}
}
for (var i in invalid_cookies) {
var k = invalid_cookies[i].key;
if (keys.includes(k)) {
throw 'Key ' + k + ' should not exist in document.cookie but it does.';
}
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
} finally {
clearTestCookies();
}
};
/**
* Ensure browser could set cookie and save properly.
*/
var testSetExpiredCookie = createFunctionalTest('Set Expired Cookie', 'Cookie');
testSetExpiredCookie.prototype.title = 'Test if browser ignores expired cookie.'
testSetExpiredCookie.prototype.start = function(runner) {
try {
for (var i in valid_cookies) {
setCookieObj(valid_cookies[i]);
}
var keys = Object.keys(getCookieObject());
for (var i in valid_cookies) {
var k = valid_cookies[i].key;
if (!keys.includes(k)) {
throw 'Key ' + k + ' should exist in document.cookie but it does not.';
}
}
for (var i in valid_cookies) {
clearCookie(valid_cookies[i]);
}
keys = Object.keys(getCookieObject());
for (var i in valid_cookies) {
var k = valid_cookies[i].key;
if (keys.includes(k)) {
throw 'Key ' + k + ' should not exist in document.cookie but it does.';
}
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
} finally {
clearTestCookies();
}
};
/**
* Validate certain number of cookies is supported by browser.
*/
var createCookieStorageTest = function(name, count) {
var test = createFunctionalTest(name, 'Cookie');
test.prototype.title = 'Test if browser supports ' + count + '4kb cookies.';
test.prototype.start = function(runner) {
var cookie_size = 4000;
try {
var cookie_storage = document.cookie.length;
for (var i = 0; i < count; i++) {
var prefix = 'CookieStorageTest' + ('000' + i).substr(-3) + '=';
var payload =
prefix + new Array(cookie_size - prefix.length + 1).join('m');
document.cookie = payload;
if (document.cookie.length - cookie_storage < cookie_size)
throw 'Fail to add the No.' + i + ' cookie';
else
cookie_storage = document.cookie.length;
}
return runner.succeed();
} catch (e) {
return runner.fail(e);
} finally {
document.cookie.split(';').forEach(function(c) {
if (c.trim().startsWith('CookieStorageTest')) {
document.cookie = c.replace(/^ +/, '').replace(/=.*/, '=;expires=' +
new Date().toUTCString());
}
});
}
};
};
createCookieStorageTest('Min Cookie Storage', 50);
createCookieStorageTest('Max Cookie Storage', 150);
/**
* Validate browser could correctly handle request from server with valid and
* invalid certificates.
*/
var createSslTest = function(name, origin, shouldFail = false) {
var test = createFunctionalTest(name, 'SSL');
test.prototype.title =
'Test if browser correctly handle to request to ' + origin;
test.prototype.start = function(runner) {
var url = origin + 'test/dashboard/small-image.png';
var img = new Image();
img.addEventListener('load', function() {
if (shouldFail) {
return runner.fail(
'Request to website with invalid SSL certificate succeeds.' +
'Invalid SSL requests must not be allowed to load.');
}
return runner.succeed();
}, false);
img.addEventListener('error', function(event) {
if (shouldFail) {
return runner.succeed();
}
return runner.fail('Request to website with valid SSL certificate fails.');
}, false);
img.src = url;
};
};
createSslTest('Self-Signed', 'https://self-signed.badssl.com/', true);
createSslTest('expired', 'https://expired.badssl.com/', true);
createSslTest('sha256', 'https://sha256.badssl.com/');
createSslTest('TLS', 'https://tls-v1-2.badssl.com:1012/');
var testGlobalSignR2 = createFunctionalTest('GlobalSign RootCA R2', 'SSL');
testGlobalSignR2.prototype.title = 'Test if browser correctly handle to' +
'request with GlobalSign RootCA R2 certificate.';
testGlobalSignR2.prototype.start = function(runner) {
try {
var xhr = new XMLHttpRequest();
xhr.open('GET', 'https://cert-test.sandbox.google.com/', true);
xhr.send();
xhr.onreadystatechange = function() {
if (xhr.readyState == 4) {
if (xhr.status != 200) {
runner.fail(Error('Xhr request status is not 200 ok'));
}
if (!xhr.responseText.includes('Client test successful.')) {
runner.fail(
Error('XHR Response not as expected: ' + xhr.responseText));
}
return runner.succeed();
}
};
} catch (e) {
return runner.fail(e);
}
};
var testGlobalSignR3 = createFunctionalTest('GlobalSign RootCA R3', 'SSL');
testGlobalSignR3.prototype.title = 'Test if browser correctly handle to' +
'request with GlobalSign RootCA R3 certificate.';
testGlobalSignR3.prototype.start = function(runner) {
try {
var url =
"https://www.globalsign.com/files/4114/9494/4842/switch-to-secure-ssl-globalsign.jpg"
var img = new Image();
img.addEventListener('load', function() {
return runner.succeed();
}, false);
img.addEventListener('error', function(event) {
runner.fail(Error('Request to website with valid SSL certificate fails.'));
}, false);
img.src = url;
} catch (e) {
return runner.fail(e);
}
};
return {tests: tests, info: info, fields: fields, viewType: 'default'};
};